feat: 初步支持查看【回复我的】【at我】【收到的赞】内容

This commit is contained in:
orz12
2024-02-03 06:25:18 +08:00
parent 8abbab4c88
commit 4d9e15ad50
17 changed files with 1452 additions and 43 deletions

View File

@@ -6,6 +6,100 @@ import 'init.dart';
class MsgHttp {
static Future msgFeedReplyMe({int cursor = -1, int cursorTime = -1}) async {
var res = await Request().get(Api.msgFeedReply, data: {
'id': cursor == -1 ? null : cursor,
'reply_time': cursorTime == -1 ? null : cursorTime,
});
if (res.data['code'] == 0) {
return {
'status': true,
'data': res.data['data'],
};
} else {
return {
'status': false,
'date': [],
'msg': res.data['message'],
};
}
}
static Future msgFeedAtMe({int cursor = -1, int cursorTime = -1}) async {
var res = await Request().get(Api.msgFeedAt,data: {
'id': cursor == -1 ? null : cursor,
'at_time': cursorTime == -1 ? null : cursorTime,
});
if (res.data['code'] == 0) {
return {
'status': true,
'data': res.data['data'],
};
} else {
return {
'status': false,
'date': [],
'msg': res.data['message'],
};
}
}
static Future msgFeedLikeMe({int cursor = -1, int cursorTime = -1}) async {
var res = await Request().get(Api.msgFeedLike,data: {
'id': cursor == -1 ? null : cursor,
'like_time': cursorTime == -1 ? null : cursorTime,
});
if (res.data['code'] == 0) {
return {
'status': true,
'data': res.data['data'],
};
} else {
return {
'status': false,
'date': [],
'msg': res.data['message'],
};
}
}
static Future msgFeedSysUserNotify() async {
String csrf = await Request.getCsrf();
var res = await Request().get(Api.msgSysUserNotify, data: {
'csrf': csrf,
'csrf': csrf,
'page_size': 20,
});
if (res.data['code'] == 0) {
return {
'status': true,
'data': res.data['data'],
};
} else {
return {
'status': false,
'date': [],
'msg': res.data['message'],
};
}
}
static Future msgFeedSysUnifiedNotify() async {
String csrf = await Request.getCsrf();
var res = await Request().get(Api.msgSysUnifiedNotify, data: {
'csrf': csrf,
'csrf': csrf,
'page_size': 10,
});
if (res.data['code'] == 0) {
return {
'status': true,
'data': res.data['data'],
};
} else {
return {
'status': false,
'date': [],
'msg': res.data['message'],
};
}
}
static Future msgFeedUnread() async {
var res = await Request().get(Api.msgFeedUnread);
if (res.data['code'] == 0) {

View File

@@ -0,0 +1,222 @@
class MsgFeedAtMe {
Cursor? cursor;
List<AtMeItems>? items;
MsgFeedAtMe({cursor, items});
MsgFeedAtMe.fromJson(Map<String, dynamic> json) {
cursor = json['cursor'] != null ? Cursor.fromJson(json['cursor']) : null;
if (json['items'] != null) {
items = <AtMeItems>[];
json['items'].forEach((v) {
items!.add(AtMeItems.fromJson(v));
});
}
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['cursor'] = cursor?.toJson();
data['items'] = items?.map((v) => v.toJson()).toList();
return data;
}
}
class Cursor {
bool? isEnd;
int? id;
int? time;
Cursor({isEnd, id, time});
Cursor.fromJson(Map<String, dynamic> json) {
isEnd = json['is_end'];
id = json['id'];
time = json['time'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['is_end'] = isEnd;
data['id'] = id;
data['time'] = time;
return data;
}
}
class AtMeItems {
int? id;
User? user;
Item? item;
int? atTime;
AtMeItems({id, user, item, atTime});
AtMeItems.fromJson(Map<String, dynamic> json) {
id = json['id'];
user = json['user'] != null ? User.fromJson(json['user']) : null;
item = json['item'] != null ? Item.fromJson(json['item']) : null;
atTime = json['at_time'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['id'] = id;
data['user'] = user?.toJson();
data['item'] = item?.toJson();
data['at_time'] = atTime;
return data;
}
}
class User {
int? mid;
int? fans;
String? nickname;
String? avatar;
String? midLink;
bool? follow;
User(
{this.mid,
this.fans,
this.nickname,
this.avatar,
this.midLink,
this.follow});
User.fromJson(Map<String, dynamic> json) {
mid = json['mid'];
fans = json['fans'];
nickname = json['nickname'];
avatar = json['avatar'];
midLink = json['mid_link'];
follow = json['follow'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['mid'] = mid;
data['fans'] = fans;
data['nickname'] = nickname;
data['avatar'] = avatar;
data['mid_link'] = midLink;
data['follow'] = follow;
return data;
}
}
class Item {
String? type;
String? business;
int? businessId;
String? title;
String? image;
String? uri;
int? subjectId;
int? rootId;
int? targetId;
int? sourceId;
String? sourceContent;
String? nativeUri;
List<AtDetails>? atDetails;
List? topicDetails;
bool? hideReplyButton;
Item(
{this.type,
this.business,
this.businessId,
this.title,
this.image,
this.uri,
this.subjectId,
this.rootId,
this.targetId,
this.sourceId,
this.sourceContent,
this.nativeUri,
this.atDetails,
this.topicDetails,
this.hideReplyButton});
Item.fromJson(Map<String, dynamic> json) {
type = json['type'];
business = json['business'];
businessId = json['business_id'];
title = json['title'];
image = json['image'];
uri = json['uri'];
subjectId = json['subject_id'];
rootId = json['root_id'];
targetId = json['target_id'];
sourceId = json['source_id'];
sourceContent = json['source_content'];
nativeUri = json['native_uri'];
if (json['at_details'] != null) {
atDetails = <AtDetails>[];
json['at_details'].forEach((v) {
atDetails!.add(AtDetails.fromJson(v));
});
}
topicDetails = json['topic_details'];
hideReplyButton = json['hide_reply_button'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['type'] = type;
data['business'] = business;
data['business_id'] = businessId;
data['title'] = title;
data['image'] = image;
data['uri'] = uri;
data['subject_id'] = subjectId;
data['root_id'] = rootId;
data['target_id'] = targetId;
data['source_id'] = sourceId;
data['source_content'] = sourceContent;
data['native_uri'] = nativeUri;
data['at_details'] = atDetails?.map((v) => v.toJson()).toList();
data['topic_details'] = topicDetails?.map((v) => v.toJson()).toList();
data['hide_reply_button'] = hideReplyButton;
return data;
}
}
class AtDetails {
int? mid;
int? fans;
String? nickname;
String? avatar;
String? midLink;
bool? follow;
AtDetails(
{this.mid,
this.fans,
this.nickname,
this.avatar,
this.midLink,
this.follow});
AtDetails.fromJson(Map<String, dynamic> json) {
mid = json['mid'];
fans = json['fans'];
nickname = json['nickname'];
avatar = json['avatar'];
midLink = json['mid_link'];
follow = json['follow'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['mid'] = mid;
data['fans'] = fans;
data['nickname'] = nickname;
data['avatar'] = avatar;
data['mid_link'] = midLink;
data['follow'] = follow;
return data;
}
}

View File

@@ -0,0 +1,238 @@
class MsgFeedLikeMe {
Latest? latest;
Total? total;
MsgFeedLikeMe({latest, total});
MsgFeedLikeMe.fromJson(Map<String, dynamic> json) {
latest =
json['latest'] != null ? Latest.fromJson(json['latest']) : null;
total = json['total'] != null ? Total.fromJson(json['total']) : null;
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['latest'] = latest?.toJson();
data['total'] = total?.toJson();
return data;
}
}
class Latest {
List<LikeMeItems>? items;
int? lastViewAt;
Latest({items, lastViewAt});
Latest.fromJson(Map<String, dynamic> json) {
if (json['items'] != null) {
items = <LikeMeItems>[];
json['items'].forEach((v) {
items!.add(LikeMeItems.fromJson(v));
});
}
lastViewAt = json['last_view_at'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['items'] = items?.map((v) => v.toJson()).toList();
data['last_view_at'] = lastViewAt;
return data;
}
}
class LikeMeItems {
int? id;
List<Users>? users;
Item? item;
int? counts;
int? likeTime;
int? noticeState;
LikeMeItems(
{id,
users,
item,
counts,
likeTime,
noticeState});
LikeMeItems.fromJson(Map<String, dynamic> json) {
id = json['id'];
if (json['users'] != null) {
users = <Users>[];
json['users'].forEach((v) {
users!.add(Users.fromJson(v));
});
}
item = json['item'] != null ? Item.fromJson(json['item']) : null;
counts = json['counts'];
likeTime = json['like_time'];
noticeState = json['notice_state'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['id'] = id;
data['users'] = users?.map((v) => v.toJson()).toList();
data['item'] = item?.toJson();
data['counts'] = counts;
data['like_time'] = likeTime;
data['notice_state'] = noticeState;
return data;
}
}
class Users {
int? mid;
int? fans;
String? nickname;
String? avatar;
String? midLink;
bool? follow;
Users(
{mid,
fans,
nickname,
avatar,
midLink,
follow});
Users.fromJson(Map<String, dynamic> json) {
mid = json['mid'];
fans = json['fans'];
nickname = json['nickname'];
avatar = json['avatar'];
midLink = json['mid_link'];
follow = json['follow'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['mid'] = mid;
data['fans'] = fans;
data['nickname'] = nickname;
data['avatar'] = avatar;
data['mid_link'] = midLink;
data['follow'] = follow;
return data;
}
}
class Item {
int? itemId;
int? pid;
String? type;
String? business;
int? businessId;
int? replyBusinessId;
int? likeBusinessId;
String? title;
String? desc;
String? image;
String? uri;
String? detailName;
String? nativeUri;
int? ctime;
Item(
{itemId,
pid,
type,
business,
businessId,
replyBusinessId,
likeBusinessId,
title,
desc,
image,
uri,
detailName,
nativeUri,
ctime});
Item.fromJson(Map<String, dynamic> json) {
itemId = json['item_id'];
pid = json['pid'];
type = json['type'];
business = json['business'];
businessId = json['business_id'];
replyBusinessId = json['reply_business_id'];
likeBusinessId = json['like_business_id'];
title = json['title'];
desc = json['desc'];
image = json['image'];
uri = json['uri'];
detailName = json['detail_name'];
nativeUri = json['native_uri'];
ctime = json['ctime'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['item_id'] = itemId;
data['pid'] = pid;
data['type'] = type;
data['business'] = business;
data['business_id'] = businessId;
data['reply_business_id'] = replyBusinessId;
data['like_business_id'] = likeBusinessId;
data['title'] = title;
data['desc'] = desc;
data['image'] = image;
data['uri'] = uri;
data['detail_name'] = detailName;
data['native_uri'] = nativeUri;
data['ctime'] = ctime;
return data;
}
}
class Total {
Cursor? cursor;
List<LikeMeItems>? items;
Total({cursor, items});
Total.fromJson(Map<String, dynamic> json) {
cursor =
json['cursor'] != null ? Cursor.fromJson(json['cursor']) : null;
if (json['items'] != null) {
items = <LikeMeItems>[];
json['items'].forEach((v) {
items!.add(LikeMeItems.fromJson(v));
});
}
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['cursor'] = cursor?.toJson();
data['items'] = items?.map((v) => v.toJson()).toList();
return data;
}
}
class Cursor {
bool? isEnd;
int? id;
int? time;
Cursor({isEnd, id, time});
Cursor.fromJson(Map<String, dynamic> json) {
isEnd = json['is_end'];
id = json['id'];
time = json['time'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['is_end'] = isEnd;
data['id'] = id;
data['time'] = time;
return data;
}
}

View File

@@ -0,0 +1,274 @@
class MsgFeedReplyMe {
Cursor? cursor;
List<ReplyMeItems>? items;
int? lastViewAt;
MsgFeedReplyMe({this.cursor, this.items, this.lastViewAt});
MsgFeedReplyMe.fromJson(Map<String, dynamic> json) {
cursor =
json['cursor'] != null ? Cursor.fromJson(json['cursor']) : null;
if (json['items'] != null) {
items = <ReplyMeItems>[];
json['items'].forEach((v) {
items!.add(ReplyMeItems.fromJson(v));
});
}
lastViewAt = json['last_view_at'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['cursor'] = cursor?.toJson();
data['items'] = items?.map((v) => v.toJson()).toList();
data['last_view_at'] = lastViewAt;
return data;
}
}
class Cursor {
bool? isEnd;
int? id;
int? time;
Cursor({this.isEnd, this.id, this.time});
Cursor.fromJson(Map<String, dynamic> json) {
isEnd = json['is_end'];
id = json['id'];
time = json['time'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['is_end'] = isEnd;
data['id'] = id;
data['time'] = time;
return data;
}
}
class ReplyMeItems {
int? id;
User? user;
Item? item;
int? counts;
int? isMulti;
int? replyTime;
ReplyMeItems(
{this.id,
this.user,
this.item,
this.counts,
this.isMulti,
this.replyTime});
ReplyMeItems.fromJson(Map<String, dynamic> json) {
id = json['id'];
user = json['user'] != null ? User.fromJson(json['user']) : null;
item = json['item'] != null ? Item.fromJson(json['item']) : null;
counts = json['counts'];
isMulti = json['is_multi'];
replyTime = json['reply_time'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['id'] = id;
if (user != null) {
data['user'] = user!.toJson();
}
if (item != null) {
data['item'] = item!.toJson();
}
data['counts'] = counts;
data['is_multi'] = isMulti;
data['reply_time'] = replyTime;
return data;
}
}
class User {
int? mid;
int? fans;
String? nickname;
String? avatar;
String? midLink;
bool? follow;
User(
{this.mid,
this.fans,
this.nickname,
this.avatar,
this.midLink,
this.follow});
User.fromJson(Map<String, dynamic> json) {
mid = json['mid'];
fans = json['fans'];
nickname = json['nickname'];
avatar = json['avatar'];
midLink = json['mid_link'];
follow = json['follow'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['mid'] = mid;
data['fans'] = fans;
data['nickname'] = nickname;
data['avatar'] = avatar;
data['mid_link'] = midLink;
data['follow'] = follow;
return data;
}
}
class Item {
int? subjectId;
int? rootId;
int? sourceId;
int? targetId;
String? type;
int? businessId;
String? business;
String? title;
String? desc;
String? image;
String? uri;
String? nativeUri;
String? detailTitle;
String? rootReplyContent;
String? sourceContent;
String? targetReplyContent;
List<AtDetails>? atDetails;
List? topicDetails;
bool? hideReplyButton;
bool? hideLikeButton;
int? likeState;
dynamic danmu;
String? message;
Item(
{this.subjectId,
this.rootId,
this.sourceId,
this.targetId,
this.type,
this.businessId,
this.business,
this.title,
this.desc,
this.image,
this.uri,
this.nativeUri,
this.detailTitle,
this.rootReplyContent,
this.sourceContent,
this.targetReplyContent,
this.atDetails,
this.topicDetails,
this.hideReplyButton,
this.hideLikeButton,
this.likeState,
this.danmu,
this.message});
Item.fromJson(Map<String, dynamic> json) {
subjectId = json['subject_id'];
rootId = json['root_id'];
sourceId = json['source_id'];
targetId = json['target_id'];
type = json['type'];
businessId = json['business_id'];
business = json['business'];
title = json['title'];
desc = json['desc'];
image = json['image'];
uri = json['uri'];
nativeUri = json['native_uri'];
detailTitle = json['detail_title'];
rootReplyContent = json['root_reply_content'];
sourceContent = json['source_content'];
targetReplyContent = json['target_reply_content'];
if (json['at_details'] != null) {
atDetails = <AtDetails>[];
json['at_details'].forEach((v) {
atDetails!.add(AtDetails.fromJson(v));
});
}
topicDetails = json['topic_details'];
hideReplyButton = json['hide_reply_button'];
hideLikeButton = json['hide_like_button'];
likeState = json['like_state'];
danmu = json['danmu'];
message = json['message'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['subject_id'] = subjectId;
data['root_id'] = rootId;
data['source_id'] = sourceId;
data['target_id'] = targetId;
data['type'] = type;
data['business_id'] = businessId;
data['business'] = business;
data['title'] = title;
data['desc'] = desc;
data['image'] = image;
data['uri'] = uri;
data['native_uri'] = nativeUri;
data['detail_title'] = detailTitle;
data['root_reply_content'] = rootReplyContent;
data['source_content'] = sourceContent;
data['target_reply_content'] = targetReplyContent;
data['at_details'] = atDetails?.map((v) => v.toJson()).toList();
data['topic_details'] = topicDetails?.map((v) => v.toJson()).toList();
data['hide_reply_button'] = hideReplyButton;
data['hide_like_button'] = hideLikeButton;
data['like_state'] = likeState;
data['danmu'] = danmu;
data['message'] = message;
return data;
}
}
class AtDetails {
int? mid;
int? fans;
String? nickname;
String? avatar;
String? midLink;
bool? follow;
AtDetails(
{this.mid,
this.fans,
this.nickname,
this.avatar,
this.midLink,
this.follow});
AtDetails.fromJson(Map<String, dynamic> json) {
mid = json['mid'];
fans = json['fans'];
nickname = json['nickname'];
avatar = json['avatar'];
midLink = json['mid_link'];
follow = json['follow'];
}
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = <String, dynamic>{};
data['mid'] = mid;
data['fans'] = fans;
data['nickname'] = nickname;
data['avatar'] = avatar;
data['mid_link'] = midLink;
data['follow'] = follow;
return data;
}
}

View File

@@ -0,0 +1,43 @@
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/http/msg.dart';
import 'package:pilipala/models/msg/msgfeed_at_me.dart';
class AtMeController extends GetxController {
RxList<AtMeItems> msgFeedAtMeList = <AtMeItems>[].obs;
bool isLoading = false;
int cursor = -1;
int cursorTime = -1;
bool isEnd = false;
Future queryMsgFeedAtMe() async {
if (isLoading) return;
isLoading = true;
var res = await MsgHttp.msgFeedAtMe(cursor: cursor, cursorTime: cursorTime);
isLoading = false;
if (res['status']) {
MsgFeedAtMe data = MsgFeedAtMe.fromJson(res['data']);
isEnd = data.cursor?.isEnd ?? false;
if (cursor == -1) {
msgFeedAtMeList.assignAll(data.items!);
} else {
msgFeedAtMeList.addAll(data.items!);
}
cursor = data.cursor?.id ?? -1;
cursorTime = data.cursor?.time ?? -1;
} else {
SmartDialog.showToast(res['msg']);
}
}
Future onLoad() async {
if (isEnd) return;
queryMsgFeedAtMe();
}
Future onRefresh() async {
cursor = -1;
queryMsgFeedAtMe();
}
}

View File

@@ -0,0 +1,4 @@
library whisper;
export './controller.dart';
export './view.dart';

View File

@@ -0,0 +1,124 @@
import 'package:easy_debounce/easy_throttle.dart';
import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/common/widgets/network_img_layer.dart';
import 'controller.dart';
class AtMePage extends StatefulWidget {
const AtMePage({super.key});
@override
State<AtMePage> createState() => _AtMePageState();
}
class _AtMePageState extends State<AtMePage> {
late final AtMeController _atMeController = Get.put(AtMeController());
final ScrollController _scrollController = ScrollController();
@override
void initState() {
_atMeController.queryMsgFeedAtMe();
super.initState();
_scrollController.addListener(_scrollListener);
}
Future _scrollListener() async {
if (_scrollController.position.pixels >=
_scrollController.position.maxScrollExtent - 200) {
EasyThrottle.throttle('my-throttler', const Duration(milliseconds: 800),
() async {
await _atMeController.onLoad();
});
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('@我的'),
),
body: RefreshIndicator(
onRefresh: () async {
await _atMeController.onRefresh();
},
child: SingleChildScrollView(
controller: _scrollController,
child: LayoutBuilder(
builder: (BuildContext context, BoxConstraints constraints) {
return Obx(
() {
if (_atMeController.msgFeedAtMeList.isEmpty) {
return const Center(
child: CircularProgressIndicator(),
);
}
return ListView.separated(
itemCount: _atMeController.msgFeedAtMeList.length,
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemBuilder: (_, int i) {
return ListTile(
onTap: () {
String nativeUri = _atMeController
.msgFeedAtMeList[i].item?.nativeUri ??
"";
SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
},
leading: NetworkImgLayer(
width: 45,
height: 45,
type: 'avatar',
src: _atMeController.msgFeedAtMeList[i].user?.avatar,
),
title: Text(
"${_atMeController.msgFeedAtMeList[i].user?.nickname} "
"${_atMeController.msgFeedAtMeList[i].item?.business}中@了我",
style: Theme.of(context).textTheme.bodyMedium!,
),
subtitle: Text(
_atMeController
.msgFeedAtMeList[i].item?.sourceContent ??
"",
maxLines: 2,
overflow: TextOverflow.ellipsis,
style: Theme.of(context)
.textTheme
.labelMedium!
.copyWith(
color:
Theme.of(context).colorScheme.outline)),
trailing: _atMeController
.msgFeedAtMeList[i].item?.image !=
null &&
_atMeController.msgFeedAtMeList[i].item?.image !=
""
? NetworkImgLayer(
width: 45,
height: 45,
type: 'cover',
src: _atMeController
.msgFeedAtMeList[i].item?.image,
)
: null,
);
},
separatorBuilder: (BuildContext context, int index) {
return Divider(
indent: 72,
endIndent: 20,
height: 6,
color: Colors.grey.withOpacity(0.1),
);
},
);
},
);
}),
),
),
);
}
}

View File

@@ -0,0 +1,43 @@
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/http/msg.dart';
import '../../../models/msg/msgfeed_like_me.dart';
class LikeMeController extends GetxController {
RxList<LikeMeItems> msgFeedLikeMeList = <LikeMeItems>[].obs;
bool isLoading = false;
int cursor = -1;
int cursorTime = -1;
bool isEnd = false;
Future queryMsgFeedLikeMe() async {
if (isLoading) return;
isLoading = true;
var res = await MsgHttp.msgFeedLikeMe(cursor: cursor, cursorTime: cursorTime);
isLoading = false;
if (res['status']) {
MsgFeedLikeMe data = MsgFeedLikeMe.fromJson(res['data']);
isEnd = data.total?.cursor?.isEnd ?? false;
if (cursor == -1) {
msgFeedLikeMeList.assignAll(data.total!.items!);
} else {
msgFeedLikeMeList.addAll(data.total!.items!);
}
cursor = data.total?.cursor?.id ?? -1;
cursorTime = data.total?.cursor?.time ?? -1;
} else {
SmartDialog.showToast(res['msg']);
}
}
Future onLoad() async {
if (isEnd) return;
queryMsgFeedLikeMe();
}
Future onRefresh() async {
cursor = -1;
queryMsgFeedLikeMe();
}
}

View File

@@ -0,0 +1,4 @@
library whisper;
export './controller.dart';
export './view.dart';

View File

@@ -0,0 +1,167 @@
import 'package:easy_debounce/easy_throttle.dart';
import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/common/widgets/network_img_layer.dart';
import 'controller.dart';
class LikeMePage extends StatefulWidget {
const LikeMePage({super.key});
@override
State<LikeMePage> createState() => _LikeMePageState();
}
class _LikeMePageState extends State<LikeMePage> {
late final LikeMeController _likeMeController = Get.put(LikeMeController());
final ScrollController _scrollController = ScrollController();
@override
void initState() {
_likeMeController.queryMsgFeedLikeMe();
super.initState();
_scrollController.addListener(_scrollListener);
}
Future _scrollListener() async {
if (_scrollController.position.pixels >=
_scrollController.position.maxScrollExtent - 200) {
EasyThrottle.throttle('my-throttler', const Duration(milliseconds: 800),
() async {
await _likeMeController.onLoad();
});
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('收到的赞'),
),
body: RefreshIndicator(
onRefresh: () async {
await _likeMeController.onRefresh();
},
child: SingleChildScrollView(
controller: _scrollController,
child: LayoutBuilder(
builder: (BuildContext context, BoxConstraints constraints) {
return Obx(
() {
if (_likeMeController.msgFeedLikeMeList.isEmpty) {
return const Center(
child: CircularProgressIndicator(),
);
}
return ListView.separated(
itemCount: _likeMeController.msgFeedLikeMeList.length,
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemBuilder: (_, int i) {
return ListTile(
onTap: () {
String nativeUri = _likeMeController
.msgFeedLikeMeList[i].item?.nativeUri ??
"";
SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
},
leading: SizedBox(
width: 50,
height: 50,
child: Stack(
children: [
for (var j = 0;
j <
_likeMeController.msgFeedLikeMeList[i]
.users!.length &&
j < 4;
j++) ...<Widget>[
Positioned(
left: 15 * (j % 2).toDouble(),
top: 15 * (j ~/ 2).toDouble(),
child: NetworkImgLayer(
width: _likeMeController
.msgFeedLikeMeList[i]
.users!
.length >
1
? 30
: 45,
height: _likeMeController
.msgFeedLikeMeList[i]
.users!
.length >
1
? 30
: 45,
type: 'avatar',
src: _likeMeController
.msgFeedLikeMeList[i]
.users![j]
.avatar,
)),
]
],
)),
title: Text(
"${_likeMeController.msgFeedLikeMeList[i].users!.map((e) => e.nickname).join("")} "
"赞了我的${_likeMeController.msgFeedLikeMeList[i].item?.business}",
style: Theme.of(context).textTheme.bodyMedium!,
maxLines: 2,
overflow: TextOverflow.ellipsis,
),
subtitle:
_likeMeController.msgFeedLikeMeList[i].item?.title !=
null &&
_likeMeController
.msgFeedLikeMeList[i].item?.title !=
""
? Text(
_likeMeController
.msgFeedLikeMeList[i].item?.title ??
"",
maxLines: 2,
overflow: TextOverflow.ellipsis,
style: Theme.of(context)
.textTheme
.labelMedium!
.copyWith(
color: Theme.of(context)
.colorScheme
.outline))
: null,
trailing:
_likeMeController.msgFeedLikeMeList[i].item?.image !=
null &&
_likeMeController
.msgFeedLikeMeList[i].item?.image !=
""
? NetworkImgLayer(
width: 45,
height: 45,
type: 'cover',
src: _likeMeController
.msgFeedLikeMeList[i].item?.image,
)
: null,
);
},
separatorBuilder: (BuildContext context, int index) {
return Divider(
indent: 72,
endIndent: 20,
height: 6,
color: Colors.grey.withOpacity(0.1),
);
},
);
},
);
}),
),
),
);
}
}

View File

@@ -0,0 +1,44 @@
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/http/msg.dart';
import '../../../models/msg/msgfeed_reply_me.dart';
class ReplyMeController extends GetxController {
RxList<ReplyMeItems> msgFeedReplyMeList = <ReplyMeItems>[].obs;
bool isLoading = false;
int cursor = -1;
int cursorTime = -1;
bool isEnd = false;
Future queryMsgFeedReplyMe() async {
if (isLoading) return;
isLoading = true;
var res = await MsgHttp.msgFeedReplyMe(cursor: cursor, cursorTime: cursorTime);
isLoading = false;
if (res['status']) {
MsgFeedReplyMe data = MsgFeedReplyMe.fromJson(res['data']);
isEnd = data.cursor?.isEnd ?? false;
if (cursor == -1) {
msgFeedReplyMeList.assignAll(data.items!);
} else {
msgFeedReplyMeList.addAll(data.items!);
}
cursor = data.cursor?.id ?? -1;
cursorTime = data.cursor?.time ?? -1;
} else {
SmartDialog.showToast(res['msg']);
}
}
Future onLoad() async {
if (isEnd) return;
queryMsgFeedReplyMe();
}
Future onRefresh() async {
cursor = -1;
queryMsgFeedReplyMe();
}
}

View File

@@ -0,0 +1,4 @@
library whisper;
export './controller.dart';
export './view.dart';

View File

@@ -0,0 +1,149 @@
import 'package:easy_debounce/easy_throttle.dart';
import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/common/widgets/network_img_layer.dart';
import 'controller.dart';
class ReplyMePage extends StatefulWidget {
const ReplyMePage({super.key});
@override
State<ReplyMePage> createState() => _ReplyMePageState();
}
class _ReplyMePageState extends State<ReplyMePage> {
late final ReplyMeController _replyMeController =
Get.put(ReplyMeController());
final ScrollController _scrollController = ScrollController();
@override
void initState() {
_replyMeController.queryMsgFeedReplyMe();
super.initState();
_scrollController.addListener(_scrollListener);
}
Future _scrollListener() async {
if (_scrollController.position.pixels >=
_scrollController.position.maxScrollExtent - 200) {
EasyThrottle.throttle('my-throttler', const Duration(milliseconds: 800),
() async {
await _replyMeController.onLoad();
});
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('回复我的'),
),
body: RefreshIndicator(
onRefresh: () async {
await _replyMeController.onRefresh();
},
child: SingleChildScrollView(
controller: _scrollController,
child: LayoutBuilder(
builder: (BuildContext context, BoxConstraints constraints) {
return Obx(
() {
if (_replyMeController.msgFeedReplyMeList.isEmpty) {
return const Center(
child: CircularProgressIndicator(),
);
}
return ListView.separated(
itemCount: _replyMeController.msgFeedReplyMeList.length,
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
itemBuilder: (_, int i) {
return ListTile(
onTap: () {
String nativeUri = _replyMeController
.msgFeedReplyMeList[i].item?.nativeUri ??
"";
SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
},
leading: NetworkImgLayer(
width: 45,
height: 45,
type: 'avatar',
src: _replyMeController
.msgFeedReplyMeList[i].user?.avatar,
),
title: Text(
"${_replyMeController.msgFeedReplyMeList[i].user?.nickname} "
"回复了我的${_replyMeController.msgFeedReplyMeList[i].item?.business}",
style: Theme.of(context).textTheme.bodyMedium!,
),
subtitle: Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const SizedBox(height: 4),
Text(
_replyMeController.msgFeedReplyMeList[i].item
?.sourceContent ??
"",
style: Theme.of(context).textTheme.bodyMedium),
const SizedBox(height: 4),
if (_replyMeController.msgFeedReplyMeList[i].item
?.targetReplyContent !=
null &&
_replyMeController.msgFeedReplyMeList[i].item
?.targetReplyContent !=
"")
Text(
"| ${_replyMeController.msgFeedReplyMeList[i].item?.targetReplyContent}",
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: Theme.of(context)
.textTheme
.labelMedium!
.copyWith(
color: Theme.of(context)
.colorScheme
.outline,
height: 1.5)),
if (_replyMeController.msgFeedReplyMeList[i].item
?.rootReplyContent !=
null &&
_replyMeController.msgFeedReplyMeList[i].item
?.rootReplyContent !=
"")
Text(
" | ${_replyMeController.msgFeedReplyMeList[i].item?.rootReplyContent}",
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: Theme.of(context)
.textTheme
.labelMedium!
.copyWith(
color: Theme.of(context)
.colorScheme
.outline,
height: 1.5)),
]),
);
},
separatorBuilder: (BuildContext context, int index) {
return Divider(
indent: 72,
endIndent: 20,
height: 6,
color: Colors.grey.withOpacity(0.1),
);
},
);
},
);
}),
),
),
);
}
}

View File

@@ -16,25 +16,25 @@ class WhisperController extends GetxController {
{
"name":"回复我的",
"icon":Icons.message_outlined,
"route": "/",
"route": "/replyMe",
"value": 0
},
{
"name":"@我",
"icon":Icons.alternate_email_outlined,
"route": "/",
"route": "/atMe",
"value": 0
},
{
"name":"收到的赞",
"icon":Icons.favorite_border_outlined,
"route": "/",
"route": "/likeMe",
"value": 0
},
{
"name":"系统通知",
"icon":Icons.notifications_none_outlined,
"route": "/",
"route": "/sysMsg",
"value": 0
},
].obs;

View File

@@ -58,20 +58,20 @@ class _WhisperPageState extends State<WhisperPage> {
return Padding(
padding: const EdgeInsets.only(left: 20, right: 20),
child: SizedBox(
height: constraints.maxWidth / 5,
height: constraints.maxWidth / 4,
child: Obx(
() => GridView.count(
primary: false,
crossAxisCount: 4,
padding: const EdgeInsets.all(0),
childAspectRatio: 1.25,
children:
_whisperController.msgFeedTop.map((item) {
return Column(
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Badge(
children: _whisperController.msgFeedTop.map((item) {
return GestureDetector(
child: Column(
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Badge(
isLabelVisible: item['value'] > 0,
backgroundColor:
Theme.of(context).colorScheme.primary,
@@ -80,38 +80,25 @@ class _WhisperPageState extends State<WhisperPage> {
.onInverseSurface,
label: Text(" ${item['value']} "),
alignment: Alignment.topRight,
child: SizedBox(
width: 36,
height: 36,
child: IconButton(
style: ButtonStyle(
padding: MaterialStateProperty.all(
EdgeInsets.zero),
backgroundColor:
MaterialStateProperty.resolveWith(
(states) {
return Theme.of(context)
.colorScheme
.primary
.withOpacity(0.1);
}),
),
onPressed: () => Get.toNamed(
item['route'],
),
icon: Icon(
item['icon'],
size: 18,
color: Theme.of(context)
.colorScheme
.primary,
),
child: CircleAvatar(
radius: 22,
backgroundColor: Theme.of(context)
.colorScheme
.onInverseSurface,
child: Icon(
item['icon'],
size: 20,
color:
Theme.of(context).colorScheme.primary,
),
)),
const SizedBox(height: 6),
Text(item['name'],
style: const TextStyle(fontSize: 13))
],
),
),
const SizedBox(height: 6),
Text(item['name'],
style: const TextStyle(fontSize: 13))
],
),
onTap: () => Get.toNamed(item['route']),
);
}).toList(),
),

View File

@@ -133,6 +133,7 @@ class ChatItem extends StatelessWidget {
jsonDecode(content['content'])
.map((m) => m['text'] as String)
.join("\n"),
textAlign: TextAlign.center,
style: TextStyle(
letterSpacing: 0.6,
height: 5,

View File

@@ -3,6 +3,9 @@
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:hive/hive.dart';
import 'package:pilipala/pages/msg_feed_top/at_me/view.dart';
import 'package:pilipala/pages/msg_feed_top/reply_me/view.dart';
import 'package:pilipala/pages/msg_feed_top/like_me/view.dart';
import 'package:pilipala/pages/setting/pages/logs.dart';
import '../pages/about/index.dart';
@@ -139,6 +142,14 @@ class Routes {
// 私信详情
CustomGetPage(
name: '/whisperDetail', page: () => const WhisperDetailPage()),
// 回复我的
CustomGetPage(name: '/replyMe', page: () => const ReplyMePage()),
// @我的
CustomGetPage(name: '/atMe', page: () => const AtMePage()),
// 收到的赞
CustomGetPage(name: '/likeMe', page: () => const LikeMePage()),
// 系统消息
CustomGetPage(name: '/sysMsg', page: () => const WhisperPage()),
// 登录页面
CustomGetPage(name: '/loginPage', page: () => const LoginPage()),
// 用户动态