|
|
@ -67,11 +67,15 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
List<Message> messages = []; |
|
|
|
List<Message> messages = []; |
|
|
|
|
|
|
|
|
|
|
|
loadMessageList() async { |
|
|
|
loadMessageList() async { |
|
|
|
|
|
|
|
ImUser imUser = await hxDatabase.queryImUserById(_toUser.mid); |
|
|
|
|
|
|
|
if (imUser == null) { |
|
|
|
|
|
|
|
await hxDatabase.insertOrUpdateImUser(_toUser.toJson()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
selfUserId = (await SharedPreferences.getInstance()).getString("userId"); |
|
|
|
selfUserId = (await SharedPreferences.getInstance()).getString("userId"); |
|
|
|
// unread msg 2 read state |
|
|
|
// unread msg 2 read state |
|
|
|
await hxDatabase.readMessage(selfUserId, _toUser.mid); |
|
|
|
await hxDatabase.readMessage(selfUserId, _toUser.mid); |
|
|
|
messages = await hxDatabase.queryUList(_toUser.mid); |
|
|
|
messages = await hxDatabase.queryUList(_toUser.mid, pageSize: 100); |
|
|
|
|
|
|
|
|
|
|
|
socketClient.addCallback(_toUser.mid, (Message message) { |
|
|
|
socketClient.addCallback(_toUser.mid, (Message message) { |
|
|
|
messages.add(message); |
|
|
|
messages.add(message); |
|
|
@ -98,7 +102,6 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
commentFocus.addListener(_focusNodeListener); |
|
|
|
commentFocus.addListener(_focusNodeListener); |
|
|
|
|
|
|
|
|
|
|
|
loadMessageList(); |
|
|
|
loadMessageList(); |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void jumpToBottom() { |
|
|
|
void jumpToBottom() { |
|
|
@ -259,21 +262,21 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
leading: true, |
|
|
|
leading: true, |
|
|
|
leadingColor: Colors.black, |
|
|
|
leadingColor: Colors.black, |
|
|
|
action: GestureDetector( |
|
|
|
action: GestureDetector( |
|
|
|
behavior: HitTestBehavior.opaque, |
|
|
|
behavior: HitTestBehavior.opaque, |
|
|
|
onTap: () { |
|
|
|
onTap: () { |
|
|
|
setState(() { |
|
|
|
setState(() { |
|
|
|
Navigator.of(context).pushNamed('/router/chat_setting'); |
|
|
|
Navigator.of(context).pushNamed('/router/chat_setting'); |
|
|
|
copyIndex = 0; |
|
|
|
copyIndex = 0; |
|
|
|
}); |
|
|
|
}); |
|
|
|
}, |
|
|
|
}, |
|
|
|
child: Container( |
|
|
|
child: Container( |
|
|
|
alignment: Alignment.center, |
|
|
|
alignment: Alignment.center, |
|
|
|
child: Icon( |
|
|
|
child: Icon( |
|
|
|
Icons.more_horiz, |
|
|
|
Icons.more_horiz, |
|
|
|
color: Colors.black, |
|
|
|
color: Colors.black, |
|
|
|
size: 30, |
|
|
|
size: 30, |
|
|
|
), |
|
|
|
|
|
|
|
), |
|
|
|
), |
|
|
|
|
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
body: Container( |
|
|
|
body: Container( |
|
|
@ -286,17 +289,17 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
child: Column( |
|
|
|
child: Column( |
|
|
|
children: [ |
|
|
|
children: [ |
|
|
|
GestureDetector( |
|
|
|
GestureDetector( |
|
|
|
behavior: HitTestBehavior.translucent, |
|
|
|
behavior: HitTestBehavior.translucent, |
|
|
|
onTap: () { |
|
|
|
onTap: () { |
|
|
|
setState(() { |
|
|
|
setState(() { |
|
|
|
emojiShowing = false; |
|
|
|
emojiShowing = false; |
|
|
|
isKeyBoardShow = emojiShowing; |
|
|
|
isKeyBoardShow = emojiShowing; |
|
|
|
moreShow = false; |
|
|
|
moreShow = false; |
|
|
|
isKeyBoardShow = moreShow; |
|
|
|
isKeyBoardShow = moreShow; |
|
|
|
FocusScope.of(context).requestFocus(FocusNode()); |
|
|
|
FocusScope.of(context).requestFocus(FocusNode()); |
|
|
|
}); |
|
|
|
}); |
|
|
|
}, |
|
|
|
}, |
|
|
|
child: chatDetailsList(), |
|
|
|
child: chatDetailsList(), |
|
|
|
), |
|
|
|
), |
|
|
|
], |
|
|
|
], |
|
|
|
), |
|
|
|
), |
|
|
@ -344,7 +347,7 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
child: Column( |
|
|
|
child: Column( |
|
|
|
children: [ |
|
|
|
children: [ |
|
|
|
Text( |
|
|
|
Text( |
|
|
|
"3月08日 上午 12:10", |
|
|
|
"${DateTime.fromMillisecondsSinceEpoch(int.parse(message.time))}", |
|
|
|
textAlign: TextAlign.center, |
|
|
|
textAlign: TextAlign.center, |
|
|
|
style: TextStyle( |
|
|
|
style: TextStyle( |
|
|
|
color: Color(0xFFA29E9E), |
|
|
|
color: Color(0xFFA29E9E), |
|
|
@ -352,23 +355,23 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
fontWeight: MyFontWeight.regular, |
|
|
|
fontWeight: MyFontWeight.regular, |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
if (messages.indexOf(message) == (messages.length - 1)) |
|
|
|
// if (messages.indexOf(message) == (messages.length - 1)) |
|
|
|
Padding( |
|
|
|
// Padding( |
|
|
|
padding: EdgeInsets.only(top: 10.h, bottom: 24.h), |
|
|
|
// padding: EdgeInsets.only(top: 10.h, bottom: 24.h), |
|
|
|
child: Text( |
|
|
|
// child: Text( |
|
|
|
"在对方未回复或关注你之前,你只能发送一条信息", |
|
|
|
// "在对方未回复或关注你之前,你只能发送一条信息", |
|
|
|
textAlign: TextAlign.center, |
|
|
|
// textAlign: TextAlign.center, |
|
|
|
style: TextStyle( |
|
|
|
// style: TextStyle( |
|
|
|
color: Color(0xFFA29E9E), |
|
|
|
// color: Color(0xFFA29E9E), |
|
|
|
fontSize: 10.sp, |
|
|
|
// fontSize: 10.sp, |
|
|
|
fontWeight: MyFontWeight.regular, |
|
|
|
// fontWeight: MyFontWeight.regular, |
|
|
|
), |
|
|
|
// ), |
|
|
|
), |
|
|
|
// ), |
|
|
|
), |
|
|
|
// ), |
|
|
|
if (messages.indexOf(message) == (messages.length - 1)) |
|
|
|
// if (messages.indexOf(message) == (messages.length - 1)) |
|
|
|
SizedBox( |
|
|
|
// SizedBox( |
|
|
|
height: 16.h, |
|
|
|
// height: 16.h, |
|
|
|
), |
|
|
|
// ), |
|
|
|
if (copyIndex == 1) |
|
|
|
if (copyIndex == 1) |
|
|
|
Stack( |
|
|
|
Stack( |
|
|
|
alignment: Alignment.bottomCenter, |
|
|
|
alignment: Alignment.bottomCenter, |
|
|
@ -382,7 +385,8 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
), |
|
|
|
), |
|
|
|
padding: EdgeInsets.symmetric( |
|
|
|
padding: EdgeInsets.symmetric( |
|
|
|
horizontal: 32.w, vertical: 7.5.h, |
|
|
|
horizontal: 32.w, |
|
|
|
|
|
|
|
vertical: 7.5.h, |
|
|
|
), |
|
|
|
), |
|
|
|
child: Row( |
|
|
|
child: Row( |
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween, |
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween, |
|
|
@ -454,13 +458,17 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
), |
|
|
|
), |
|
|
|
|
|
|
|
|
|
|
|
/// not self |
|
|
|
/// not self |
|
|
|
if (!isSelf && !isText) |
|
|
|
if (!isSelf && isText) |
|
|
|
|
|
|
|
SizedBox( |
|
|
|
|
|
|
|
height: 10.h, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
if (!isSelf && isText) |
|
|
|
Padding( |
|
|
|
Padding( |
|
|
|
padding: EdgeInsets.only(left: 17.w, right: 39.w), |
|
|
|
padding: EdgeInsets.only(left: 17.w, right: 39.w), |
|
|
|
child: Row( |
|
|
|
child: Row( |
|
|
|
children: [ |
|
|
|
children: [ |
|
|
|
MImage( |
|
|
|
MImage( |
|
|
|
_toUser.avatar, |
|
|
|
_toUser.avatar, |
|
|
|
isCircle: true, |
|
|
|
isCircle: true, |
|
|
|
height: 44.h, |
|
|
|
height: 44.h, |
|
|
|
width: 44.h, |
|
|
|
width: 44.h, |
|
|
@ -472,45 +480,48 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
width: 12.w, |
|
|
|
width: 12.w, |
|
|
|
), |
|
|
|
), |
|
|
|
Expanded( |
|
|
|
Expanded( |
|
|
|
|
|
|
|
child: Container( |
|
|
|
|
|
|
|
alignment: Alignment.centerLeft, |
|
|
|
child: Container( |
|
|
|
child: Container( |
|
|
|
decoration: BoxDecoration( |
|
|
|
decoration: BoxDecoration( |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
color: Color(0xFFFFFFFF), |
|
|
|
color: Color(0xFFFFFFFF), |
|
|
|
boxShadow: [ |
|
|
|
boxShadow: [ |
|
|
|
BoxShadow( |
|
|
|
BoxShadow( |
|
|
|
color: Color(0xFFA8A3A3).withAlpha(12), |
|
|
|
color: Color(0xFFA8A3A3).withAlpha(12), |
|
|
|
offset: Offset(0, 4), |
|
|
|
offset: Offset(0, 4), |
|
|
|
blurRadius: 4, |
|
|
|
blurRadius: 4, |
|
|
|
spreadRadius: 0, |
|
|
|
spreadRadius: 0, |
|
|
|
) |
|
|
|
|
|
|
|
], |
|
|
|
|
|
|
|
), |
|
|
|
), |
|
|
|
padding: EdgeInsets.symmetric( |
|
|
|
], |
|
|
|
vertical: 8.h, horizontal: 12.w), |
|
|
|
), |
|
|
|
child: GestureDetector( |
|
|
|
padding: EdgeInsets.symmetric( |
|
|
|
onLongPress: () { |
|
|
|
vertical: 8.h, |
|
|
|
setState(() { |
|
|
|
horizontal: 12.w, |
|
|
|
copyIndex = 1; |
|
|
|
), |
|
|
|
}); |
|
|
|
child: GestureDetector( |
|
|
|
}, |
|
|
|
onLongPress: () { |
|
|
|
child: Text( |
|
|
|
setState(() { |
|
|
|
tex = message.content, |
|
|
|
copyIndex = 1; |
|
|
|
textAlign: TextAlign.left, |
|
|
|
}); |
|
|
|
style: TextStyle( |
|
|
|
}, |
|
|
|
height: 1.2.h, |
|
|
|
child: Text( |
|
|
|
color: Color(0XFF0D0D0D), |
|
|
|
tex = message.content, |
|
|
|
fontSize: 17.sp, |
|
|
|
textAlign: TextAlign.left, |
|
|
|
fontWeight: MyFontWeight.medium, |
|
|
|
style: TextStyle( |
|
|
|
), |
|
|
|
height: 1.2.h, |
|
|
|
), |
|
|
|
color: Color(0XFF0D0D0D), |
|
|
|
))), |
|
|
|
fontSize: 17.sp, |
|
|
|
|
|
|
|
fontWeight: MyFontWeight.medium, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
),), |
|
|
|
|
|
|
|
), |
|
|
|
], |
|
|
|
], |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
), |
|
|
|
if (!isSelf && isText) |
|
|
|
|
|
|
|
SizedBox( |
|
|
|
|
|
|
|
height: 40.h, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
if (copyIndex == 1) |
|
|
|
if (copyIndex == 1) |
|
|
|
Stack( |
|
|
|
Stack( |
|
|
|
alignment: Alignment.bottomCenter, |
|
|
|
alignment: Alignment.bottomCenter, |
|
|
@ -595,6 +606,10 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
), |
|
|
|
), |
|
|
|
|
|
|
|
|
|
|
|
/// self |
|
|
|
/// self |
|
|
|
|
|
|
|
if (isSelf && isText) |
|
|
|
|
|
|
|
SizedBox( |
|
|
|
|
|
|
|
height: 10.h, |
|
|
|
|
|
|
|
), |
|
|
|
if (isSelf && isText) |
|
|
|
if (isSelf && isText) |
|
|
|
Padding( |
|
|
|
Padding( |
|
|
|
padding: EdgeInsets.only(left: 36.w, right: 16.w), |
|
|
|
padding: EdgeInsets.only(left: 36.w, right: 16.w), |
|
|
@ -630,9 +645,19 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
decoration: BoxDecoration( |
|
|
|
decoration: BoxDecoration( |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
borderRadius: BorderRadius.circular(6), |
|
|
|
color: Color(0xFF32A060), |
|
|
|
color: Color(0xFF32A060), |
|
|
|
|
|
|
|
boxShadow: [ |
|
|
|
|
|
|
|
BoxShadow( |
|
|
|
|
|
|
|
color: Color(0xFFA8A3A3).withAlpha(12), |
|
|
|
|
|
|
|
offset: Offset(0, 4), |
|
|
|
|
|
|
|
blurRadius: 4, |
|
|
|
|
|
|
|
spreadRadius: 0, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
], |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
padding: EdgeInsets.symmetric( |
|
|
|
|
|
|
|
vertical: 8.h, |
|
|
|
|
|
|
|
horizontal: 12.w, |
|
|
|
), |
|
|
|
), |
|
|
|
padding: |
|
|
|
|
|
|
|
EdgeInsets.symmetric(vertical: 8.h, horizontal: 12.w), |
|
|
|
|
|
|
|
child: GestureDetector( |
|
|
|
child: GestureDetector( |
|
|
|
onLongPress: () { |
|
|
|
onLongPress: () { |
|
|
|
setState(() { |
|
|
|
setState(() { |
|
|
@ -732,19 +757,19 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
), |
|
|
|
), |
|
|
|
|
|
|
|
|
|
|
|
/// no reply long time |
|
|
|
/// no reply long time |
|
|
|
if (messages.indexOf(message) == 0) |
|
|
|
// if (messages.indexOf(message) == 0) |
|
|
|
Padding( |
|
|
|
// Padding( |
|
|
|
padding: EdgeInsets.only(left: 17.w, right: 17.w, top: 24.h), |
|
|
|
// padding: EdgeInsets.only(left: 17.w, right: 17.w, top: 24.h), |
|
|
|
child: Text( |
|
|
|
// child: Text( |
|
|
|
"由于对方没有回复你,你只能发送一条消息,需要对方关注或回复后才能恢复正常聊天", |
|
|
|
// "由于对方没有回复你,你只能发送一条消息,需要对方关注或回复后才能恢复正常聊天", |
|
|
|
textAlign: TextAlign.center, |
|
|
|
// textAlign: TextAlign.center, |
|
|
|
style: TextStyle( |
|
|
|
// style: TextStyle( |
|
|
|
color: Color(0xFFA29E9E), |
|
|
|
// color: Color(0xFFA29E9E), |
|
|
|
fontSize: 10.sp, |
|
|
|
// fontSize: 10.sp, |
|
|
|
fontWeight: MyFontWeight.regular, |
|
|
|
// fontWeight: MyFontWeight.regular, |
|
|
|
), |
|
|
|
// ), |
|
|
|
), |
|
|
|
// ), |
|
|
|
), |
|
|
|
// ), |
|
|
|
], |
|
|
|
], |
|
|
|
), |
|
|
|
), |
|
|
|
); |
|
|
|
); |
|
|
@ -784,14 +809,21 @@ class _ChatDetailsPage extends State<ChatDetailsPage> |
|
|
|
if (commentText.trim() == "") { |
|
|
|
if (commentText.trim() == "") { |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
socketClient.sendMessage(_toUser.mid, commentText).then((value) { |
|
|
|
socketClient |
|
|
|
|
|
|
|
.sendMessage(_toUser.mid, commentText) |
|
|
|
|
|
|
|
.then((value) { |
|
|
|
Message message = value; |
|
|
|
Message message = value; |
|
|
|
messages.insert(0, message); |
|
|
|
messages.insert(0, message); |
|
|
|
chatController.clear(); |
|
|
|
chatController.clear(); |
|
|
|
if (mounted) |
|
|
|
if (mounted) setState(() {}); |
|
|
|
setState(() {}); |
|
|
|
|
|
|
|
|
|
|
|
if (scrollController.position != null) { |
|
|
|
|
|
|
|
double offset = scrollController.position.maxScrollExtent; |
|
|
|
|
|
|
|
debugPrint("offset: $offset"); |
|
|
|
|
|
|
|
scrollController.animateTo(offset + 50, duration: const Duration(milliseconds: 500), curve: Curves.easeIn); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
}); |
|
|
|
}); |
|
|
|
// widget.queryMemberComment(commentText); |
|
|
|
|
|
|
|
}, |
|
|
|
}, |
|
|
|
maxLines: 8, |
|
|
|
maxLines: 8, |
|
|
|
minLines: 1, |
|
|
|
minLines: 1, |
|
|
|