mirror of
https://github.com/HChaZZY/PiliPlus.git
synced 2025-12-06 09:13:48 +08:00
feat: 评论区添加表情
This commit is contained in:
@@ -120,6 +120,11 @@ class Api {
|
|||||||
// https://api.bilibili.com/x/relation/stat?vmid=697166795
|
// https://api.bilibili.com/x/relation/stat?vmid=697166795
|
||||||
static const String userStat = '/x/relation/stat';
|
static const String userStat = '/x/relation/stat';
|
||||||
|
|
||||||
|
// 获取我的表情列表
|
||||||
|
// business:reply(回复)dynamic(动态)
|
||||||
|
//https://api.bilibili.com/x/emote/user/panel/web?business=reply
|
||||||
|
static const String myEmote = '/x/emote/user/panel/web';
|
||||||
|
|
||||||
// 获取用户信息
|
// 获取用户信息
|
||||||
static const String userInfo = '/x/web-interface/nav';
|
static const String userInfo = '/x/web-interface/nav';
|
||||||
|
|
||||||
|
|||||||
@@ -100,4 +100,24 @@ class ReplyHttp {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Future getMyEmote({
|
||||||
|
required String business,
|
||||||
|
}) async {
|
||||||
|
var res = await Request().get(Api.myEmote, data: {
|
||||||
|
'business': business,
|
||||||
|
});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {
|
||||||
|
'status': true,
|
||||||
|
'data': res.data['data'],
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'date': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
265
lib/models/user/my_emote.dart
Normal file
265
lib/models/user/my_emote.dart
Normal file
@@ -0,0 +1,265 @@
|
|||||||
|
class MyEmote {
|
||||||
|
Setting? setting;
|
||||||
|
List<Packages>? packages;
|
||||||
|
|
||||||
|
MyEmote({this.setting, this.packages});
|
||||||
|
|
||||||
|
MyEmote.fromJson(Map<String, dynamic> json) {
|
||||||
|
setting =
|
||||||
|
json['setting'] != null ? Setting.fromJson(json['setting']) : null;
|
||||||
|
if (json['packages'] != null) {
|
||||||
|
packages = <Packages>[];
|
||||||
|
json['packages'].forEach((v) {
|
||||||
|
packages!.add(Packages.fromJson(v));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
if (setting != null) {
|
||||||
|
data['setting'] = setting!.toJson();
|
||||||
|
}
|
||||||
|
if (packages != null) {
|
||||||
|
data['packages'] = packages!.map((v) => v.toJson()).toList();
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Setting {
|
||||||
|
int? recentLimit;
|
||||||
|
int? attr;
|
||||||
|
int? focusPkgId;
|
||||||
|
String? schema;
|
||||||
|
|
||||||
|
Setting({this.recentLimit, this.attr, this.focusPkgId, this.schema});
|
||||||
|
|
||||||
|
Setting.fromJson(Map<String, dynamic> json) {
|
||||||
|
recentLimit = json['recent_limit'];
|
||||||
|
attr = json['attr'];
|
||||||
|
focusPkgId = json['focus_pkg_id'];
|
||||||
|
schema = json['schema'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['recent_limit'] = recentLimit;
|
||||||
|
data['attr'] = attr;
|
||||||
|
data['focus_pkg_id'] = focusPkgId;
|
||||||
|
data['schema'] = schema;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Packages {
|
||||||
|
int? id;
|
||||||
|
String? text;
|
||||||
|
String? url;
|
||||||
|
int? mtime;
|
||||||
|
int? type;
|
||||||
|
int? attr;
|
||||||
|
PackagesMeta? meta;
|
||||||
|
List<Emote>? emote;
|
||||||
|
PackagesFlags? flags;
|
||||||
|
dynamic label;
|
||||||
|
String? packageSubTitle;
|
||||||
|
int? refMid;
|
||||||
|
|
||||||
|
Packages(
|
||||||
|
{this.id,
|
||||||
|
this.text,
|
||||||
|
this.url,
|
||||||
|
this.mtime,
|
||||||
|
this.type,
|
||||||
|
this.attr,
|
||||||
|
this.meta,
|
||||||
|
this.emote,
|
||||||
|
this.flags,
|
||||||
|
this.label,
|
||||||
|
this.packageSubTitle,
|
||||||
|
this.refMid});
|
||||||
|
|
||||||
|
Packages.fromJson(Map<String, dynamic> json) {
|
||||||
|
id = json['id'];
|
||||||
|
text = json['text'];
|
||||||
|
url = json['url'];
|
||||||
|
mtime = json['mtime'];
|
||||||
|
type = json['type'];
|
||||||
|
attr = json['attr'];
|
||||||
|
meta = json['meta'] != null ? PackagesMeta.fromJson(json['meta']) : null;
|
||||||
|
if (json['emote'] != null) {
|
||||||
|
emote = <Emote>[];
|
||||||
|
json['emote'].forEach((v) {
|
||||||
|
emote!.add(Emote.fromJson(v));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
flags = json['flags'] != null ? PackagesFlags.fromJson(json['flags']) : null;
|
||||||
|
label = json['label'];
|
||||||
|
packageSubTitle = json['package_sub_title'];
|
||||||
|
refMid = json['ref_mid'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['id'] = id;
|
||||||
|
data['text'] = text;
|
||||||
|
data['url'] = url;
|
||||||
|
data['mtime'] = mtime;
|
||||||
|
data['type'] = type;
|
||||||
|
data['attr'] = attr;
|
||||||
|
if (meta != null) {
|
||||||
|
data['meta'] = meta!.toJson();
|
||||||
|
}
|
||||||
|
if (emote != null) {
|
||||||
|
data['emote'] = emote!.map((v) => v.toJson()).toList();
|
||||||
|
}
|
||||||
|
if (flags != null) {
|
||||||
|
data['flags'] = flags!.toJson();
|
||||||
|
}
|
||||||
|
data['label'] = label;
|
||||||
|
data['package_sub_title'] = packageSubTitle;
|
||||||
|
data['ref_mid'] = refMid;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class PackagesMeta {
|
||||||
|
int? size;
|
||||||
|
int? itemId;
|
||||||
|
|
||||||
|
PackagesMeta({this.size, this.itemId});
|
||||||
|
|
||||||
|
PackagesMeta.fromJson(Map<String, dynamic> json) {
|
||||||
|
size = json['size'];
|
||||||
|
itemId = json['item_id'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['size'] = size;
|
||||||
|
data['item_id'] = itemId;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Emote {
|
||||||
|
int? id;
|
||||||
|
int? packageId;
|
||||||
|
String? text;
|
||||||
|
String? url;
|
||||||
|
int? mtime;
|
||||||
|
int? type;
|
||||||
|
int? attr;
|
||||||
|
EmoteMeta? meta;
|
||||||
|
EmoteFlags? flags;
|
||||||
|
dynamic activity;
|
||||||
|
String? gifUrl;
|
||||||
|
|
||||||
|
Emote(
|
||||||
|
{this.id,
|
||||||
|
this.packageId,
|
||||||
|
this.text,
|
||||||
|
this.url,
|
||||||
|
this.mtime,
|
||||||
|
this.type,
|
||||||
|
this.attr,
|
||||||
|
this.meta,
|
||||||
|
this.flags,
|
||||||
|
this.activity,
|
||||||
|
this.gifUrl});
|
||||||
|
|
||||||
|
Emote.fromJson(Map<String, dynamic> json) {
|
||||||
|
id = json['id'];
|
||||||
|
packageId = json['package_id'];
|
||||||
|
text = json['text'];
|
||||||
|
url = json['url'];
|
||||||
|
mtime = json['mtime'];
|
||||||
|
type = json['type'];
|
||||||
|
attr = json['attr'];
|
||||||
|
meta = json['meta'] != null ? EmoteMeta.fromJson(json['meta']) : null;
|
||||||
|
flags = json['flags'] != null ? EmoteFlags.fromJson(json['flags']) : null;
|
||||||
|
activity = json['activity'];
|
||||||
|
gifUrl = json['gif_url'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['id'] = id;
|
||||||
|
data['package_id'] = packageId;
|
||||||
|
data['text'] = text;
|
||||||
|
data['url'] = url;
|
||||||
|
data['mtime'] = mtime;
|
||||||
|
data['type'] = type;
|
||||||
|
data['attr'] = attr;
|
||||||
|
if (meta != null) {
|
||||||
|
data['meta'] = meta!.toJson();
|
||||||
|
}
|
||||||
|
if (flags != null) {
|
||||||
|
data['flags'] = flags!.toJson();
|
||||||
|
}
|
||||||
|
data['activity'] = activity;
|
||||||
|
data['gif_url'] = gifUrl;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class EmoteMeta {
|
||||||
|
int? size;
|
||||||
|
List<String>? suggest;
|
||||||
|
String? alias;
|
||||||
|
String? gifUrl;
|
||||||
|
|
||||||
|
EmoteMeta({this.size, this.suggest, this.alias, this.gifUrl});
|
||||||
|
|
||||||
|
EmoteMeta.fromJson(Map<String, dynamic> json) {
|
||||||
|
size = json['size'];
|
||||||
|
suggest = json['suggest'].cast<String>();
|
||||||
|
alias = json['alias'];
|
||||||
|
gifUrl = json['gif_url'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['size'] = size;
|
||||||
|
data['suggest'] = suggest;
|
||||||
|
data['alias'] = alias;
|
||||||
|
data['gif_url'] = gifUrl;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class EmoteFlags {
|
||||||
|
bool? unlocked;
|
||||||
|
|
||||||
|
EmoteFlags({this.unlocked});
|
||||||
|
|
||||||
|
EmoteFlags.fromJson(Map<String, dynamic> json) {
|
||||||
|
unlocked = json['unlocked'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['unlocked'] = unlocked;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class PackagesFlags {
|
||||||
|
bool? added;
|
||||||
|
bool? preview;
|
||||||
|
|
||||||
|
PackagesFlags({this.added, this.preview});
|
||||||
|
|
||||||
|
PackagesFlags.fromJson(Map<String, dynamic> json) {
|
||||||
|
added = json['added'];
|
||||||
|
preview = json['preview'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final Map<String, dynamic> data = <String, dynamic>{};
|
||||||
|
data['added'] = added;
|
||||||
|
data['preview'] = preview;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
104
lib/pages/video/detail/reply/reply_emote/view.dart
Normal file
104
lib/pages/video/detail/reply/reply_emote/view.dart
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
import 'package:PiliPalaX/models/user/my_emote.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
|
|
||||||
|
import '../../../../../common/widgets/network_img_layer.dart';
|
||||||
|
import '../../../../../http/reply.dart';
|
||||||
|
|
||||||
|
class EmoteTab extends StatefulWidget {
|
||||||
|
final Function(String) onEmoteTap;
|
||||||
|
const EmoteTab({Key? key, required this.onEmoteTap}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<StatefulWidget> createState() => _EmoteTabState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _EmoteTabState extends State<EmoteTab> with TickerProviderStateMixin {
|
||||||
|
late TabController _myEmoteTabController;
|
||||||
|
late MyEmote myEmote;
|
||||||
|
late Future futureBuild;
|
||||||
|
Future getMyEmote() async {
|
||||||
|
var result = await ReplyHttp.getMyEmote(business: "reply");
|
||||||
|
if (result['status']) {
|
||||||
|
myEmote = MyEmote.fromJson(result['data']);
|
||||||
|
_myEmoteTabController = TabController(
|
||||||
|
length: myEmote.packages!.length,
|
||||||
|
initialIndex: myEmote.setting!.focusPkgId! - 1,
|
||||||
|
vsync: this);
|
||||||
|
} else {
|
||||||
|
SmartDialog.showToast(result['msg']);
|
||||||
|
myEmote = MyEmote();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
futureBuild = getMyEmote();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return FutureBuilder(
|
||||||
|
future: futureBuild,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.done &&
|
||||||
|
myEmote.packages != null) {
|
||||||
|
return Column(
|
||||||
|
children: [
|
||||||
|
Expanded(child: TabBarView(controller: _myEmoteTabController, children: [
|
||||||
|
for (Packages i in myEmote.packages!) ...<Widget>[
|
||||||
|
GridView.builder(
|
||||||
|
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(
|
||||||
|
maxCrossAxisExtent: i.type == 4 ? 100 : 36,
|
||||||
|
mainAxisSpacing: 10,
|
||||||
|
crossAxisSpacing: 10,
|
||||||
|
mainAxisExtent: 36,
|
||||||
|
),
|
||||||
|
itemCount: i.emote!.length,
|
||||||
|
itemBuilder: (BuildContext context, int index) {
|
||||||
|
return GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
widget.onEmoteTap(i.emote![index].text!);
|
||||||
|
},
|
||||||
|
child: i.type == 4
|
||||||
|
? Text(i.emote![index].text!,overflow: TextOverflow.clip,maxLines: 1,)
|
||||||
|
: NetworkImgLayer(
|
||||||
|
width: 36,
|
||||||
|
height: 36,
|
||||||
|
type: 'emote',
|
||||||
|
src: i.emote![index].url,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
]),),
|
||||||
|
SizedBox(
|
||||||
|
height: 45,
|
||||||
|
child: TabBar(
|
||||||
|
isScrollable: true,
|
||||||
|
controller: _myEmoteTabController,
|
||||||
|
tabs: [
|
||||||
|
for (var i in myEmote.packages!)
|
||||||
|
NetworkImgLayer(
|
||||||
|
width: 36,
|
||||||
|
height: 36,
|
||||||
|
type: 'emote',
|
||||||
|
src: i.url,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
))
|
||||||
|
],
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const Center(
|
||||||
|
child: CircularProgressIndicator(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,6 +7,9 @@ import 'package:PiliPalaX/models/common/reply_type.dart';
|
|||||||
import 'package:PiliPalaX/models/video/reply/item.dart';
|
import 'package:PiliPalaX/models/video/reply/item.dart';
|
||||||
import 'package:PiliPalaX/utils/feed_back.dart';
|
import 'package:PiliPalaX/utils/feed_back.dart';
|
||||||
|
|
||||||
|
import '../../../../common/constants.dart';
|
||||||
|
import '../reply/reply_emote/view.dart';
|
||||||
|
|
||||||
class VideoReplyNewDialog extends StatefulWidget {
|
class VideoReplyNewDialog extends StatefulWidget {
|
||||||
final int? oid;
|
final int? oid;
|
||||||
final int? root;
|
final int? root;
|
||||||
@@ -32,6 +35,7 @@ class _VideoReplyNewDialogState extends State<VideoReplyNewDialog>
|
|||||||
final TextEditingController _replyContentController = TextEditingController();
|
final TextEditingController _replyContentController = TextEditingController();
|
||||||
final FocusNode replyContentFocusNode = FocusNode();
|
final FocusNode replyContentFocusNode = FocusNode();
|
||||||
final GlobalKey _formKey = GlobalKey<FormState>();
|
final GlobalKey _formKey = GlobalKey<FormState>();
|
||||||
|
bool isShowEmote = false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
@@ -141,9 +145,13 @@ class _VideoReplyNewDialogState extends State<VideoReplyNewDialog>
|
|||||||
width: 36,
|
width: 36,
|
||||||
height: 36,
|
height: 36,
|
||||||
child: IconButton(
|
child: IconButton(
|
||||||
onPressed: () {
|
onPressed: () async {
|
||||||
FocusScope.of(context)
|
FocusScope.of(context)
|
||||||
.requestFocus(replyContentFocusNode);
|
.requestFocus(replyContentFocusNode);
|
||||||
|
await Future.delayed(const Duration(milliseconds: 200));
|
||||||
|
setState(() {
|
||||||
|
isShowEmote = false;
|
||||||
|
});
|
||||||
},
|
},
|
||||||
icon: Icon(Icons.keyboard,
|
icon: Icon(Icons.keyboard,
|
||||||
size: 22,
|
size: 22,
|
||||||
@@ -154,7 +162,44 @@ class _VideoReplyNewDialogState extends State<VideoReplyNewDialog>
|
|||||||
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
MaterialStateProperty.resolveWith((states) {
|
MaterialStateProperty.resolveWith((states) {
|
||||||
return Theme.of(context).highlightColor;
|
if (states.contains(MaterialState.pressed) || !isShowEmote) {
|
||||||
|
return Theme.of(context).highlightColor;
|
||||||
|
}
|
||||||
|
// 默认状态下,返回透明颜色
|
||||||
|
return Colors.transparent;
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
width: 10,
|
||||||
|
),
|
||||||
|
SizedBox(
|
||||||
|
width: 36,
|
||||||
|
height: 36,
|
||||||
|
child: IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
//收起输入法
|
||||||
|
FocusScope.of(context).unfocus();
|
||||||
|
// 弹出表情选择
|
||||||
|
setState(() {
|
||||||
|
isShowEmote = true;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
icon: Icon(Icons.emoji_emotions,
|
||||||
|
size: 22,
|
||||||
|
color: Theme.of(context).colorScheme.onBackground),
|
||||||
|
highlightColor:
|
||||||
|
Theme.of(context).colorScheme.onInverseSurface,
|
||||||
|
style: ButtonStyle(
|
||||||
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
|
backgroundColor:
|
||||||
|
MaterialStateProperty.resolveWith((states) {
|
||||||
|
if (states.contains(MaterialState.pressed) || isShowEmote) {
|
||||||
|
return Theme.of(context).highlightColor;
|
||||||
|
}
|
||||||
|
// 默认状态下,返回透明颜色
|
||||||
|
return Colors.transparent;
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -165,16 +210,42 @@ class _VideoReplyNewDialogState extends State<VideoReplyNewDialog>
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
AnimatedSize(
|
if (!isShowEmote)
|
||||||
curve: Curves.easeInOut,
|
SizedBox(
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
child: SizedBox(
|
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
height: keyboardHeight,
|
height: keyboardHeight,
|
||||||
),
|
),
|
||||||
),
|
if (isShowEmote)
|
||||||
|
SizedBox(
|
||||||
|
width: double.infinity,
|
||||||
|
height: 310,
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: StyleString.safeSpace),
|
||||||
|
child: EmoteTab(
|
||||||
|
onEmoteTap: onEmoteTap,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void onEmoteTap(String emoteString) {
|
||||||
|
// 在光标处插入表情
|
||||||
|
final String currentText = _replyContentController.text;
|
||||||
|
final TextSelection selection = _replyContentController.selection;
|
||||||
|
final String newText = currentText.replaceRange(
|
||||||
|
selection.start,
|
||||||
|
selection.end,
|
||||||
|
emoteString,
|
||||||
|
);
|
||||||
|
_replyContentController.text = newText;
|
||||||
|
final int newCursorIndex = selection.start + emoteString.length;
|
||||||
|
_replyContentController.selection = selection.copyWith(
|
||||||
|
baseOffset: newCursorIndex,
|
||||||
|
extentOffset: newCursorIndex,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user