commit
8028b23086
@ -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)));
|
||||||
}
|
}
|
||||||
|
@ -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(
|
||||||
|
@ -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);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
@ -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);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
@ -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),
|
||||||
|
@ -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();
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
|
@ -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) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user