Skip to content

Commit

Permalink
TW-1827: update online status correctly
Browse files Browse the repository at this point in the history
  • Loading branch information
sherlockvn committed Jun 20, 2024
1 parent 8de41a9 commit 8b4d523
Show file tree
Hide file tree
Showing 3 changed files with 105 additions and 36 deletions.
40 changes: 40 additions & 0 deletions lib/pages/chat/chat.dart
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import 'dart:async';
import 'dart:io';
import 'package:connectivity_plus/connectivity_plus.dart';
import 'package:fluffychat/pages/chat/chat_actions.dart';
import 'package:fluffychat/pages/chat/events/message_content_mixin.dart';
import 'package:fluffychat/presentation/extensions/event_update_extension.dart';
Expand All @@ -12,6 +13,7 @@ import 'package:fluffychat/presentation/model/chat/view_event_list_ui_state.dart
import 'package:fluffychat/utils/extension/basic_event_extension.dart';
import 'package:fluffychat/utils/extension/event_status_custom_extension.dart';
import 'package:fluffychat/utils/matrix_sdk_extensions/filtered_timeline_extension.dart';
import 'package:fluffychat/utils/room_status_extension.dart';
import 'package:fluffychat/widgets/context_menu/context_menu_action.dart';
import 'package:fluffychat/widgets/mixins/popup_menu_widget_style.dart';
import 'package:fluffychat/widgets/mixins/twake_context_menu_mixin.dart';
Expand Down Expand Up @@ -258,6 +260,42 @@ class ChatController extends State<Chat>
SuggestionsController<Map<String, String?>> suggestionsController =
SuggestionsController();

ValueNotifier<CachedPresence?> cachedPresenceNotifier = ValueNotifier(null);

final StreamController<ConnectivityResult>
connectivityResultStreamController = StreamController.broadcast();

StreamController<CachedPresence> cachedPresenceStreamController =
StreamController.broadcast();

Future<void> initCachedPresence() async {
cachedPresenceNotifier.value = room?.directChatPresence;
if (room?.directChatMatrixID != null) {
Matrix.of(context).client.onlatestPresenceChanged.stream.listen((event) {
if (event.userid == room!.directChatMatrixID) {
Logs().v(
'onlatestPresenceChanged: ${event.presence}, ${event.lastActiveTimestamp}',
);
cachedPresenceStreamController.add(event);
}
});
try {
final getPresenceResponse = await client.getPresence(
room!.directChatMatrixID!,
);

cachedPresenceNotifier.value = CachedPresence.fromPresenceResponse(
getPresenceResponse,
room!.directChatMatrixID!,
);
} catch (e) {
Logs().e('Failed to get presence', e);
cachedPresenceNotifier.value =
CachedPresence.neverSeen(room!.directChatMatrixID!);
}
}
}

bool isUnpinEvent(Event event) =>
room?.pinnedEventIds
.firstWhereOrNull((eventId) => eventId == event.eventId) !=
Expand Down Expand Up @@ -1923,6 +1961,7 @@ class ChatController extends State<Chat>
}
_handleReceivedShareFiles();
_listenRoomUpdateEvent();
initCachedPresence();
});
}

Expand Down Expand Up @@ -1968,6 +2007,7 @@ class ChatController extends State<Chat>
keyboardVisibilitySubscription?.cancel();
InViewNotifierListCustom.of(context)?.dispose();
replyEventNotifier.dispose();
cachedPresenceStreamController.close();
super.dispose();
}

Expand Down
97 changes: 61 additions & 36 deletions lib/pages/chat/chat_app_bar_title.dart
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import 'dart:async';

import 'package:connectivity_plus/connectivity_plus.dart';
import 'package:fluffychat/pages/chat/chat_app_bar_title_style.dart';
import 'package:fluffychat/resource/image_paths.dart';
Expand All @@ -11,7 +13,6 @@ import 'package:lottie/lottie.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/utils/matrix_sdk_extensions/matrix_locals.dart';
import 'package:fluffychat/widgets/avatar/avatar.dart';
import 'package:rxdart/rxdart.dart';

