Merge pull request #4934 from 21pages/fix_chat

opt mobile chat page
This commit is contained in:
RustDesk 2023-07-10 13:27:09 +08:00 committed by GitHub
commit 8028b23086
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 222 additions and 76 deletions

View File

@ -2193,19 +2193,21 @@ Widget buildRemoteBlock({required Widget child, WhetherUseRemoteBlock? use}) {
)); ));
} }
Widget unreadMessageCountBuilder(RxInt? count) { Widget unreadMessageCountBuilder(RxInt? count,
{double? size, double? fontSize, double? marginLeft}) {
return Obx(() => Offstage( return Obx(() => Offstage(
offstage: !((count?.value ?? 0) > 0), offstage: !((count?.value ?? 0) > 0),
child: Container( child: Container(
width: 16, width: size ?? 16,
height: 16, height: size ?? 16,
decoration: BoxDecoration( decoration: BoxDecoration(
color: Colors.red, color: Colors.red,
shape: BoxShape.circle, shape: BoxShape.circle,
), ),
child: Center( child: Center(
child: Text("${count?.value ?? 0}", child: Text("${count?.value ?? 0}",
maxLines: 1, style: TextStyle(color: Colors.white, fontSize: 10)), maxLines: 1,
style: TextStyle(color: Colors.white, fontSize: fontSize ?? 10)),
), ),
).marginOnly(left: 4))); ).marginOnly(left: marginLeft ?? 4)));
} }

View File

