feat: Add scroll-to-event

This commit is contained in:
Sorunome 2020-09-19 19:21:33 +02:00
parent 94f8f34849
commit 8547422f80
No known key found for this signature in database
GPG Key ID: B19471D07FC9BE9C
8 changed files with 269 additions and 76 deletions

View File

@ -1,6 +1,10 @@
# Version 0.19.0 - 2020-??-?? # Version 0.19.0 - 2020-??-??
### Features ### Features
- Implemented ignore list - Implemented ignore list
- Jump to events in timeline: When tapping on a reply and when tapping a matrix.to link
### Fixes
- Timeline randomly resorting while more history is being fetched
- Automatically request history if the "load more" button is on the screen
# Version 0.18.0 - 2020-09-13 # Version 0.18.0 - 2020-09-13
### Features ### Features

View File

@ -20,6 +20,7 @@ class Message extends StatelessWidget {
final Event nextEvent; final Event nextEvent;
final Function(Event) onSelect; final Function(Event) onSelect;
final Function(Event) onAvatarTab; final Function(Event) onAvatarTab;
final Function(String) scrollToEventId;
final bool longPressSelect; final bool longPressSelect;
final bool selected; final bool selected;
final Timeline timeline; final Timeline timeline;
@ -29,6 +30,7 @@ class Message extends StatelessWidget {
this.longPressSelect, this.longPressSelect,
this.onSelect, this.onSelect,
this.onAvatarTab, this.onAvatarTab,
this.scrollToEventId,
this.selected, this.selected,
this.timeline}); this.timeline});
@ -110,10 +112,19 @@ class Message extends StatelessWidget {
status: 1, status: 1,
originServerTs: DateTime.now(), originServerTs: DateTime.now(),
); );
return Container( return InkWell(
margin: EdgeInsets.symmetric(vertical: 4.0), child: AbsorbPointer(
child: ReplyContent(replyEvent, child: Container(
lightText: ownMessage, timeline: timeline), margin: EdgeInsets.symmetric(vertical: 4.0),
child: ReplyContent(replyEvent,
lightText: ownMessage, timeline: timeline),
),
),
onTap: () {
if (scrollToEventId != null) {
scrollToEventId(replyEvent.eventId);
}
},
); );
}, },
), ),

View File

@ -0,0 +1,24 @@
extension MatrixIdentifierStringExtension on String {
/// Separates room identifiers with an event id and possibly a query parameter into its components.
MatrixIdentifierStringExtensionResults parseIdentifierIntoParts() {
final match = RegExp(r'^([#!][^:]*:[^\/?]*)(?:\/(\$[^?]*))?(?:\?(.*))?$')
.firstMatch(this);
if (match == null) {
return null;
}
return MatrixIdentifierStringExtensionResults(
roomIdOrAlias: match.group(1),
eventId: match.group(2),
queryString: match.group(3),
);
}
}
class MatrixIdentifierStringExtensionResults {
final String roomIdOrAlias;
final String eventId;
final String queryString;
MatrixIdentifierStringExtensionResults(
{this.roomIdOrAlias, this.eventId, this.queryString});
}

View File

