* 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 commit2625e89a33. * 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 commit0f000f952f. * Revert "build: Update flutter 3.29.2" This reverts commitbfd23952b7. * 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 commitcd12f773fe. * 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, sincee93fdebe20upgraded 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>
583 lines
20 KiB
Dart
583 lines
20 KiB
Dart
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
|
|
import 'package:emojis/emoji.dart';
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
import 'package:flutter_typeahead/flutter_typeahead.dart';
|
|
import 'package:matrix/matrix.dart';
|
|
import 'package:slugify/slugify.dart';
|
|
|
|
import 'package:fluffychat/pangea/choreographer/widgets/igc/pangea_text_controller.dart';
|
|
import 'package:fluffychat/pangea/toolbar/utils/shrinkable_text.dart';
|
|
import 'package:fluffychat/utils/markdown_context_builder.dart';
|
|
import 'package:fluffychat/widgets/mxc_image.dart';
|
|
import '../../widgets/avatar.dart';
|
|
import '../../widgets/matrix.dart';
|
|
import 'command_hints.dart';
|
|
|
|
class InputBar extends StatelessWidget {
|
|
final Room room;
|
|
final int? minLines;
|
|
final int? maxLines;
|
|
final TextInputType? keyboardType;
|
|
final TextInputAction? textInputAction;
|
|
final ValueChanged<String>? onSubmitted;
|
|
final ValueChanged<Uint8List?>? onSubmitImage;
|
|
final FocusNode? focusNode;
|
|
// #Pangea
|
|
// final TextEditingController? controller;
|
|
final PangeaTextController? controller;
|
|
final String hintText;
|
|
// Pangea#
|
|
final InputDecoration? decoration;
|
|
final ValueChanged<String>? onChanged;
|
|
final bool? autofocus;
|
|
final bool readOnly;
|
|
|
|
const InputBar({
|
|
required this.room,
|
|
this.minLines,
|
|
this.maxLines,
|
|
this.keyboardType,
|
|
this.onSubmitted,
|
|
this.onSubmitImage,
|
|
this.focusNode,
|
|
this.controller,
|
|
this.decoration,
|
|
this.onChanged,
|
|
this.autofocus,
|
|
this.textInputAction,
|
|
this.readOnly = false,
|
|
// #Pangea
|
|
required this.hintText,
|
|
// Pangea#
|
|
super.key,
|
|
});
|
|
|
|
List<Map<String, String?>> getSuggestions(String text) {
|
|
if (controller!.selection.baseOffset !=
|
|
controller!.selection.extentOffset ||
|
|
controller!.selection.baseOffset < 0) {
|
|
return []; // no entries if there is selected text
|
|
}
|
|
final searchText =
|
|
controller!.text.substring(0, controller!.selection.baseOffset);
|
|
final ret = <Map<String, String?>>[];
|
|
const maxResults = 30;
|
|
|
|
final commandMatch = RegExp(r'^/(\w*)$').firstMatch(searchText);
|
|
if (commandMatch != null) {
|
|
final commandSearch = commandMatch[1]!.toLowerCase();
|
|
for (final command in room.client.commands.keys) {
|
|
if (command.contains(commandSearch)) {
|
|
ret.add({
|
|
'type': 'command',
|
|
'name': command,
|
|
});
|
|
}
|
|
|
|
if (ret.length > maxResults) return ret;
|
|
}
|
|
}
|
|
final emojiMatch =
|
|
RegExp(r'(?:\s|^):(?:([-\w]+)~)?([-\w]+)$').firstMatch(searchText);
|
|
if (emojiMatch != null) {
|
|
final packSearch = emojiMatch[1];
|
|
final emoteSearch = emojiMatch[2]!.toLowerCase();
|
|
final emotePacks = room.getImagePacks(ImagePackUsage.emoticon);
|
|
if (packSearch == null || packSearch.isEmpty) {
|
|
for (final pack in emotePacks.entries) {
|
|
for (final emote in pack.value.images.entries) {
|
|
if (emote.key.toLowerCase().contains(emoteSearch)) {
|
|
ret.add({
|
|
'type': 'emote',
|
|
'name': emote.key,
|
|
'pack': pack.key,
|
|
'pack_avatar_url': pack.value.pack.avatarUrl?.toString(),
|
|
'pack_display_name': pack.value.pack.displayName ?? pack.key,
|
|
'mxc': emote.value.url.toString(),
|
|
});
|
|
}
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
} else if (emotePacks[packSearch] != null) {
|
|
for (final emote in emotePacks[packSearch]!.images.entries) {
|
|
if (emote.key.toLowerCase().contains(emoteSearch)) {
|
|
ret.add({
|
|
'type': 'emote',
|
|
'name': emote.key,
|
|
'pack': packSearch,
|
|
'pack_avatar_url':
|
|
emotePacks[packSearch]!.pack.avatarUrl?.toString(),
|
|
'pack_display_name':
|
|
emotePacks[packSearch]!.pack.displayName ?? packSearch,
|
|
'mxc': emote.value.url.toString(),
|
|
});
|
|
}
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
// aside of emote packs, also propose normal (tm) unicode emojis
|
|
final matchingUnicodeEmojis = Emoji.all()
|
|
.where(
|
|
(element) => [element.name, ...element.keywords]
|
|
.any((element) => element.toLowerCase().contains(emoteSearch)),
|
|
)
|
|
.toList();
|
|
// sort by the index of the search term in the name in order to have
|
|
// best matches first
|
|
// (thanks for the hint by github.com/nextcloud/circles devs)
|
|
matchingUnicodeEmojis.sort((a, b) {
|
|
final indexA = a.name.indexOf(emoteSearch);
|
|
final indexB = b.name.indexOf(emoteSearch);
|
|
if (indexA == -1 || indexB == -1) {
|
|
if (indexA == indexB) return 0;
|
|
if (indexA == -1) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
return indexA.compareTo(indexB);
|
|
});
|
|
for (final emoji in matchingUnicodeEmojis) {
|
|
ret.add({
|
|
'type': 'emoji',
|
|
'emoji': emoji.char,
|
|
// don't include sub-group names, splitting at `:` hence
|
|
'label': '${emoji.char} - ${emoji.name.split(':').first}',
|
|
'current_word': ':$emoteSearch',
|
|
});
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
final userMatch = RegExp(r'(?:\s|^)@([-\w]+)$').firstMatch(searchText);
|
|
if (userMatch != null) {
|
|
final userSearch = userMatch[1]!.toLowerCase();
|
|
for (final user in room.getParticipants()) {
|
|
if ((user.displayName != null &&
|
|
(user.displayName!.toLowerCase().contains(userSearch) ||
|
|
slugify(user.displayName!.toLowerCase())
|
|
.contains(userSearch))) ||
|
|
user.id.split(':')[0].toLowerCase().contains(userSearch)) {
|
|
ret.add({
|
|
'type': 'user',
|
|
'mxid': user.id,
|
|
'mention': user.mention,
|
|
'displayname': user.displayName,
|
|
'avatar_url': user.avatarUrl?.toString(),
|
|
});
|
|
}
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
final roomMatch = RegExp(r'(?:\s|^)#([-\w]+)$').firstMatch(searchText);
|
|
if (roomMatch != null) {
|
|
final roomSearch = roomMatch[1]!.toLowerCase();
|
|
for (final r in room.client.rooms) {
|
|
if (r.getState(EventTypes.RoomTombstone) != null) {
|
|
continue; // we don't care about tombstoned rooms
|
|
}
|
|
final state = r.getState(EventTypes.RoomCanonicalAlias);
|
|
if ((state != null &&
|
|
((state.content['alias'] is String &&
|
|
state.content
|
|
.tryGet<String>('alias')!
|
|
.split(':')[0]
|
|
.toLowerCase()
|
|
.contains(roomSearch)) ||
|
|
(state.content['alt_aliases'] is List &&
|
|
(state.content['alt_aliases'] as List).any(
|
|
(l) =>
|
|
l is String &&
|
|
l
|
|
.split(':')[0]
|
|
.toLowerCase()
|
|
.contains(roomSearch),
|
|
)))) ||
|
|
(r.name.toLowerCase().contains(roomSearch))) {
|
|
ret.add({
|
|
'type': 'room',
|
|
'mxid': (r.canonicalAlias.isNotEmpty) ? r.canonicalAlias : r.id,
|
|
'displayname': r.getLocalizedDisplayname(),
|
|
'avatar_url': r.avatar?.toString(),
|
|
});
|
|
}
|
|
if (ret.length > maxResults) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
Widget buildSuggestion(
|
|
BuildContext context,
|
|
Map<String, String?> suggestion,
|
|
Client? client,
|
|
) {
|
|
final theme = Theme.of(context);
|
|
const size = 30.0;
|
|
// #Pangea
|
|
// const padding = EdgeInsets.all(4.0);
|
|
const padding = EdgeInsets.all(8.0);
|
|
// Pangea#
|
|
if (suggestion['type'] == 'command') {
|
|
final command = suggestion['name']!;
|
|
final hint = commandHint(L10n.of(context), command);
|
|
return Tooltip(
|
|
message: hint,
|
|
waitDuration: const Duration(days: 1), // don't show on hover
|
|
child: Container(
|
|
padding: padding,
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
commandExample(command),
|
|
style: const TextStyle(fontFamily: 'RobotoMono'),
|
|
),
|
|
Text(
|
|
hint,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
style: theme.textTheme.bodySmall,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'emoji') {
|
|
final label = suggestion['label']!;
|
|
return Tooltip(
|
|
message: label,
|
|
waitDuration: const Duration(days: 1), // don't show on hover
|
|
child: Container(
|
|
padding: padding,
|
|
child: Text(label, style: const TextStyle(fontFamily: 'RobotoMono')),
|
|
),
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'emote') {
|
|
return Container(
|
|
padding: padding,
|
|
child: Row(
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
children: <Widget>[
|
|
MxcImage(
|
|
// ensure proper ordering ...
|
|
key: ValueKey(suggestion['name']),
|
|
uri: suggestion['mxc'] is String
|
|
? Uri.parse(suggestion['mxc'] ?? '')
|
|
: null,
|
|
width: size,
|
|
height: size,
|
|
isThumbnail: false,
|
|
),
|
|
const SizedBox(width: 6),
|
|
Text(suggestion['name']!),
|
|
Expanded(
|
|
child: Align(
|
|
alignment: Alignment.centerRight,
|
|
child: Opacity(
|
|
opacity: suggestion['pack_avatar_url'] != null ? 0.8 : 0.5,
|
|
child: suggestion['pack_avatar_url'] != null
|
|
? Avatar(
|
|
mxContent: Uri.tryParse(
|
|
suggestion.tryGet<String>('pack_avatar_url') ?? '',
|
|
),
|
|
name: suggestion.tryGet<String>('pack_display_name'),
|
|
size: size * 0.9,
|
|
client: client,
|
|
)
|
|
: Text(suggestion['pack_display_name']!),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'user' || suggestion['type'] == 'room') {
|
|
final url = Uri.parse(suggestion['avatar_url'] ?? '');
|
|
return Container(
|
|
padding: padding,
|
|
child: Row(
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
children: <Widget>[
|
|
Avatar(
|
|
mxContent: url,
|
|
name: suggestion.tryGet<String>('displayname') ??
|
|
suggestion.tryGet<String>('mxid'),
|
|
size: size,
|
|
client: client,
|
|
),
|
|
const SizedBox(width: 6),
|
|
// #Pangea
|
|
// Text(suggestion['displayname'] ?? suggestion['mxid']!),
|
|
Flexible(
|
|
child: Text(
|
|
suggestion['displayname'] ?? suggestion['mxid']!,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
),
|
|
// Pangea#
|
|
],
|
|
),
|
|
);
|
|
}
|
|
return const SizedBox.shrink();
|
|
}
|
|
|
|
void insertSuggestion(_, Map<String, String?> suggestion) {
|
|
final replaceText =
|
|
controller!.text.substring(0, controller!.selection.baseOffset);
|
|
var startText = '';
|
|
final afterText = replaceText == controller!.text
|
|
? ''
|
|
: controller!.text.substring(controller!.selection.baseOffset + 1);
|
|
var insertText = '';
|
|
if (suggestion['type'] == 'command') {
|
|
insertText = '${suggestion['name']!} ';
|
|
startText = replaceText.replaceAllMapped(
|
|
RegExp(r'^(/\w*)$'),
|
|
(Match m) => '/$insertText',
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'emoji') {
|
|
insertText = '${suggestion['emoji']!} ';
|
|
startText = replaceText.replaceAllMapped(
|
|
// #Pangea
|
|
RegExp(suggestion['current_word']!, caseSensitive: false),
|
|
// suggestion['current_word']!,
|
|
// Pangea#
|
|
(Match m) => insertText,
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'emote') {
|
|
var isUnique = true;
|
|
final insertEmote = suggestion['name'];
|
|
final insertPack = suggestion['pack'];
|
|
final emotePacks = room.getImagePacks(ImagePackUsage.emoticon);
|
|
for (final pack in emotePacks.entries) {
|
|
if (pack.key == insertPack) {
|
|
continue;
|
|
}
|
|
for (final emote in pack.value.images.entries) {
|
|
if (emote.key == insertEmote) {
|
|
isUnique = false;
|
|
break;
|
|
}
|
|
}
|
|
if (!isUnique) {
|
|
break;
|
|
}
|
|
}
|
|
insertText = ':${isUnique ? '' : '${insertPack!}~'}$insertEmote: ';
|
|
startText = replaceText.replaceAllMapped(
|
|
RegExp(r'(\s|^)(:(?:[-\w]+~)?[-\w]+)$'),
|
|
(Match m) => '${m[1]}$insertText',
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'user') {
|
|
insertText = '${suggestion['mention']!} ';
|
|
startText = replaceText.replaceAllMapped(
|
|
RegExp(r'(\s|^)(@[-\w]+)$'),
|
|
(Match m) => '${m[1]}$insertText',
|
|
);
|
|
}
|
|
if (suggestion['type'] == 'room') {
|
|
insertText = '${suggestion['mxid']!} ';
|
|
startText = replaceText.replaceAllMapped(
|
|
RegExp(r'(\s|^)(#[-\w]+)$'),
|
|
(Match m) => '${m[1]}$insertText',
|
|
);
|
|
}
|
|
if (insertText.isNotEmpty && startText.isNotEmpty) {
|
|
controller!.text = startText + afterText;
|
|
controller!.selection = TextSelection(
|
|
baseOffset: startText.length,
|
|
extentOffset: startText.length,
|
|
);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
// #Pangea
|
|
final enableAutocorrect = MatrixState
|
|
.pangeaController.userController.profile.toolSettings.enableAutocorrect;
|
|
// Pangea#
|
|
return TypeAheadField<Map<String, String?>>(
|
|
direction: VerticalDirection.up,
|
|
hideOnEmpty: true,
|
|
hideOnLoading: true,
|
|
// #Pangea
|
|
// if should obscure text (to make it looks that a message has been sent after sending fake message),
|
|
// use hideTextController
|
|
|
|
// controller: controller,
|
|
controller:
|
|
(controller?.choreographer.chatController.obscureText) ?? false
|
|
? controller?.choreographer.chatController.hideTextController
|
|
: controller,
|
|
// Pangea#
|
|
focusNode: focusNode,
|
|
hideOnSelect: false,
|
|
debounceDuration: const Duration(milliseconds: 50),
|
|
// show suggestions after 50ms idle time (default is 300)
|
|
// #Pangea
|
|
builder: (context, _, focusNode) {
|
|
final textField = TextField(
|
|
enableSuggestions: enableAutocorrect,
|
|
readOnly: controller != null &&
|
|
(controller!.choreographer.isRunningIT ||
|
|
controller!.choreographer.chatController.obscureText),
|
|
autocorrect: enableAutocorrect,
|
|
controller:
|
|
(controller?.choreographer.chatController.obscureText) ?? false
|
|
? controller?.choreographer.chatController.hideTextController
|
|
: controller,
|
|
focusNode: focusNode,
|
|
contextMenuBuilder: (c, e) => markdownContextBuilder(
|
|
c,
|
|
e,
|
|
_,
|
|
),
|
|
contentInsertionConfiguration: ContentInsertionConfiguration(
|
|
onContentInserted: (KeyboardInsertedContent content) {
|
|
final data = content.data;
|
|
if (data == null) return;
|
|
|
|
final file = MatrixFile(
|
|
mimeType: content.mimeType,
|
|
bytes: data,
|
|
name: content.uri.split('/').last,
|
|
);
|
|
room.sendFileEvent(
|
|
file,
|
|
shrinkImageMaxDimension: 1600,
|
|
);
|
|
},
|
|
),
|
|
minLines: minLines,
|
|
maxLines: maxLines,
|
|
keyboardType: keyboardType!,
|
|
textInputAction: textInputAction,
|
|
autofocus: autofocus!,
|
|
inputFormatters: [
|
|
//LengthLimitingTextInputFormatter((maxPDUSize / 3).floor()),
|
|
//setting max character count to 1000
|
|
//after max, nothing else can be typed
|
|
LengthLimitingTextInputFormatter(1000),
|
|
],
|
|
onSubmitted: (text) {
|
|
// fix for library for now
|
|
// it sets the types for the callback incorrectly
|
|
onSubmitted!(text);
|
|
},
|
|
style: controller?.exceededMaxLength ?? false
|
|
? const TextStyle(color: Colors.red)
|
|
: null,
|
|
onTap: () {
|
|
controller?.onInputTap(
|
|
context,
|
|
fNode: focusNode,
|
|
);
|
|
},
|
|
decoration: decoration!,
|
|
onChanged: (text) {
|
|
// fix for the library for now
|
|
// it sets the types for the callback incorrectly
|
|
onChanged!(text);
|
|
},
|
|
textCapitalization: TextCapitalization.sentences,
|
|
);
|
|
// fix for issue with typing not working sometimes on Firefox and Safari
|
|
return Stack(
|
|
alignment: Alignment.centerLeft,
|
|
children: [
|
|
if (controller != null && controller!.text.isEmpty)
|
|
Padding(
|
|
padding: const EdgeInsets.only(left: 8.0),
|
|
child: ShrinkableText(
|
|
text: hintText,
|
|
maxWidth: double.infinity,
|
|
style: Theme.of(context).textTheme.bodyLarge?.copyWith(
|
|
color: Theme.of(context).disabledColor,
|
|
),
|
|
),
|
|
),
|
|
kIsWeb ? SelectionArea(child: textField) : textField,
|
|
],
|
|
);
|
|
},
|
|
// builder: (context, controller, focusNode) => TextField(
|
|
// controller: controller,
|
|
// focusNode: focusNode,
|
|
// contextMenuBuilder: (c, e) => markdownContextBuilder(c, e, controller),
|
|
// contentInsertionConfiguration: ContentInsertionConfiguration(
|
|
// onContentInserted: (KeyboardInsertedContent content) {
|
|
// final data = content.data;
|
|
// if (data == null) return;
|
|
|
|
// final file = MatrixFile(
|
|
// mimeType: content.mimeType,
|
|
// bytes: data,
|
|
// name: content.uri.split('/').last,
|
|
// );
|
|
// room.sendFileEvent(
|
|
// file,
|
|
// shrinkImageMaxDimension: 1600,
|
|
// );
|
|
// },
|
|
// ),
|
|
// minLines: minLines,
|
|
// maxLines: maxLines,
|
|
// keyboardType: keyboardType!,
|
|
// textInputAction: textInputAction,
|
|
// autofocus: autofocus!,
|
|
// inputFormatters: [
|
|
// LengthLimitingTextInputFormatter((maxPDUSize / 3).floor()),
|
|
// ],
|
|
// onSubmitted: (text) {
|
|
// // fix for library for now
|
|
// // it sets the types for the callback incorrectly
|
|
// onSubmitted!(text);
|
|
// },
|
|
// decoration: decoration!,
|
|
// onChanged: (text) {
|
|
// // fix for the library for now
|
|
// // it sets the types for the callback incorrectly
|
|
// onChanged!(text);
|
|
// },
|
|
// textCapitalization: TextCapitalization.sentences,
|
|
// ),
|
|
// Pangea#
|
|
suggestionsCallback: getSuggestions,
|
|
itemBuilder: (c, s) => buildSuggestion(c, s, Matrix.of(context).client),
|
|
onSelected: (Map<String, String?> suggestion) =>
|
|
insertSuggestion(context, suggestion),
|
|
errorBuilder: (BuildContext context, Object? error) =>
|
|
const SizedBox.shrink(),
|
|
loadingBuilder: (BuildContext context) => const SizedBox.shrink(),
|
|
// fix loading briefly flickering a dark box
|
|
emptyBuilder: (BuildContext context) =>
|
|
const SizedBox.shrink(), // fix loading briefly showing no suggestions
|
|
);
|
|
}
|
|
}
|