@ -7,10 +7,15 @@ import 'package:provider/provider.dart';
import '../../mobile/pages/home_page.dart'; import '../../mobile/pages/home_page.dart';
enum ChatPageType {
mobileMain,
}
class ChatPage extends StatelessWidget implements PageShape { class ChatPage extends StatelessWidget implements PageShape {
late final ChatModel chatModel; late final ChatModel chatModel;
final ChatPageType? type;
ChatPage({ChatModel? chatModel}) { ChatPage({ChatModel? chatModel, this.type}) {
this.chatModel = chatModel ?? gFFI.chatModel; this.chatModel = chatModel ?? gFFI.chatModel;
} }
@ -22,17 +27,40 @@ class ChatPage extends StatelessWidget implements PageShape {
@override @override
final appBarActions = [ final appBarActions = [
PopupMenuButton<int>( PopupMenuButton<MessageKey>(
tooltip: "", tooltip: "",
icon: Icon(Icons.group), icon: Stack(
children: [
Icon(Icons.group),
Positioned(
top: 0,
right: 0,
child: unreadMessageCountBuilder(gFFI.chatModel.mobileUnreadSum,
marginLeft: 0, size: 12, fontSize: 8))
],
),
itemBuilder: (context) { itemBuilder: (context) {
// only mobile need [appBarActions], just bind gFFI.chatModel // only mobile need [appBarActions], just bind gFFI.chatModel
final chatModel = gFFI.chatModel; final chatModel = gFFI.chatModel;
return chatModel.messages.entries.map((entry) { return chatModel.messages.entries.map((entry) {
final id = entry.key; final id = entry.key;
final user = entry.value.chatUser; final user = entry.value.chatUser;
return PopupMenuItem<int>( final client = gFFI.serverModel.clients
child: Text("${user.firstName} ${user.id}"), .firstWhereOrNull((e) => e.id == id.connId);
return PopupMenuItem<MessageKey>(
child: Row(
children: [
Icon(
id.isOut
? Icons.call_made_rounded
: Icons.call_received_rounded,
color: MyTheme.accent)
.marginOnly(right: 6),
Text("${user.firstName} ${user.id}"),
if (client != null)
unreadMessageCountBuilder(client.unreadChatMessageCount)
],
),
value: id, value: id,
); );
}).toList(); }).toList();
@ -57,9 +85,14 @@ class ChatPage extends StatelessWidget implements PageShape {
final chat = DashChat( final chat = DashChat(
onSend: chatModel.send, onSend: chatModel.send,
currentUser: chatModel.me, currentUser: chatModel.me,
messages: messages: chatModel
chatModel.messages[chatModel.currentID]?.chatMessages ?? .messages[chatModel.currentKey]?.chatMessages ??
[], [],
readOnly: type == ChatPageType.mobileMain &&
(chatModel.currentKey.connId ==
ChatModel.clientModeID ||
gFFI.serverModel.clients.every(
(e) => e.id != chatModel.currentKey.connId)),
inputOptions: InputOptions( inputOptions: InputOptions(
focusNode: chatModel.inputNode, focusNode: chatModel.inputNode,
textController: chatModel.textController, textController: chatModel.textController,
@ -128,12 +161,18 @@ class ChatPage extends StatelessWidget implements PageShape {
return SelectionArea(child: chat); return SelectionArea(child: chat);
}), }),
desktopType == DesktopType.cm || desktopType == DesktopType.cm ||
chatModel.currentID == ChatModel.clientModeID type != ChatPageType.mobileMain ||
currentUser == null
? SizedBox.shrink() ? SizedBox.shrink()
: Padding( : Padding(
padding: EdgeInsets.all(12), padding: EdgeInsets.all(12),
child: Row( child: Row(
children: [ children: [
Icon(
chatModel.currentKey.isOut
? Icons.call_made_rounded
: Icons.call_received_rounded,
color: MyTheme.accent),
Icon(Icons.account_circle, color: MyTheme.accent80), Icon(Icons.account_circle, color: MyTheme.accent80),
SizedBox(width: 5), SizedBox(width: 5),
Text( Text(

View File

@ -32,7 +32,7 @@ class DraggableChatWindow extends StatelessWidget {
width: width, width: width,
height: height, height: height,
builder: (context, onPanUpdate) { builder: (context, onPanUpdate) {
return isIOS final child = isIOS
? ChatPage(chatModel: chatModel) ? ChatPage(chatModel: chatModel)
: Scaffold( : Scaffold(
resizeToAvoidBottomInset: false, resizeToAvoidBottomInset: false,
@ -44,6 +44,10 @@ class DraggableChatWindow extends StatelessWidget {
), ),
body: ChatPage(chatModel: chatModel), body: ChatPage(chatModel: chatModel),
); );
return Container(
decoration:
BoxDecoration(border: Border.all(color: MyTheme.border)),
child: child);
}); });
} }

View File

@ -100,14 +100,14 @@ class ConnectionManagerState extends State<ConnectionManager> {
gFFI.serverModel.tabController.onSelected = (client_id_str) { gFFI.serverModel.tabController.onSelected = (client_id_str) {
final client_id = int.tryParse(client_id_str); final client_id = int.tryParse(client_id_str);
if (client_id != null) { if (client_id != null) {
gFFI.chatModel.changeCurrentID(client_id);
final client = final client =
gFFI.serverModel.clients.firstWhereOrNull((e) => e.id == client_id); gFFI.serverModel.clients.firstWhereOrNull((e) => e.id == client_id);
if (client != null) { if (client != null) {
gFFI.chatModel.changeCurrentID(MessageKey(client.peerId, client.id));
if (client.unreadChatMessageCount.value > 0) { if (client.unreadChatMessageCount.value > 0) {
Future.delayed(Duration.zero, () { Future.delayed(Duration.zero, () {
client.unreadChatMessageCount.value = 0; client.unreadChatMessageCount.value = 0;
gFFI.chatModel.showChatPage(client.id); gFFI.chatModel.showChatPage(MessageKey(client.peerId, client.id));
}); });
} }
windowManager.setTitle(getWindowNameWithId(client.peerId)); windowManager.setTitle(getWindowNameWithId(client.peerId));
@ -444,7 +444,8 @@ class _CmHeaderState extends State<_CmHeader>
child: IconButton( child: IconButton(
onPressed: () => checkClickTime( onPressed: () => checkClickTime(
client.id, client.id,
() => gFFI.chatModel.toggleCMChatPage(client.id), () => gFFI.chatModel
.toggleCMChatPage(MessageKey(client.peerId, client.id)),
), ),
icon: SvgPicture.asset('assets/chat2.svg'), icon: SvgPicture.asset('assets/chat2.svg'),
splashRadius: kDesktopIconButtonSplashRadius, splashRadius: kDesktopIconButtonSplashRadius,

View File

@ -1453,7 +1453,8 @@ class _ChatMenuState extends State<_ChatMenu> {
initPos = Offset(pos.dx, pos.dy + _ToolbarTheme.dividerHeight); initPos = Offset(pos.dx, pos.dy + _ToolbarTheme.dividerHeight);
} }
widget.ffi.chatModel.changeCurrentID(ChatModel.clientModeID); widget.ffi.chatModel.changeCurrentID(
MessageKey(widget.ffi.id, ChatModel.clientModeID));
widget.ffi.chatModel.toggleChatOverlay(chatInitPos: initPos); widget.ffi.chatModel.toggleChatOverlay(chatInitPos: initPos);
}); });
} }

View File

@ -37,6 +37,7 @@ class ConnectionPage extends StatefulWidget implements PageShape {
class _ConnectionPageState extends State<ConnectionPage> { class _ConnectionPageState extends State<ConnectionPage> {
/// Controller for the id input bar. /// Controller for the id input bar.
final _idController = IDTextEditingController(); final _idController = IDTextEditingController();
final RxBool _idEmpty = true.obs;
/// Update url. If it's not null, means an update is available. /// Update url. If it's not null, means an update is available.
var _updateUrl = ''; var _updateUrl = '';
@ -60,6 +61,10 @@ class _ConnectionPageState extends State<ConnectionPage> {
if (_updateUrl.isNotEmpty) setState(() {}); if (_updateUrl.isNotEmpty) setState(() {});
}); });
} }
_idController.addListener(() {
_idEmpty.value = _idController.text.isEmpty;
});
} }
@override @override
@ -158,6 +163,14 @@ class _ConnectionPageState extends State<ConnectionPage> {
), ),
), ),
), ),
Obx(() => Offstage(
offstage: _idEmpty.value,
child: IconButton(
onPressed: () {
_idController.clear();
},
icon: Icon(Icons.clear, color: MyTheme.darkGray)),
)),
SizedBox( SizedBox(
width: 60, width: 60,
height: 60, height: 60,

View File

@ -22,6 +22,7 @@ class HomePage extends StatefulWidget {
class _HomePageState extends State<HomePage> { class _HomePageState extends State<HomePage> {
var _selectedIndex = 0; var _selectedIndex = 0;
int get selectedIndex => _selectedIndex;
final List<PageShape> _pages = []; final List<PageShape> _pages = [];
void refreshPages() { void refreshPages() {
@ -40,7 +41,7 @@ class _HomePageState extends State<HomePage> {
_pages.clear(); _pages.clear();
_pages.add(ConnectionPage()); _pages.add(ConnectionPage());
if (isAndroid) { if (isAndroid) {
_pages.addAll([ChatPage(), ServerPage()]); _pages.addAll([ChatPage(type: ChatPageType.mobileMain), ServerPage()]);
} }
_pages.add(SettingsPage()); _pages.add(SettingsPage());
} }
@ -82,6 +83,8 @@ class _HomePageState extends State<HomePage> {
gFFI.chatModel.hideChatWindowOverlay(); gFFI.chatModel.hideChatWindowOverlay();
} }
_selectedIndex = index; _selectedIndex = index;
gFFI.chatModel
.mobileClearClientUnread(gFFI.chatModel.currentKey.connId);
}), }),
), ),
body: _pages.elementAt(_selectedIndex), body: _pages.elementAt(_selectedIndex),

View File

@ -351,8 +351,8 @@ class _RemotePageState extends State<RemotePage> {
color: Colors.white, color: Colors.white,
icon: Icon(Icons.message), icon: Icon(Icons.message),
onPressed: () { onPressed: () {
gFFI.chatModel gFFI.chatModel.changeCurrentID(MessageKey(
.changeCurrentID(ChatModel.clientModeID); widget.id, ChatModel.clientModeID));
gFFI.chatModel.toggleChatOverlay(); gFFI.chatModel.toggleChatOverlay();
}, },
) )

View File

@ -3,6 +3,7 @@ import 'dart:async';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:flutter_hbb/mobile/widgets/dialog.dart'; import 'package:flutter_hbb/mobile/widgets/dialog.dart';
import 'package:flutter_hbb/models/chat_model.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@ -419,7 +420,8 @@ class ConnectionManager extends StatelessWidget {
? const SizedBox.shrink() ? const SizedBox.shrink()
: IconButton( : IconButton(
onPressed: () { onPressed: () {
gFFI.chatModel.changeCurrentID(client.id); gFFI.chatModel.changeCurrentID(
MessageKey(client.peerId, client.id));
final bar = navigationBarKey.currentWidget; final bar = navigationBarKey.currentWidget;
if (bar != null) { if (bar != null) {
bar as BottomNavigationBar; bar as BottomNavigationBar;

View File

@ -7,6 +7,7 @@ import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:flutter_hbb/common/shared_state.dart'; import 'package:flutter_hbb/common/shared_state.dart';
import 'package:flutter_hbb/desktop/widgets/tabbar_widget.dart'; import 'package:flutter_hbb/desktop/widgets/tabbar_widget.dart';
import 'package:flutter_hbb/mobile/pages/home_page.dart';
import 'package:flutter_hbb/models/platform_model.dart'; import 'package:flutter_hbb/models/platform_model.dart';
import 'package:flutter_hbb/models/state_model.dart'; import 'package:flutter_hbb/models/state_model.dart';
import 'package:get/get_rx/src/rx_types/rx_types.dart'; import 'package:get/get_rx/src/rx_types/rx_types.dart';
@ -20,6 +21,24 @@ import '../common/widgets/overlay.dart';
import '../main.dart'; import '../main.dart';
import 'model.dart'; import 'model.dart';
class MessageKey {
final String peerId;
final int connId;
bool get isOut => connId != ChatModel.clientModeID;
MessageKey(this.peerId, this.connId);
@override
bool operator ==(other) {
return other is MessageKey &&
other.peerId == peerId &&
other.isOut == isOut;
}
@override
int get hashCode => peerId.hashCode ^ isOut.hashCode;
}
class MessageBody { class MessageBody {
ChatUser chatUser; ChatUser chatUser;
List<ChatMessage> chatMessages; List<ChatMessage> chatMessages;
@ -49,6 +68,7 @@ class ChatModel with ChangeNotifier {
Rx<VoiceCallStatus> get voiceCallStatus => _voiceCallStatus; Rx<VoiceCallStatus> get voiceCallStatus => _voiceCallStatus;
TextEditingController textController = TextEditingController(); TextEditingController textController = TextEditingController();
RxInt mobileUnreadSum = 0.obs;
@override @override
void dispose() { void dispose() {
@ -61,15 +81,14 @@ class ChatModel with ChangeNotifier {
firstName: translate("Me"), firstName: translate("Me"),
); );
late final Map<int, MessageBody> _messages = {}..[clientModeID] = late final Map<MessageKey, MessageBody> _messages = {};
MessageBody(me, []);
var _currentID = clientModeID; MessageKey _currentKey = MessageKey('', clientModeID);
late bool _isShowCMChatPage = false; late bool _isShowCMChatPage = false;
Map<int, MessageBody> get messages => _messages; Map<MessageKey, MessageBody> get messages => _messages;
int get currentID => _currentID; MessageKey get currentKey => _currentKey;
bool get isShowCMChatPage => _isShowCMChatPage; bool get isShowCMChatPage => _isShowCMChatPage;
@ -119,15 +138,7 @@ class ChatModel with ChangeNotifier {
); );
} }
ChatUser get currentUser { ChatUser? get currentUser => _messages[_currentKey]?.chatUser;
final user = messages[currentID]?.chatUser;
if (user == null) {
_currentID = clientModeID;
return me;
} else {
return user;
}
}
showChatIconOverlay({Offset offset = const Offset(200, 50)}) { showChatIconOverlay({Offset offset = const Offset(200, 50)}) {
if (chatIconOverlayEntry != null) { if (chatIconOverlayEntry != null) {
@ -233,11 +244,11 @@ class ChatModel with ChangeNotifier {
} }
} }
showChatPage(int id) async { showChatPage(MessageKey key) async {
if (isDesktop) { if (isDesktop) {
if (isConnManager) { if (isConnManager) {
if (!_isShowCMChatPage) { if (!_isShowCMChatPage) {
await toggleCMChatPage(id); await toggleCMChatPage(key);
} }
} else { } else {
if (_isChatOverlayHide()) { if (_isChatOverlayHide()) {
@ -245,7 +256,7 @@ class ChatModel with ChangeNotifier {
} }
} }
} else { } else {
if (id == clientModeID) { if (key.connId == clientModeID) {
if (_isChatOverlayHide()) { if (_isChatOverlayHide()) {
await toggleChatOverlay(); await toggleChatOverlay();
} }
@ -253,9 +264,9 @@ class ChatModel with ChangeNotifier {
} }
} }
toggleCMChatPage(int id) async { toggleCMChatPage(MessageKey key) async {
if (gFFI.chatModel.currentID != id) { if (gFFI.chatModel.currentKey != key) {
gFFI.chatModel.changeCurrentID(id); gFFI.chatModel.changeCurrentID(key);
} }
if (_isShowCMChatPage) { if (_isShowCMChatPage) {
_isShowCMChatPage = !_isShowCMChatPage; _isShowCMChatPage = !_isShowCMChatPage;
@ -273,25 +284,29 @@ class ChatModel with ChangeNotifier {
} }
} }
changeCurrentID(int id) { changeCurrentID(MessageKey key) {
if (_messages.containsKey(id)) { updateConnIdOfKey(key);
_currentID = id; if (_messages.containsKey(key)) {
_currentKey = key;
notifyListeners(); notifyListeners();
} else { } else {
final client = parent.target?.serverModel.clients String? peerName;
.firstWhere((client) => client.id == id); if (key.connId == clientModeID) {
if (client == null) { peerName = parent.target?.ffiModel.pi.username;
return debugPrint( } else {
"Failed to changeCurrentID,remote user doesn't exist"); peerName = parent.target?.serverModel.clients
.firstWhereOrNull((client) => client.peerId == key.peerId)
?.name;
} }
final chatUser = ChatUser( final chatUser = ChatUser(
id: client.peerId, id: key.peerId,
firstName: client.name, firstName: peerName,
); );
_messages[id] = MessageBody(chatUser, []); _messages[key] = MessageBody(chatUser, []);
_currentID = id; _currentKey = key;
notifyListeners(); notifyListeners();
} }
mobileClearClientUnread(key.connId);
} }
receive(int id, String text) async { receive(int id, String text) async {
@ -304,23 +319,33 @@ class ChatModel with ChangeNotifier {
if (desktopType == DesktopType.cm) { if (desktopType == DesktopType.cm) {
await showCmWindow(); await showCmWindow();
} }
String? peerId;
if (id == clientModeID) {
peerId = session.id;
} else {
peerId = session.serverModel.clients
.firstWhereOrNull((e) => e.id == id)
?.peerId;
}
if (peerId == null) {
debugPrint("Failed to receive msg, peerId is null");
return;
}
final messagekey = MessageKey(peerId, id);
// mobile: first message show overlay icon // mobile: first message show overlay icon
if (!isDesktop && chatIconOverlayEntry == null) { if (!isDesktop && chatIconOverlayEntry == null && id == clientModeID) {
showChatIconOverlay(); showChatIconOverlay();
} }
// show chat page // show chat page
await showChatPage(id); await showChatPage(messagekey);
int toId = currentID;
late final ChatUser chatUser; late final ChatUser chatUser;
if (id == clientModeID) { if (id == clientModeID) {
chatUser = ChatUser( chatUser = ChatUser(
firstName: session.ffiModel.pi.username, firstName: session.ffiModel.pi.username,
id: session.id, id: peerId,
); );
toId = id;
if (isDesktop) { if (isDesktop) {
if (Get.isRegistered<DesktopTabController>()) { if (Get.isRegistered<DesktopTabController>()) {
@ -339,14 +364,18 @@ class ChatModel with ChangeNotifier {
} }
} else { } else {
if (notSelected) { if (notSelected) {
UnreadChatCountState.find(session.id).value += 1; UnreadChatCountState.find(peerId).value += 1;
} }
} }
} }
} }
} else { } else {
final client = final client = session.serverModel.clients
session.serverModel.clients.firstWhere((client) => client.id == id); .firstWhereOrNull((client) => client.id == id);
if (client == null) {
debugPrint("Failed to receive msg, client is null");
return;
}
if (isDesktop) { if (isDesktop) {
window_on_top(null); window_on_top(null);
// disable auto jumpTo other tab when hasFocus, and mark unread message // disable auto jumpTo other tab when hasFocus, and mark unread message
@ -356,20 +385,23 @@ class ChatModel with ChangeNotifier {
client.unreadChatMessageCount.value += 1; client.unreadChatMessageCount.value += 1;
} else { } else {
parent.target?.serverModel.jumpTo(id); parent.target?.serverModel.jumpTo(id);
toId = id;
} }
} else { } else {
toId = id; if (HomePage.homeKey.currentState?.selectedIndex != 1 ||
_currentKey.peerId != client.peerId) {
client.unreadChatMessageCount.value += 1;
mobileUpdateUnreadSum();
}
} }
chatUser = ChatUser(id: client.peerId, firstName: client.name); chatUser = ChatUser(id: client.peerId, firstName: client.name);
} }
insertMessage(messagekey,
if (!_messages.containsKey(id)) {
_messages[id] = MessageBody(chatUser, []);
}
_messages[id]!.insert(
ChatMessage(text: text, user: chatUser, createdAt: DateTime.now())); ChatMessage(text: text, user: chatUser, createdAt: DateTime.now()));
_currentID = toId; if (id == clientModeID || _currentKey.peerId.isEmpty) {
// Invalid
_currentKey = messagekey;
mobileClearClientUnread(messagekey.connId);
}
notifyListeners(); notifyListeners();
} }
@ -379,17 +411,63 @@ class ChatModel with ChangeNotifier {
return; return;
} }
message.text = trimmedText; message.text = trimmedText;
_messages[_currentID]?.insert(message); insertMessage(_currentKey, message);
if (_currentID == clientModeID && parent.target != null) { if (_currentKey.connId == clientModeID && parent.target != null) {
bind.sessionSendChat(sessionId: sessionId, text: message.text); bind.sessionSendChat(sessionId: sessionId, text: message.text);
} else { } else {
bind.cmSendChat(connId: _currentID, msg: message.text); bind.cmSendChat(connId: _currentKey.connId, msg: message.text);
} }
notifyListeners(); notifyListeners();
inputNode.requestFocus(); inputNode.requestFocus();
} }
insertMessage(MessageKey key, ChatMessage message) {
updateConnIdOfKey(key);
if (!_messages.containsKey(key)) {
_messages[key] = MessageBody(message.user, []);
}
_messages[key]?.insert(message);
}
updateConnIdOfKey(MessageKey key) {
if (_messages.keys
.toList()
.firstWhereOrNull((e) => e == key && e.connId != key.connId) !=
null) {
final old = _messages.remove(key);
if (old != null) {
_messages[key] = old;
}
}
if (_currentKey == key) {
_currentKey = key; // hash != assign
}
}
void mobileUpdateUnreadSum() {
if (!isMobile) return;
var sum = 0;
parent.target?.serverModel.clients
.map((e) => sum += e.unreadChatMessageCount.value)
.toList();
Future.delayed(Duration.zero, () {
mobileUnreadSum.value = sum;
});
}
void mobileClearClientUnread(int id) {
if (!isMobile) return;
final client = parent.target?.serverModel.clients
.firstWhereOrNull((client) => client.id == id);
if (client != null) {
Future.delayed(Duration.zero, () {
client.unreadChatMessageCount.value = 0;
mobileUpdateUnreadSum();
});
}
}
close() { close() {
hideChatIconOverlay(); hideChatIconOverlay();
hideChatWindowOverlay(); hideChatWindowOverlay();

View File

@ -5,6 +5,7 @@ import 'dart:io';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/consts.dart';
import 'package:flutter_hbb/main.dart'; import 'package:flutter_hbb/main.dart';
import 'package:flutter_hbb/models/chat_model.dart';
import 'package:flutter_hbb/models/platform_model.dart'; import 'package:flutter_hbb/models/platform_model.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:wakelock/wakelock.dart'; import 'package:wakelock/wakelock.dart';
@ -474,6 +475,8 @@ class ServerModel with ChangeNotifier {
cmHiddenTimer = null; cmHiddenTimer = null;
}); });
} }
parent.target?.chatModel
.updateConnIdOfKey(MessageKey(client.peerId, client.id));
} }
void showLoginDialog(Client client) { void showLoginDialog(Client client) {