class ChatAppBarTitle extends StatelessWidget {
final Widget? actions;
Expand All @@ -22,6 +23,8 @@ class ChatAppBarTitle extends StatelessWidget {
final Stream<ConnectivityResult> connectivityResultStream;
final VoidCallback onPushDetails;
final String? roomName;
final ValueNotifier<CachedPresence?> cachedPresenceNotifier;
final StreamController<CachedPresence>? cachedPresenceStreamController;

const ChatAppBarTitle({
super.key,
Expand All @@ -33,6 +36,8 @@ class ChatAppBarTitle extends StatelessWidget {
required this.sendController,
required this.connectivityResultStream,
required this.onPushDetails,
required this.cachedPresenceNotifier,
this.cachedPresenceStreamController,
});

@override
Expand Down Expand Up @@ -106,6 +111,9 @@ class ChatAppBarTitle extends StatelessWidget {
_ChatAppBarStatusContent(
connectivityResultStream: connectivityResultStream,
room: room!,
cachedPresenceNotifier: cachedPresenceNotifier,
cachedPresenceStreamController:
cachedPresenceStreamController,
),
],
),
Expand All @@ -120,17 +128,23 @@ class _ChatAppBarStatusContent extends StatelessWidget {
const _ChatAppBarStatusContent({
required this.connectivityResultStream,
required this.room,
required this.cachedPresenceNotifier,
this.cachedPresenceStreamController,
});

final Stream<ConnectivityResult> connectivityResultStream;
final Room room;
final ValueNotifier<CachedPresence?> cachedPresenceNotifier;
final StreamController<CachedPresence>? cachedPresenceStreamController;

@override
Widget build(BuildContext context) {
if (room.isDirectChat) {
return _DirectChatAppBarStatusContent(
connectivityResultStream: connectivityResultStream,
room: room,
cachedPresenceNotifier: cachedPresenceNotifier,
cachedPresenceStreamController: cachedPresenceStreamController!,
);
}

Expand All @@ -145,50 +159,61 @@ class _DirectChatAppBarStatusContent extends StatelessWidget {
const _DirectChatAppBarStatusContent({
required this.connectivityResultStream,
required this.room,
required this.cachedPresenceNotifier,
required this.cachedPresenceStreamController,
});

final Stream<ConnectivityResult> connectivityResultStream;
final Room room;
final ValueNotifier<CachedPresence?> cachedPresenceNotifier;
final StreamController<CachedPresence> cachedPresenceStreamController;

@override
Widget build(BuildContext context) {
CachedPresence? directChatPresence = room.directChatPresence;
return FutureBuilder<GetPresenceResponse>(
future: room.client.getPresence(room.directChatMatrixID!),
builder: (context, futureSnapshot) {
if (futureSnapshot.hasData) {
directChatPresence = CachedPresence.fromPresenceResponse(
futureSnapshot.data!,
room.directChatMatrixID!,
);
}
return StreamBuilder<List>(
stream: CombineLatestStream.list(
[connectivityResultStream, room.directChatPresenceStream],
),
builder: (context, snapshot) {
final connectivityResult = tryCast<ConnectivityResult>(
snapshot.data?[0],
fallback: ConnectivityResult.none,
);
directChatPresence = tryCast<CachedPresence>(
snapshot.data?[1],
fallback: directChatPresence,
return ValueListenableBuilder(
valueListenable: cachedPresenceNotifier,
builder: (context, directChatCachedPresence, child) {
return StreamBuilder(
stream: connectivityResultStream,
builder: (context, connectivitySnapshot) {
return StreamBuilder(
stream: cachedPresenceStreamController.stream,
builder: (context, cachedPresenceSnapshot) {
final connectivityResult = tryCast<ConnectivityResult>(
connectivitySnapshot.data,
fallback: ConnectivityResult.none,
);
directChatPresence = tryCast<CachedPresence>(
cachedPresenceSnapshot.data,
fallback: directChatCachedPresence,
);
if (connectivitySnapshot.hasData &&
connectivityResult == ConnectivityResult.none) {
return _ChatAppBarTitleText(
text: L10n.of(context)!.noConnection,
);
}
if (directChatPresence == null) {
return _ChatAppBarTitleText(
text: L10n.of(context)!.loading,
);
}
final typingText = room.getLocalizedTypingText(context);
if (typingText.isEmpty) {
return _ChatAppBarTitleText(
text: room
.getLocalizedStatus(
context,
presence: directChatPresence,
)
.capitalize(context),
);
} else {
return _ChatAppBarTitleTyping(typingText: typingText);
}
},
);
if (snapshot.hasData &&
connectivityResult == ConnectivityResult.none) {
return _ChatAppBarTitleText(text: L10n.of(context)!.noConnection);
}
final typingText = room.getLocalizedTypingText(context);
if (typingText.isEmpty) {
return _ChatAppBarTitleText(
text: room
.getLocalizedStatus(context, presence: directChatPresence)
.capitalize(context),
);
} else {
return _ChatAppBarTitleTyping(typingText: typingText);
}
},
);
},
Expand Down
4 changes: 4 additions & 0 deletions lib/pages/chat/chat_view.dart
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,10 @@ class ChatView extends StatelessWidget with MessageContentMixin {
actions: _appBarActions(context),
onPushDetails: controller.onPushDetails,
roomName: controller.roomName,
cachedPresenceNotifier:
controller.cachedPresenceNotifier,
cachedPresenceStreamController:
controller.cachedPresenceStreamController,
),
),
],
Expand Down

0 comments on commit 8b4d523

Please sign in to comment.