fluffychat/lib/pangea/toolbar/widgets/message_selection_positioner.dart
ggurdin 0d30a8fab2
Fluffychat merge (#2731)
* build: Update snapcraft

* build: Follow up snapcraft build

* build: Install flutter via git in snapcraft

* chore: Follow up typo

* fix: Request notification permissions on iOS before getToken

* chore: Follow up request iOS permissions

* Revert "chore: Follow up request iOS permissions"

This reverts commit 2625e89a33.

* chore: Combine mimetype types in send file dialog logic

* build: Update flutter to 3.29.0

* Translated using Weblate (Basque)

Currently translated at 99.8% (758 of 759 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/

* Revert "build: Update flutter to 3.29.0"

* fix: Crash in settings when using MAS

* build: Fix build tailwindcss for website

* feat: Navigate in image viewer with keyboard keys

* chore: Nicer colors for reactions

* chore: Better error handling for image rendering

* Translated using Weblate (Latvian)

Currently translated at 100.0% (759 of 759 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* fix: Index of numbered lists are off

* fix(macos): update dependencies to make the build work

This commit was generated mostly by running `flutter run -d macos` and then
`pod update` in the `macos/` directory after that failed.

* fix: never use a transition on the shell route

Changing the PageBuilder here based on a MediaQuery causes the child to briefly
be rendered twice with the same GlobalKey, blowing up the rendering.

I believe this fixes https://github.com/krille-chan/fluffychat/issues/1534.

* feat: New video file picker button

* feat: Send optional message with images or files

* chore: Follow up send file dialog design

* chore: Follow up paddings in room input row

* chore: Follow up paddings

* chore: Follow up paddings

* chore: Follow up input row

* Translated using Weblate (Italian)

Currently translated at 99.6% (756 of 759 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/it/

* Translated using Weblate (Latvian)

Currently translated at 100.0% (759 of 759 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Estonian)

Currently translated at 100.0% (762 of 762 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* Translated using Weblate (Chinese (Simplified Han script))

Currently translated at 100.0% (762 of 762 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/

* chore: Follow up gallery picker

* chore: Better no compression supported UX

* fix: prevent users from creating spaces with empty names

* fix: update condition in account deletion function to allow deletion to go through

* Translated using Weblate (Latvian)

Currently translated at 100.0% (762 of 762 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Estonian)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* Translated using Weblate (Basque)

Currently translated at 99.8% (762 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/

* Translated using Weblate (Galician)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/

* Translated using Weblate (Chinese (Simplified Han script))

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/

* Translated using Weblate (Latvian)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Basque)

Currently translated at 99.8% (762 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/

* Translated using Weblate (Indonesian)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/

* Translated using Weblate (Korean)

Currently translated at 96.4% (736 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ko/

* Translated using Weblate (Irish)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/

* Translated using Weblate (Filipino)

Currently translated at 25.8% (197 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/fil/

* Translated using Weblate (Polish)

Currently translated at 98.4% (751 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/pl/

* Translated using Weblate (Polish)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/pl/

* fix: Remove too sensitive dismiss gesture on chat list items

* fix: Add missing <s> html tag to render

* Translated using Weblate (Dutch)

Currently translated at 81.6% (623 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* refactor: Remove custom font and emoji font workaround

* build: Add android namespace

* build: Update kotlin gradle plugin

* Revert "build: Update kotlin gradle plugin"

* feat: Add advanced configuration page

* refactor: Improved UX for room upgrades

* Translated using Weblate (French)

Currently translated at 86.3% (659 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/fr/

* Translated using Weblate (Dutch)

Currently translated at 82.0% (626 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (Chinese (Traditional Han script))

Currently translated at 88.8% (678 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hant/

* Translated using Weblate (Dutch)

Currently translated at 83.3% (636 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (German)

Currently translated at 93.9% (717 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* Translated using Weblate (German)

Currently translated at 93.9% (717 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* Translated using Weblate (German)

Currently translated at 93.9% (717 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* Translated using Weblate (German)

Currently translated at 95.6% (730 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* Translated using Weblate (Dutch)

Currently translated at 94.4% (721 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (Spanish)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* Translated using Weblate (Spanish)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* Translated using Weblate (Spanish)

Currently translated at 100.0% (763 of 763 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* Translated using Weblate (Spanish)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* Translated using Weblate (Polish)

Currently translated at 99.8% (763 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/pl/

* Translated using Weblate (Ukrainian)

Currently translated at 93.3% (713 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/

* Translated using Weblate (Chinese (Simplified Han script))

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/

* Translated using Weblate (Indonesian)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/

* chore: divider when scrolled up

* refactor: Easier shift enter logic for text input

* Translated using Weblate (Irish)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/

* Translated using Weblate (Latvian)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Estonian)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* Translated using Weblate (Ukrainian)

Currently translated at 94.8% (725 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/

* build: Downgrade packages and move to fixed flutter typeahead fork

* chore: Use other join endpoint for room upgrades

* chore: disable echoCancel for audio messages

* chore: Simpler changing config variables

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up config editor

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Make push gateway configurable

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up code formatting

* build: Update flutter 3.29.2

Signed-off-by: Krille <c.kussowski@famedly.com>

* Revert "chore: Follow up code formatting"

This reverts commit 0f000f952f.

* Revert "build: Update flutter 3.29.2"

This reverts commit bfd23952b7.

* refactor: Formatting

* build: Update matrix dart sdk

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up update matrix dart sdk

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up formatting

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Update openssl to 0.5.0

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Update gorouter package

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Update to flutter 3.29.2

Signed-off-by: Krille <c.kussowski@famedly.com>

* Translated using Weblate (Dutch)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Added translation using Weblate (Telugu)

* Translated using Weblate (Dutch)

Currently translated at 100.0% (764 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (Telugu)

Currently translated at 0.5% (4 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/te/

* Translated using Weblate (German)

Currently translated at 96.5% (738 of 764 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* Translated using Weblate (Estonian)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* Translated using Weblate (Irish)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/

* Translated using Weblate (Chinese (Simplified Han script))

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/

* Translated using Weblate (Galician)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/

* Translated using Weblate (Dutch)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (Latvian)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Latvian)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/

* Translated using Weblate (Ukrainian)

Currently translated at 95.9% (734 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/

* Translated using Weblate (Indonesian)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/

* fix: Consistent element padding between server picker and login view

* refactor: Migrate more config options to config viewer

Signed-off-by: Krille <c.kussowski@famedly.com>

* refactor: Reuse unused kotlin imports

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Update pubspec.lock

Signed-off-by: Krille <c.kussowski@famedly.com>

* Revert "build: Install flutter via git in snapcraft"

This reverts commit cd12f773fe.

* chore: Update locale config for localizations

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Add libpciaccess0 package to snap

Signed-off-by: Krille <c.kussowski@famedly.com>

* Translated using Weblate (Dutch)

Currently translated at 100.0% (765 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* Translated using Weblate (Chinese (Traditional Han script))

Currently translated at 93.3% (714 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hant/

* Translated using Weblate (Russian)

Currently translated at 95.6% (732 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/

* chore: upgrade chewie and video_player packages

This bumps the minimum Flutter version to 3.27. I think this is not an issue, since e93fdebe20 upgraded to 3.29.2 already.

* fix: properly dispose VideoPlayerController 

This ensures that a playing video stops playing when we navigate away from the chat.

I also reorganized the code a little.

* feat: support inline video playback on macOS

It turns out that video_player supports macOS, so we can simply enable it.

* feat: clearly mark when a video is to be downloaded

This shows a download icon instead of the play icon on top of the video if the video player isn't supported.

* Translated using Weblate (Chinese (Traditional Han script))

Currently translated at 93.7% (717 of 765 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hant/

* build: Add libpciaccess0 for snapcraft

* build: Add libpciaccess-dev for snapcraft

* build: use singleInstance as launchmode

* fix: Null error in ClientChooserButton

* chore: Improve avatar designg

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up new room design

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Correct availability of desktop builds

* refactor: Replace user bottom sheet with menu and small dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* refactor: Replace user bottom sheet with menu and small dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up dialog themes

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up dialog themes

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up dialog themes

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Update matrix dart sdk to 0.39.0

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up user dialog theme

* chore: Use Cupertino Activity Indicator in ChatEventList

* chore: Follow up permissions slider dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* refactor: Implement avatar image viewer and adjust design

Signed-off-by: Krille <c.kussowski@famedly.com>

* feat: Filter for room members page and easier approve knocking users

Signed-off-by: Krille <c.kussowski@famedly.com>

* refactor: Move public room bottom sheet into dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up public rooms dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* fix: Text scale factor in Linkify widgets

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Add matrix notifications for issues

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up matrix notification

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up matrix notification

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up matrix notification

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up matrix notification

Signed-off-by: Krille <c.kussowski@famedly.com>

* android updates

* chore: update fetching of chat details display setting in message overlay positioner

* fluffychat merge

* build: Flutter 3.29.3

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Nicer scaffold dialog for column mode

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up scaffold dialog

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up members list

* chore: Follow up message design

* chore: Follow up message design

* chore: Follow up file message design

Signed-off-by: Krille <c.kussowski@famedly.com>

* build: Bump version to 1.26.0

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: Follow up message design

* build: Use 0.1.0 fcm_shared_isolate on ios

Signed-off-by: Krille <c.kussowski@famedly.com>

* chore: disable matrix notification github action

* fix import error

* make overlay message padding match message bubble padding

---------

Signed-off-by: Krille <c.kussowski@famedly.com>
Co-authored-by: Krille-chan <christian-kussowski@posteo.de>
Co-authored-by: Krille <c.kussowski@famedly.com>
Co-authored-by: xabirequejo <xabi.rn@gmail.com>
Co-authored-by: Edgars Andersons <Edgars+Weblate@gaitenis.id.lv>
Co-authored-by: Rafał Hirsch <rafal@hirsch.net>
Co-authored-by: Angelo Schirinzi <Odi-3@users.noreply.hosted.weblate.org>
Co-authored-by: Priit Jõerüüt <hwlate@joeruut.com>
Co-authored-by: Poesty Li <poesty7450@gmail.com>
Co-authored-by: josé m <correoxm@disroot.org>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
Co-authored-by: Linerly <linerly@proton.me>
Co-authored-by: kdh8219 <kdh8219@monamo.dev>
Co-authored-by: Aindriú Mac Giolla Eoin <aindriu80@gmail.com>
Co-authored-by: searinminecraft <kitakita@disroot.org>
Co-authored-by: Piotr Orzechowski <piotr@orzechowski.tech>
Co-authored-by: Jelv <post@jelv.nl>
Co-authored-by: Antonin Del Fabbro <message@antonin.one>
Co-authored-by: Mare JP <seraphmare@gmail.com>
Co-authored-by: nautilusx <translate@disroot.org>
Co-authored-by: Very Able <veryable@proton.me>
Co-authored-by: Kimby <kimisaes@naver.com>
Co-authored-by: José Muñoz <dr.cabra@disroot.org>
Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com>
Co-authored-by: katakam chakri <katakam.chakri@gmail.com>
Co-authored-by: ℂ𝕠𝕠𝕠𝕝 (𝕘𝕚𝕥𝕙𝕦𝕓.𝕔𝕠𝕞/ℂ𝕠𝕠𝕠𝕝) <coool@mail.lv>
Co-authored-by: xegim <ja3lpark@gmail.com>
Co-authored-by: miullu <satou.ide@gmail.com>
Co-authored-by: Yurt Page <yurtpage@gmail.com>
Co-authored-by: Lenni <87639068+Lenni-builder@users.noreply.github.com>
2025-05-08 13:22:26 -04:00

768 lines
26 KiB
Dart

import 'dart:async';
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/config/setting_keys.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/pages/chat/chat.dart';
import 'package:fluffychat/pangea/common/utils/error_handler.dart';
import 'package:fluffychat/pangea/events/event_wrappers/pangea_message_event.dart';
import 'package:fluffychat/pangea/events/models/pangea_token_model.dart';
import 'package:fluffychat/pangea/instructions/instructions_enum.dart';
import 'package:fluffychat/pangea/instructions/instructions_inline_tooltip.dart';
import 'package:fluffychat/pangea/toolbar/enums/message_mode_enum.dart';
import 'package:fluffychat/pangea/toolbar/enums/reading_assistance_mode_enum.dart';
import 'package:fluffychat/pangea/toolbar/reading_assistance_input_row/overlay_footer.dart';
import 'package:fluffychat/pangea/toolbar/widgets/measure_render_box.dart';
import 'package:fluffychat/pangea/toolbar/widgets/message_selection_overlay.dart';
import 'package:fluffychat/pangea/toolbar/widgets/overlay_center_content.dart';
import 'package:fluffychat/pangea/toolbar/widgets/overlay_header.dart';
import 'package:fluffychat/pangea/toolbar/widgets/select_mode_buttons.dart';
import 'package:fluffychat/widgets/avatar.dart';
import 'package:fluffychat/widgets/matrix.dart';
/// Controls positioning of the message overlay.
class MessageSelectionPositioner extends StatefulWidget {
final MessageOverlayController overlayController;
final ChatController chatController;
final Event event;
final PangeaMessageEvent? pangeaMessageEvent;
final PangeaToken? initialSelectedToken;
final Event? nextEvent;
final Event? prevEvent;
const MessageSelectionPositioner({
required this.overlayController,
required this.chatController,
required this.event,
this.pangeaMessageEvent,
this.initialSelectedToken,
this.nextEvent,
this.prevEvent,
super.key,
});
@override
MessageSelectionPositionerState createState() =>
MessageSelectionPositionerState();
}
class MessageSelectionPositionerState extends State<MessageSelectionPositioner>
with TickerProviderStateMixin {
late AnimationController _animationController;
Offset? _centeredMessageOffset;
Size? _centeredMessageSize;
Size? _tooltipSize;
final Completer _centeredMessageCompleter = Completer();
final Completer _tooltipCompleter = Completer();
MessageMode _currentMode = MessageMode.noneSelected;
Animation<Offset>? _overlayOffsetAnimation;
Animation<Size>? _messageSizeAnimation;
Offset? _currentOffset;
StreamSubscription? _reactionSubscription;
final _animationDuration = const Duration(
milliseconds: AppConfig.overlayAnimationDuration,
// seconds: 5,
);
@override
void initState() {
super.initState();
_currentMode = widget.overlayController.toolbarMode;
_animationController = AnimationController(
vsync: this,
duration: _animationDuration,
);
_reactionSubscription =
widget.chatController.room.client.onSync.stream.where(
(update) {
// check if this sync update has a reaction event or a
// redaction (of a reaction event). If so, rebuild the overlay
final room = widget.chatController.room;
final timelineEvents = update.rooms?.join?[room.id]?.timeline?.events;
if (timelineEvents == null) return false;
final eventID = widget.event.eventId;
return timelineEvents.any(
(e) =>
e.type == EventTypes.Redaction ||
(e.type == EventTypes.Reaction &&
Event.fromMatrixEvent(e, room).relationshipEventId ==
eventID),
);
},
).listen((_) => setState(() {}));
WidgetsBinding.instance.addPostFrameCallback((_) async {
await _centeredMessageCompleter.future;
if (!mounted) return;
setState(() {
_currentOffset = Offset(
_ownMessage ? _messageRightOffset : _messageLeftOffset,
_originalMessageBottomOffset -
_reactionsHeight -
_selectionButtonsHeight,
);
});
_setReadingAssistanceMode(
ReadingAssistanceMode.selectMode,
);
});
}
@override
void didUpdateWidget(MessageSelectionPositioner oldWidget) {
super.didUpdateWidget(oldWidget);
final mode = widget.overlayController.toolbarMode;
if (mode != _currentMode) {
setState(() => _currentMode = mode);
}
}
@override
void dispose() {
_animationController.dispose();
_reactionSubscription?.cancel();
super.dispose();
}
void _setCenteredMessageSize(RenderBox renderBox) {
if (_centeredMessageCompleter.isCompleted) return;
_centeredMessageSize = renderBox.size;
final offset = renderBox.localToGlobal(Offset.zero);
_centeredMessageOffset = Offset(
offset.dx - _columnWidth - _horizontalPadding - 2.0,
_mediaQuery!.size.height -
(offset.dy -
((AppConfig.practiceModeInputBarHeight -
AppConfig.selectModeInputBarHeight) *
0.75)) -
renderBox.size.height -
_reactionsHeight,
);
setState(() {});
if (!_centeredMessageCompleter.isCompleted) {
_centeredMessageCompleter.complete();
}
}
void _setTooltipSize(RenderBox renderBox) {
setState(() {
_tooltipSize = renderBox.size;
});
if (!_tooltipCompleter.isCompleted) {
_tooltipCompleter.complete();
}
}
Future<void> _setReadingAssistanceMode(ReadingAssistanceMode mode) async {
if (mode == _readingAssistanceMode) {
return;
}
await _centeredMessageCompleter.future;
if (mode == ReadingAssistanceMode.practiceMode) {
setState(
() => widget.overlayController.readingAssistanceMode =
ReadingAssistanceMode.transitionMode,
);
} else if (mode == ReadingAssistanceMode.selectMode) {
setState(
() => widget.overlayController.readingAssistanceMode =
ReadingAssistanceMode.selectMode,
);
}
if (mode == ReadingAssistanceMode.selectMode) {
_overlayOffsetAnimation = Tween<Offset>(
begin: _currentOffset,
end: _adjustedOriginalMessageOffset,
).animate(
CurvedAnimation(
parent: _animationController,
curve: FluffyThemes.animationCurve,
),
)..addListener(() {
if (mounted) {
setState(() => _currentOffset = _overlayOffsetAnimation?.value);
}
});
} else if (mode == ReadingAssistanceMode.practiceMode) {
_overlayOffsetAnimation = Tween<Offset>(
begin: _currentOffset,
end: _centeredMessageOffset!,
).animate(
CurvedAnimation(
parent: _animationController,
curve: FluffyThemes.animationCurve,
),
)..addListener(() {
if (mounted) {
setState(() => _currentOffset = _overlayOffsetAnimation?.value);
}
});
_messageSizeAnimation = Tween<Size>(
begin: Size(
_originalMessageSize.width,
_originalMessageSize.height,
),
end: _adjustedCenteredMessageSize,
).animate(
CurvedAnimation(
parent: _animationController,
curve: FluffyThemes.animationCurve,
),
);
}
await _animationController.forward(from: 0);
if (mounted) {
setState(() => widget.overlayController.readingAssistanceMode = mode);
}
}
T _runWithLogging<T>(
Function runner,
String errorMessage,
T defaultValue,
) {
try {
return runner();
} catch (e, s) {
ErrorHandler.logError(
e: "$errorMessage: $e",
s: s,
data: {
"eventID": widget.event.eventId,
},
);
return defaultValue;
}
}
ReadingAssistanceMode? get _readingAssistanceMode =>
widget.overlayController.readingAssistanceMode;
double get _inputBarSize =>
_readingAssistanceMode == ReadingAssistanceMode.practiceMode ||
_readingAssistanceMode == ReadingAssistanceMode.transitionMode
? AppConfig.practiceModeInputBarHeight
: AppConfig.selectModeInputBarHeight;
bool get _showDetails =>
AppSettings.displayChatDetailsColumn.getItem(Matrix.of(context).store) &&
FluffyThemes.isThreeColumnMode(context) &&
widget.chatController.room.membership == Membership.join;
// screen size
MediaQueryData? get _mediaQuery => _runWithLogging<MediaQueryData?>(
() => MediaQuery.of(context),
"Error getting media query",
null,
);
double get _columnWidth => FluffyThemes.isColumnMode(context)
? (FluffyThemes.columnWidth + FluffyThemes.navRailWidth)
: 0;
/// Available vertical space not taken up by the header and footer
double? get _verticalSpace {
if (_mediaQuery == null) return null;
return _mediaQuery!.size.height - _headerHeight - _footerHeight;
}
double get _toolbarMaxWidth {
const double messageMargin = 16.0;
// widget.event.isActivityMessage ? 0 : Avatar.defaultSize + 16 + 8;
final bool showingDetails = widget.chatController.displayChatDetailsColumn;
final double totalMaxWidth = (FluffyThemes.columnWidth * 2.5) -
(showingDetails ? FluffyThemes.columnWidth : 0) -
messageMargin;
double? maxWidth;
if (_mediaQuery != null) {
final chatViewWidth = _mediaQuery!.size.width - _columnWidth;
maxWidth = chatViewWidth - (2 * _horizontalPadding) - messageMargin;
}
if (maxWidth == null || maxWidth > totalMaxWidth) {
maxWidth = totalMaxWidth;
}
return maxWidth;
}
// original message size and offset
RenderBox? get _messageRenderBox => _runWithLogging<RenderBox?>(
() => MatrixState.pAnyState.getRenderBox(
widget.event.eventId,
),
"Error getting message render box",
null,
);
Size get _defaultMessageSize => const Size(FluffyThemes.columnWidth / 2, 100);
/// The size of the message in the chat list (as opposed to the expanded size in the center overlay)
Size get _originalMessageSize {
if (_messageRenderBox == null || !_messageRenderBox!.hasSize) {
return _defaultMessageSize;
}
return _runWithLogging(
() => _messageRenderBox?.size,
"Error getting message size",
_defaultMessageSize,
);
}
static const _messageDefaultLeftMargin = Avatar.defaultSize + 16 + 8;
// Centered message size and offset
bool get _centeredMessageHasOverflow {
if (_verticalSpace == null ||
_centeredMessageSize == null ||
_centeredMessageOffset == null) {
return false;
}
final finalMessageHeight = _centeredMessageSize!.height + _reactionsHeight;
return finalMessageHeight > _verticalSpace!;
}
/// Size of the centered overlay message adjusted for overflow
Size? get _adjustedCenteredMessageSize {
if (_centeredMessageHasOverflow) {
return Size(
_centeredMessageSize!.width,
_verticalSpace! - (AppConfig.toolbarSpacing * 2),
);
}
return _centeredMessageSize;
}
Offset? get _adjustedCenteredMessageOffset {
if (_centeredMessageHasOverflow) {
return Offset(
_centeredMessageOffset!.dx,
_footerHeight + AppConfig.toolbarSpacing,
);
}
return _centeredMessageOffset;
}
// message offset
static const Offset _defaultMessageOffset =
Offset(_messageDefaultLeftMargin, 300);
Offset get _originalMessageOffset {
if (_messageRenderBox == null || !_messageRenderBox!.hasSize) {
return _defaultMessageOffset;
}
return _runWithLogging(
() => _messageRenderBox?.localToGlobal(Offset.zero),
"Error getting message offset",
_defaultMessageOffset,
);
}
Offset get _adjustedOriginalMessageOffset {
if (_messageRenderBox == null || !_messageRenderBox!.hasSize) {
return _defaultMessageOffset;
}
final topOffset = _originalMessageOffset.dy;
final bottomOffset = _originalMessageBottomOffset -
_reactionsHeight -
_selectionButtonsHeight;
final hasHeaderOverflow = topOffset <
(_headerHeight + AppConfig.toolbarSpacing + _audioTranscriptionHeight);
final hasFooterOverflow =
bottomOffset < (_footerHeight + AppConfig.toolbarSpacing);
if (!hasHeaderOverflow && !hasFooterOverflow) {
return Offset(
_ownMessage ? _messageRightOffset : _messageLeftOffset,
bottomOffset,
);
}
if (hasHeaderOverflow) {
final difference = topOffset -
(_headerHeight +
AppConfig.toolbarSpacing +
_audioTranscriptionHeight);
double newBottomOffset = _mediaQuery!.size.height -
_originalMessageOffset.dy +
difference -
_originalMessageSize.height -
_selectionButtonsHeight;
if (newBottomOffset < _footerHeight + AppConfig.toolbarSpacing) {
newBottomOffset = _footerHeight + AppConfig.toolbarSpacing;
}
return Offset(
_ownMessage ? _messageRightOffset : _messageLeftOffset,
newBottomOffset,
);
} else {
return Offset(
_ownMessage ? _messageRightOffset : _messageLeftOffset,
_footerHeight + (AppConfig.toolbarSpacing * 2),
);
}
}
double get _originalMessageBottomOffset =>
_mediaQuery!.size.height -
_originalMessageOffset.dy -
_originalMessageSize.height;
double? get _centeredMessageTopOffset {
if (_mediaQuery == null ||
_adjustedCenteredMessageOffset == null ||
_adjustedCenteredMessageSize == null) {
return null;
}
return _mediaQuery!.size.height -
_adjustedCenteredMessageOffset!.dy -
_adjustedCenteredMessageSize!.height -
_reactionsHeight;
}
double get _messageLeftOffset => max(
_originalMessageOffset.dx - _columnWidth - _horizontalPadding,
0,
);
double get _messageRightOffset {
if (_mediaQuery == null || !_ownMessage) {
return 0;
}
return _mediaQuery!.size.width -
_originalMessageOffset.dx -
_originalMessageSize.width -
_horizontalPadding -
(_showDetails ? FluffyThemes.columnWidth : 0);
}
// measurements for items around the toolbar
double get _horizontalPadding =>
FluffyThemes.isColumnMode(context) ? 8.0 : 0.0;
double get _headerHeight {
return (Theme.of(context).appBarTheme.toolbarHeight ??
AppConfig.defaultHeaderHeight) +
(_mediaQuery?.padding.top ?? 0);
}
double get _footerHeight {
return _inputBarSize + (_mediaQuery?.padding.bottom ?? 0);
}
// measurement for items in the toolbar
bool get _showButtons =>
(widget.pangeaMessageEvent?.shouldShowToolbar ?? false) &&
widget.pangeaMessageEvent?.event.messageType == MessageTypes.Text &&
(widget.pangeaMessageEvent?.messageDisplayLangIsL2 ?? false);
bool get showPracticeButtons =>
_showButtons &&
widget.overlayController.readingAssistanceMode ==
ReadingAssistanceMode.practiceMode;
bool get showSelectionButtons =>
_showButtons &&
[ReadingAssistanceMode.selectMode, null]
.contains(widget.overlayController.readingAssistanceMode);
double get _selectionButtonsHeight {
return showSelectionButtons ? AppConfig.toolbarButtonsHeight : 0;
}
double get _audioTranscriptionHeight {
return widget.pangeaMessageEvent?.isAudioMessage ?? false
? AppConfig.audioTranscriptionMaxHeight
: 0;
}
bool get _hasReactions {
final reactionsEvents = widget.event.aggregatedEvents(
widget.chatController.timeline!,
RelationshipTypes.reaction,
);
return reactionsEvents.where((e) => !e.redacted).isNotEmpty;
}
double get _reactionsHeight => _hasReactions ? 28 : 0;
bool get _ownMessage =>
widget.event.senderId == widget.event.room.client.userID;
double get _readingAssistanceModeOpacity {
switch (_readingAssistanceMode) {
case ReadingAssistanceMode.practiceMode:
case ReadingAssistanceMode.transitionMode:
return 0.8;
case ReadingAssistanceMode.selectMode:
case null:
return 0.4;
}
}
@override
Widget build(BuildContext context) {
if (_messageRenderBox == null || _mediaQuery == null) {
return const SizedBox.shrink();
}
widget.overlayController.maxWidth = _toolbarMaxWidth;
return Stack(
children: [
Positioned.fill(
child: IgnorePointer(
child: AnimatedOpacity(
duration: _animationDuration,
opacity: _readingAssistanceModeOpacity,
child: Container(
height: double.infinity,
width: double.infinity,
color: Colors.black,
),
),
),
),
Padding(
padding: EdgeInsets.only(
left: _horizontalPadding,
right: _horizontalPadding,
),
child: Row(
children: [
Expanded(
child: Stack(
alignment: Alignment.center,
children: [
Column(
children: [
Material(
type: MaterialType.transparency,
child: Column(
children: [
SizedBox(height: _mediaQuery?.padding.top ?? 0),
OverlayHeader(controller: widget.chatController),
],
),
),
const Expanded(
flex: 3,
child: SizedBox.shrink(),
),
Opacity(
opacity: _readingAssistanceMode ==
ReadingAssistanceMode.practiceMode
? 1.0
: 0.0,
child: OverlayCenterContent(
event: widget.event,
messageHeight: null,
messageWidth: null,
maxWidth: widget.overlayController.maxWidth,
overlayController: widget.overlayController,
chatController: widget.chatController,
pangeaMessageEvent: widget.pangeaMessageEvent,
nextEvent: widget.nextEvent,
prevEvent: widget.prevEvent,
hasReactions: _hasReactions,
onChangeMessageSize: _setCenteredMessageSize,
isTransitionAnimation: false,
maxHeight: _mediaQuery!.size.height -
_headerHeight -
_footerHeight -
AppConfig.toolbarSpacing * 2 -
_selectionButtonsHeight,
readingAssistanceMode: _readingAssistanceMode,
),
),
const Expanded(
flex: 1,
child: SizedBox.shrink(),
),
Row(
mainAxisSize: MainAxisSize.min,
children: [
Expanded(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
OverlayFooter(
controller: widget.chatController,
overlayController: widget.overlayController,
showToolbarButtons: showPracticeButtons,
readingAssistanceMode:
_readingAssistanceMode,
),
SizedBox(
height: _mediaQuery?.padding.bottom ?? 0,
),
],
),
),
],
),
],
),
if (_readingAssistanceMode !=
ReadingAssistanceMode.practiceMode &&
_readingAssistanceMode != null)
AnimatedBuilder(
animation:
_overlayOffsetAnimation ?? _animationController,
builder: (context, child) {
return Positioned(
left: _ownMessage
? null
: (_overlayOffsetAnimation?.value)?.dx ??
_messageLeftOffset,
right: _ownMessage
? (_overlayOffsetAnimation?.value)?.dx ??
_messageRightOffset
: null,
bottom: (_overlayOffsetAnimation?.value)?.dy ??
_originalMessageBottomOffset -
_reactionsHeight -
_selectionButtonsHeight,
child: Column(
crossAxisAlignment: _ownMessage
? CrossAxisAlignment.end
: CrossAxisAlignment.start,
children: [
OverlayCenterContent(
event: widget.event,
messageHeight: _originalMessageSize.height,
messageWidth: _originalMessageSize.width,
maxWidth: widget.overlayController.maxWidth,
overlayController: widget.overlayController,
chatController: widget.chatController,
pangeaMessageEvent: widget.pangeaMessageEvent,
nextEvent: widget.nextEvent,
prevEvent: widget.prevEvent,
hasReactions: _hasReactions,
sizeAnimation: _messageSizeAnimation,
isTransitionAnimation: true,
maxHeight: _mediaQuery!.size.height -
_headerHeight -
_footerHeight -
AppConfig.toolbarSpacing * 2 -
_selectionButtonsHeight,
readingAssistanceMode: _readingAssistanceMode,
),
if (showSelectionButtons)
SelectModeButtons(
overlayController: widget.overlayController,
lauchPractice: () {
_setReadingAssistanceMode(
ReadingAssistanceMode.practiceMode,
);
widget.overlayController
.updateSelectedSpan(null);
},
),
],
),
);
},
),
if (showPracticeButtons)
Positioned(
top: 0,
child: IgnorePointer(
child: MeasureRenderBox(
onChange: _setTooltipSize,
child: Opacity(
opacity: 0.0,
child: Container(
constraints: BoxConstraints(
minWidth: 200.0,
maxWidth: _toolbarMaxWidth,
),
child: InstructionsInlineTooltip(
instructionsEnum: widget.overlayController
.toolbarMode.instructionsEnum ??
InstructionsEnum
.readingAssistanceOverview,
bold: true,
),
),
),
),
),
),
if (_centeredMessageTopOffset != null &&
_tooltipSize != null &&
widget.overlayController.toolbarMode !=
MessageMode.noneSelected &&
widget.overlayController.selectedToken == null)
Positioned(
top: max(
((_headerHeight + _centeredMessageTopOffset!) / 2) -
(_tooltipSize!.height / 2),
_headerHeight,
),
child: Container(
constraints: BoxConstraints(
minWidth: 200.0,
maxWidth: widget.overlayController.maxWidth,
),
child: InstructionsInlineTooltip(
instructionsEnum: widget.overlayController
.toolbarMode.instructionsEnum ??
InstructionsEnum.readingAssistanceOverview,
bold: true,
),
),
),
],
),
),
if (_showDetails)
const SizedBox(
width: FluffyThemes.columnWidth,
),
],
),
),
],
);
}
}