@ -5,6 +5,7 @@ import 'package:fluffychat/utils/app_route.dart';
import 'package:fluffychat/views/chat.dart'; import 'package:fluffychat/views/chat.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:url_launcher/url_launcher.dart'; import 'package:url_launcher/url_launcher.dart';
import 'matrix_identifier_string_extension.dart';
class UrlLauncher { class UrlLauncher {
final String url; final String url;
@ -23,51 +24,79 @@ class UrlLauncher {
final matrix = Matrix.of(context); final matrix = Matrix.of(context);
final identifier = url.replaceAll('https://matrix.to/#/', ''); final identifier = url.replaceAll('https://matrix.to/#/', '');
if (identifier[0] == '#' || identifier[0] == '!') { if (identifier[0] == '#' || identifier[0] == '!') {
var room = matrix.client.getRoomByAlias(identifier); // sometimes we have identifiers which have an event id and additional query parameters
room ??= matrix.client.getRoomById(identifier); // we want to separate those.
final identityParts = identifier.parseIdentifierIntoParts();
if (identityParts == null) {
return; // no match, nothing to do
}
final roomIdOrAlias = identityParts.roomIdOrAlias;
final event = identityParts.eventId;
final query = identityParts.queryString;
var room = matrix.client.getRoomByAlias(roomIdOrAlias) ??
matrix.client.getRoomById(roomIdOrAlias);
var roomId = room?.id; var roomId = room?.id;
var servers = <String>[]; // we make the servers a set and later on convert to a list, so that we can easily
if (room == null && identifier == '#') { // deduplicate servers added via alias lookup and query parameter
var servers = <String>{};
if (room == null && roomIdOrAlias == '#') {
// we were unable to find the room locally...so resolve it // we were unable to find the room locally...so resolve it
final response = final response =
await SimpleDialogs(context).tryRequestWithLoadingDialog( await SimpleDialogs(context).tryRequestWithLoadingDialog(
matrix.client.requestRoomAliasInformations(identifier), matrix.client.requestRoomAliasInformations(roomIdOrAlias),
); );
if (response != false) { if (response != false) {
roomId = response.roomId; roomId = response.roomId;
servers = response.servers; servers.addAll(response.servers);
room = matrix.client.getRoomById(roomId); room = matrix.client.getRoomById(roomId);
} }
} }
if (query != null) {
// the query information might hold additional servers to try, so let's try them!
// as there might be multiple "via" tags we can't just use Uri.splitQueryString, we need to do our own thing
for (final parameter in query.split('&')) {
final index = parameter.indexOf('=');
if (index == -1) {
continue;
}
if (Uri.decodeQueryComponent(parameter.substring(0, index)) !=
'via') {
continue;
}
servers.add(Uri.decodeQueryComponent(parameter.substring(index + 1)));
}
}
if (room != null) { if (room != null) {
// we have the room, so....just open it! // we have the room, so....just open it!
await Navigator.pushAndRemoveUntil( await Navigator.pushAndRemoveUntil(
context, context,
AppRoute.defaultRoute(context, ChatView(room.id)), AppRoute.defaultRoute(
context, ChatView(room.id, scrollToEventId: event)),
(r) => r.isFirst, (r) => r.isFirst,
); );
return; return;
} }
if (identifier == '!') { if (roomIdOrAlias[0] == '!') {
roomId = identifier; roomId = roomIdOrAlias;
} }
if (roomId == null) { if (roomId == null) {
// we haven't found this room....so let's ignore it // we haven't found this room....so let's ignore it
return; return;
} }
if (await SimpleDialogs(context) if (await SimpleDialogs(context)
.askConfirmation(titleText: 'Join room $identifier')) { .askConfirmation(titleText: 'Join room $roomIdOrAlias')) {
final response = final response =
await SimpleDialogs(context).tryRequestWithLoadingDialog( await SimpleDialogs(context).tryRequestWithLoadingDialog(
matrix.client.joinRoomOrAlias( matrix.client.joinRoomOrAlias(
Uri.encodeComponent(roomId), Uri.encodeComponent(roomIdOrAlias),
servers: servers, servers: servers.toList(),
), ),
); );
if (response == false) return; if (response == false) return;
await Navigator.pushAndRemoveUntil( await Navigator.pushAndRemoveUntil(
context, context,
AppRoute.defaultRoute(context, ChatView(response['room_id'])), AppRoute.defaultRoute(
context, ChatView(response['room_id'], scrollToEventId: event)),
(r) => r.isFirst, (r) => r.isFirst,
); );
} }

View File

@ -3,6 +3,7 @@ import 'dart:io';
import 'dart:math'; import 'dart:math';
import 'package:famedlysdk/famedlysdk.dart'; import 'package:famedlysdk/famedlysdk.dart';
import 'package:flutter/scheduler.dart';
import 'package:fluffychat/components/adaptive_page_layout.dart'; import 'package:fluffychat/components/adaptive_page_layout.dart';
import 'package:fluffychat/components/avatar.dart'; import 'package:fluffychat/components/avatar.dart';
import 'package:fluffychat/components/chat_settings_popup_menu.dart'; import 'package:fluffychat/components/chat_settings_popup_menu.dart';
@ -24,6 +25,7 @@ import 'package:memoryfilepicker/memoryfilepicker.dart';
import 'package:pedantic/pedantic.dart'; import 'package:pedantic/pedantic.dart';
import 'package:image_picker/image_picker.dart'; import 'package:image_picker/image_picker.dart';
import 'package:file_picker_platform_interface/file_picker_platform_interface.dart'; import 'package:file_picker_platform_interface/file_picker_platform_interface.dart';
import 'package:scroll_to_index/scroll_to_index.dart';
import 'chat_details.dart'; import 'chat_details.dart';
import 'chat_list.dart'; import 'chat_list.dart';
@ -33,8 +35,9 @@ import '../utils/matrix_file_extension.dart';
class ChatView extends StatelessWidget { class ChatView extends StatelessWidget {
final String id; final String id;
final String scrollToEventId;
const ChatView(this.id, {Key key}) : super(key: key); const ChatView(this.id, {Key key, this.scrollToEventId}) : super(key: key);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -44,15 +47,16 @@ class ChatView extends StatelessWidget {
firstScaffold: ChatList( firstScaffold: ChatList(
activeChat: id, activeChat: id,
), ),
secondScaffold: _Chat(id), secondScaffold: _Chat(id, scrollToEventId: scrollToEventId),
); );
} }
} }
class _Chat extends StatefulWidget { class _Chat extends StatefulWidget {
final String id; final String id;
final String scrollToEventId;
const _Chat(this.id, {Key key}) : super(key: key); const _Chat(this.id, {Key key, this.scrollToEventId}) : super(key: key);
@override @override
_ChatState createState() => _ChatState(); _ChatState createState() => _ChatState();
@ -67,7 +71,7 @@ class _ChatState extends State<_Chat> {
String seenByText = ''; String seenByText = '';
final ScrollController _scrollController = ScrollController(); final AutoScrollController _scrollController = AutoScrollController();
FocusNode inputFocus = FocusNode(); FocusNode inputFocus = FocusNode();
@ -101,28 +105,33 @@ class _ChatState extends State<_Chat> {
timeline.requestHistory(historyCount: _loadHistoryCount), timeline.requestHistory(historyCount: _loadHistoryCount),
); );
if (mounted) setState(() => _loadingHistory = false); // we do NOT setState() here as then the event order will be wrong.
// instead, we just set our variable to false, and rely on timeline update to set the
// new state, thus triggering a re-render, for us
_loadingHistory = false;
}
}
void _updateScrollController() {
if (_scrollController.position.pixels ==
_scrollController.position.maxScrollExtent &&
timeline.events.isNotEmpty &&
timeline.events[timeline.events.length - 1].type !=
EventTypes.RoomCreate) {
requestHistory();
}
if (_scrollController.position.pixels > 0 &&
showScrollDownButton == false) {
setState(() => showScrollDownButton = true);
} else if (_scrollController.position.pixels == 0 &&
showScrollDownButton == true) {
setState(() => showScrollDownButton = false);
} }
} }
@override @override
void initState() { void initState() {
_scrollController.addListener(() async { _scrollController.addListener(() => _updateScrollController);
if (_scrollController.position.pixels ==
_scrollController.position.maxScrollExtent &&
timeline.events.isNotEmpty &&
timeline.events[timeline.events.length - 1].type !=
EventTypes.RoomCreate) {
requestHistory();
}
if (_scrollController.position.pixels > 0 &&
showScrollDownButton == false) {
setState(() => showScrollDownButton = true);
} else if (_scrollController.position.pixels == 0 &&
showScrollDownButton == true) {
setState(() => showScrollDownButton = false);
}
});
super.initState(); super.initState();
} }
@ -156,12 +165,22 @@ class _ChatState extends State<_Chat> {
} }
} }
Future<bool> getTimeline() async { Future<bool> getTimeline(BuildContext context) async {
if (timeline == null) { if (timeline == null) {
timeline = await room.getTimeline(onUpdate: updateView); timeline = await room.getTimeline(onUpdate: updateView);
if (timeline.events.isNotEmpty) { if (timeline.events.isNotEmpty) {
unawaited(room.sendReadReceipt(timeline.events.first.eventId)); unawaited(room.sendReadReceipt(timeline.events.first.eventId));
} }
// when the scroll controller is attached we want to scroll to an event id, if specified
// and update the scroll controller...which will trigger a request history, if the
// "load more" button is visible on the screen
SchedulerBinding.instance.addPostFrameCallback((_) async {
if (widget.scrollToEventId != null) {
_scrollToEventId(widget.scrollToEventId, context: context);
}
_updateScrollController();
});
} }
updateView(); updateView();
return true; return true;
@ -316,6 +335,66 @@ class _ChatState extends State<_Chat> {
inputFocus.requestFocus(); inputFocus.requestFocus();
} }
void _scrollToEventId(String eventId, {BuildContext context}) async {
var eventIndex =
getFilteredEvents().indexWhere((e) => e.eventId == eventId);
if (eventIndex == -1) {
// event id not found...maybe we can fetch it?
// the try...finally is here to start and close the loading dialog reliably
try {
if (context != null) {
SimpleDialogs(context).showLoadingDialog(context);
}
// okay, we first have to fetch if the event is in the room
try {
final event = await timeline.getEventById(eventId);
if (event == null) {
// event is null...meaning something is off
return;
}
} catch (err) {
if (err is MatrixException && err.errcode == 'M_NOT_FOUND') {
// event wasn't found, as the server gave a 404 or something
return;
}
rethrow;
}
// okay, we know that the event *is* in the room
while (eventIndex == -1) {
if (!_canLoadMore) {
// we can't load any more events but still haven't found ours yet...better stop here
return;
}
try {
await timeline.requestHistory(historyCount: _loadHistoryCount);
} catch (err) {
if (err is TimeoutException) {
// loading the history timed out...so let's do nothing
return;
}
rethrow;
}
eventIndex =
getFilteredEvents().indexWhere((e) => e.eventId == eventId);
}
} finally {
if (context != null) {
Navigator.of(context)?.pop();
}
}
}
await _scrollController.scrollToIndex(eventIndex,
preferPosition: AutoScrollPosition.middle);
_updateScrollController();
}
List<Event> getFilteredEvents() => timeline.events
.where((e) =>
![RelationshipTypes.Edit, RelationshipTypes.Reaction]
.contains(e.relationshipType) &&
e.type != 'm.reaction')
.toList();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
matrix = Matrix.of(context); matrix = Matrix.of(context);
@ -484,7 +563,7 @@ class _ChatState extends State<_Chat> {
ConnectionStatusHeader(), ConnectionStatusHeader(),
Expanded( Expanded(
child: FutureBuilder<bool>( child: FutureBuilder<bool>(
future: getTimeline(), future: getTimeline(context),
builder: (BuildContext context, snapshot) { builder: (BuildContext context, snapshot) {
if (!snapshot.hasData) { if (!snapshot.hasData) {
return Center( return Center(
@ -500,14 +579,7 @@ class _ChatState extends State<_Chat> {
room.sendReadReceipt(timeline.events.first.eventId); room.sendReadReceipt(timeline.events.first.eventId);
} }
final filteredEvents = timeline.events final filteredEvents = getFilteredEvents();
.where((e) =>
![
RelationshipTypes.Edit,
RelationshipTypes.Reaction
].contains(e.relationshipType) &&
e.type != 'm.reaction')
.toList();
return ListView.builder( return ListView.builder(
padding: EdgeInsets.symmetric( padding: EdgeInsets.symmetric(
@ -570,34 +642,48 @@ class _ChatState extends State<_Chat> {
bottom: 8, bottom: 8,
), ),
) )
: Message(filteredEvents[i - 1], : AutoScrollTag(
onAvatarTab: (Event event) { key: ValueKey(i - 1),
sendController.text += index: i - 1,
' ${event.senderId}'; controller: _scrollController,
}, onSelect: (Event event) { child: Message(filteredEvents[i - 1],
if (!event.redacted) { onAvatarTab: (Event event) {
if (selectedEvents.contains(event)) { sendController.text +=
setState( ' ${event.senderId}';
() => selectedEvents.remove(event), },
); onSelect: (Event event) {
} else { if (!event.redacted) {
setState( if (selectedEvents
() => selectedEvents.add(event), .contains(event)) {
); setState(
} () => selectedEvents
selectedEvents.sort( .remove(event),
(a, b) => a.originServerTs );
.compareTo(b.originServerTs), } else {
); setState(
} () =>
}, selectedEvents.add(event),
longPressSelect: selectedEvents.isEmpty, );
selected: selectedEvents }
.contains(filteredEvents[i - 1]), selectedEvents.sort(
timeline: timeline, (a, b) => a.originServerTs
nextEvent: i >= 2 .compareTo(
? filteredEvents[i - 2] b.originServerTs),
: null); );
}
},
scrollToEventId: (String eventId) =>
_scrollToEventId(eventId,
context: context),
longPressSelect:
selectedEvents.isEmpty,
selected: selectedEvents
.contains(filteredEvents[i - 1]),
timeline: timeline,
nextEvent: i >= 2
? filteredEvents[i - 2]
: null),
);
}); });
}, },
), ),

View File

@ -739,6 +739,13 @@ packages:
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "0.24.1" version: "0.24.1"
scroll_to_index:
dependency: "direct main"
description:
name: scroll_to_index
url: "https://pub.dartlang.org"
source: hosted
version: "1.0.6"
sentry: sentry:
dependency: "direct main" dependency: "direct main"
description: description:

View File

@ -68,6 +68,7 @@ dependencies:
ref: master ref: master
flutter_blurhash: ^0.5.0 flutter_blurhash: ^0.5.0
sentry: ">=3.0.0 <4.0.0" sentry: ">=3.0.0 <4.0.0"
scroll_to_index: ^1.0.6
dev_dependencies: dev_dependencies:
flutter_test: flutter_test:

View File

@ -0,0 +1,31 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:fluffychat/utils/matrix_identifier_string_extension.dart';
void main() {
group('Matrix Identifier String Extension', () {
test('parseIdentifierIntoParts', () {
var res = '#alias:beep'.parseIdentifierIntoParts();
expect(res.roomIdOrAlias, '#alias:beep');
expect(res.eventId, null);
expect(res.queryString, null);
res = 'blha'.parseIdentifierIntoParts();
expect(res, null);
res = '#alias:beep/\$event'.parseIdentifierIntoParts();
expect(res.roomIdOrAlias, '#alias:beep');
expect(res.eventId, '\$event');
expect(res.queryString, null);
res = '#alias:beep?blubb'.parseIdentifierIntoParts();
expect(res.roomIdOrAlias, '#alias:beep');
expect(res.eventId, null);
expect(res.queryString, 'blubb');
res = '#alias:beep/\$event?blubb'.parseIdentifierIntoParts();
expect(res.roomIdOrAlias, '#alias:beep');
expect(res.eventId, '\$event');
expect(res.queryString, 'blubb');
res = '#/\$?:beep/\$event?blubb?b'.parseIdentifierIntoParts();
expect(res.roomIdOrAlias, '#/\$?:beep');
expect(res.eventId, '\$event');
expect(res.queryString, 'blubb?b');
});
});
}