Merge branch 'main' of https://github.com/krille-chan/fluffychat
Some checks failed
Main Deploy Workflow / deploy_web (push) Has been cancelled
Main Deploy Workflow / deploy_playstore_internal (push) Has been cancelled

This commit is contained in:
Alexey 2026-04-23 10:59:25 +03:00
commit e12723fdaa
98 changed files with 1450 additions and 1014 deletions

View file

@ -1,5 +1,3 @@
*Thank you so much for your contribution to FluffyChat ❤️❤️❤️*
- [ ] I have read and understood the [contributing guidelines](https://github.com/krille-chan/fluffychat/blob/main/CONTRIBUTING.md).
### Pull Request has been tested on:

View file

@ -128,10 +128,7 @@ jobs:
run: sudo xcode-select --switch /Applications/Xcode_16.4.app
- run: brew install sqlcipher
- uses: moonrepo/setup-rust@v1
- name: Add Firebase Messaging
run: |
flutter pub add fcm_shared_isolate:0.1.0
sed -i '' 's,//<GOOGLE_SERVICES>,,g' lib/utils/background_push.dart
- run: ./scripts/add-firebase-messaging.sh
- run: flutter pub get
- run: flutter build ios --no-codesign

View file

@ -29,7 +29,7 @@ jobs:
run: ./scripts/prepare-web.sh
- run: rm ./assets/vodozemac/.gitignore
- name: Build Release Web
run: flutter build web --dart-define=FLUTTER_WEB_CANVASKIT_URL=canvaskit/ --release --source-maps
run: flutter build web --dart-define=FLUTTER_WEB_CANVASKIT_URL=canvaskit/ --release --source-maps --base-href "/web/"
- name: Create archive
run: tar -czf fluffychat-web.tar.gz build/web/
- name: Upload Web Build
@ -49,8 +49,8 @@ jobs:
- name: Clone fluffychat website
run: |
git clone https://github.com/krille-chan/fluffychat-website.git
cp CHANGELOG.md fluffychat-website/
cp PRIVACY.md fluffychat-website/
cat CHANGELOG.md >> fluffychat-website/src/changelog.md
cat PRIVACY.md >> fluffychat-website/src/privacy.md
- name: Build website
working-directory: fluffychat-website
run: |

View file

@ -1,2 +1,2 @@
environment:
flutter: 3.41.5 # Keep in sync with snap/snapcraft.yaml
flutter: 3.41.7 # Keep in sync with snap/snapcraft.yaml

View file

@ -1,3 +1,6 @@
## v2.5.1
Update to latest version of fcm_shared_isolate to fix push on iOS.
## v2.5.0
FluffyChat 2.5.0 introduces a new homeserver picker for onboarding, better image compression performance and several smaller new features, design adjustments and bug fixes.

View file

@ -36,7 +36,6 @@ analyzer:
- dart_code_linter
errors:
todo: ignore
use_build_context_synchronously: ignore
exclude:
- lib/l10n/*.dart

View file

@ -12,6 +12,7 @@
"audioRecordingSamplingRate": 44100,
"renderHtml": true,
"fontSizeFactor": 1,
"messagePreviewMaxLines": 128,
"hideRedactedEvents": false,
"hideUnknownEvents": true,
"separateChatTypes": false,

View file

@ -3,8 +3,8 @@ GEM
specs:
CFPropertyList (3.0.3)
abbrev (0.1.2)
addressable (2.8.4)
public_suffix (>= 2.0.2, < 6.0)
addressable (2.9.0)
public_suffix (>= 2.0.2, < 8.0)
artifactory (3.0.15)
atomos (0.1.3)
aws-eventstream (1.4.0)
@ -168,7 +168,7 @@ GEM
os (1.1.1)
ostruct (0.6.3)
plist (3.6.0)
public_suffix (5.0.3)
public_suffix (7.0.5)
rake (13.0.3)
representable (3.1.1)
declarative (< 0.1.0)

View file

@ -12,5 +12,7 @@ import Flutter
func didInitializeImplicitFlutterEngine(_ engineBridge: FlutterImplicitEngineBridge) {
GeneratedPluginRegistrant.register(with: engineBridge.pluginRegistry)
}
// From https://pub.dev/packages/flutter_local_notifications#-ios-setup
UNUserNotificationCenter.current().delegate = self as? UNUserNotificationCenterDelegate }
}

View file

@ -172,8 +172,8 @@ abstract class AppRoutes {
context,
state,
NewPrivateChat(
key: ValueKey('new_chat_${state.uri.query}'),
deeplink: state.uri.queryParameters['deeplink'],
key: ValueKey('new_chat_${state.uri.fragment}'),
deeplink: state.uri.fragment,
),
),
redirect: loggedOutRedirect,

View file

@ -9,6 +9,9 @@ import 'package:shared_preferences/shared_preferences.dart';
enum AppSettings<T> {
textMessageMaxLength<int>('textMessageMaxLength', 16384),
/// Max lines for unselected HTML/text bubbles; 0 = unlimited (no fade).
messagePreviewMaxLines<int>('chat.fluffy.message_preview_max_lines', 128),
audioRecordingNumChannels<int>('audioRecordingNumChannels', 1),
audioRecordingAutoGain<bool>('audioRecordingAutoGain', true),
audioRecordingEchoCancel<bool>('audioRecordingEchoCancel', false),
@ -68,7 +71,9 @@ enum AppSettings<T> {
tos<String>('chat.fluffy.tos_url', 'https://fluffychat.im/en/tos'),
sendTimelineEventTimeout<int>('chat.fluffy.send_timeline_event_timeout', 15),
lastSeenSupportBanner<int>('chat.fluffy.last_seen_support_banner', 0),
supportBannerOptOut<bool>('chat.fluffy.support_banner_opt_out', false);
supportBannerOptOut<bool>('chat.fluffy.support_banner_opt_out', false),
webNotificationSound<bool>('chat.fluffy.web_notification_sound', true),
chatFilter<String>('chat.fluffy.chat_filter', 'allChats');
final String key;
final T defaultValue;

File diff suppressed because it is too large Load diff

View file

@ -72,7 +72,7 @@
"type": "String",
"placeholders": {}
},
"areGuestsAllowedToJoin": "Es pot entrar al xat com a convidadi",
"areGuestsAllowedToJoin": "Es pot entrar al xat com a convidadi?",
"@areGuestsAllowedToJoin": {
"type": "String",
"placeholders": {}
@ -1781,7 +1781,7 @@
"type": "String",
"description": "Usage hint for the command /clearcache"
},
"commandHint_join": "Uneix-te a la sala",
"commandHint_join": "Uneix-te a la sala indicada",
"@commandHint_join": {
"type": "String",
"description": "Usage hint for the command /join"
@ -2754,7 +2754,7 @@
}
},
"federationBaseUrl": "URL base de federació",
"clientWellKnownInformation": "Informació de",
"clientWellKnownInformation": "Informació coneguda del client:",
"baseUrl": "URL base",
"identityServer": "Servidor d'identitats:",
"versionWithNumber": "Versió: {version}",
@ -2774,5 +2774,39 @@
"signUpGreeting": "El FluffyChat és descentralitzat! Tria un servidor on vulguis crear-t'hi un compte, i som-hi!",
"signInGreeting": "Si ja tens un compte a Matrix, benvingudi! Tria el teu servidor i inicia-hi sessió.",
"appIntro": "Pots xatejar amb lis tevis amiguis amb Fluffychat. És una app de missatgeria [matrix] descentralitzada! Llegeix-ne més a https://matrix.org si vols, o inicia sessió.",
"theProcessWasCanceled": "S'ha canceŀlat el procés."
}
"theProcessWasCanceled": "S'ha canceŀlat el procés.",
"join": "Entra",
"searchOrEnterHomeserverAddress": "Cerca o introdueix l'adreça del teu servidor",
"matrixId": "ID de Matrix",
"setPowerLevel": "Concedeix permisos",
"makeModerator": "Fes moderadori",
"makeAdmin": "Fes admin",
"removeModeratorRights": "Treu els drets de moderadori",
"removeAdminRights": "Treu els drets d'admin",
"powerLevel": "Nivell de permisos",
"setPowerLevelDescription": "Els nivells de permisos defineixen què pot fer uni membre d'aquesta sala, i es defineix per un número entre 0 i 100.",
"owner": "Propietàriï",
"mute": "Silencia",
"@mute": {
"description": "This should be a very short string because there is not much space in the button!"
},
"createNewChat": "Crea un nou xat",
"reset": "Reseteja",
"supportFluffyChat": "Dona suport a FluffyChat",
"support": "Aporta",
"fluffyChatSupportBannerMessage": "El FluffyChat necessita la teva ajuda!\n❤\nFluffyChat serà sempre gratuït, però el seu desenvolupament i allotjament costa diners.\nEl futur del projecte depèn del suport de persones com tu.",
"skipSupportingFluffyChat": "Ignora el suport a FluffyChat",
"iDoNotWantToSupport": "No vull donar suport",
"iAlreadySupportFluffyChat": "Ja estic donant-hi suport",
"setLowPriority": "Estableix una prioritat baixa",
"unsetLowPriority": "Restableix la prioritat",
"removeCallFromChat": "Treu la trucada del xat",
"removeCallFromChatDescription": "Vols treure la trucada del xat per a totis lis membres?",
"removeCallForEveryone": "Treu la trucada per tothom",
"startVoiceCall": "Inicia una trucada",
"startVideoCall": "Fes una videotrucada",
"joinVoiceCall": "Fica't a la trucada",
"joinVideoCall": "Fica't a la videotrucada",
"live": "En directe",
"playSoundOnNotification": "Notificacions sonores"
}

View file

@ -2802,5 +2802,10 @@
"startVideoCall": "Start video call",
"joinVoiceCall": "Join voice call",
"joinVideoCall": "Join video call",
"live": "Live"
"live": "Live",
"playSoundOnNotification": "Play sound on notification",
"addTag": "Add tag",
"removeTag": "Remove tag",
"tagName": "Tag name",
"createNewTag": "Create new tag"
}

View file

@ -1709,7 +1709,7 @@
"type": "String",
"placeholders": {}
},
"oopsPushError": "¡UPS¡ Desafortunadamente, se produjo un error al configurar las notificaciones push.",
"oopsPushError": "¡Ups! Desafortunadamente, se produjo un error al configurar las notificaciones push.",
"@oopsPushError": {
"type": "String",
"placeholders": {}
@ -1867,7 +1867,7 @@
"type": "String",
"placeholders": {}
},
"pleaseEnterYourPin": "Por favor ingrese su PIN",
"pleaseEnterYourPin": "Por favor ingrese su pin",
"@pleaseEnterYourPin": {
"type": "String",
"placeholders": {}
@ -2576,7 +2576,7 @@
"contactServerAdmin": "Contactar con el administrador del servidor",
"contactServerSecurity": "Contactar con seguridad del servidor",
"supportPage": "Página de atención",
"invalidUrl": "URL incorrecta",
"invalidUrl": "Url incorrecto",
"addLink": "Añadir enlace",
"unableToJoinChat": "No se puede entrar al chat. Puede que la otra parte ya haya cerrado la conversación.",
"waitingForServer": "Esperando al servidor...",
@ -2718,7 +2718,7 @@
"addAnswerOption": "Añadir respuesta",
"allowMultipleAnswers": "Permitir varias respuestas",
"pollHasBeenEnded": "La encuesta ha terminado",
"countVotes": "{count, plural, =1{One vote} other{{count} votes}}",
"countVotes": "{count, plural, =1{Un voto} other{{count} votos}}",
"@countVotes": {
"type": "int",
"placeholders": {
@ -2786,5 +2786,40 @@
"theProcessWasCanceled": "El proceso se ha cancelado.",
"join": "Unirse",
"searchOrEnterHomeserverAddress": "Buscar o pon la dirección de tu servidor local",
"matrixId": "Matrix ID"
}
"matrixId": "Matrix ID",
"setPowerLevel": "Establecer nivel de poder",
"makeModerator": "Convertir en Moderador",
"makeAdmin": "Convertir en administrador",
"removeModeratorRights": "Remover derechos de moderador",
"removeAdminRights": "Remover derechos de administrador",
"powerLevel": "Nivel de Poder",
"setPowerLevelDescription": "El nivel de poder define el nivel de acciones de un miembro, usualmente esta en el rango entre 0 a 100.",
"owner": "Dueño",
"mute": "Silenciar",
"@mute": {
"description": "This should be a very short string because there is not much space in the button!"
},
"createNewChat": "Crear nuevo chat",
"reset": "Resetear",
"supportFluffyChat": "Apoyar FluffyChat",
"support": "Apoyar",
"fluffyChatSupportBannerMessage": "FluffyChat necesita TU ayuda!\n❤\nFluffyChat siempre sera gratis, pero el desarrollo y mantenimiento cuesta dinero.\nEl futuro del proyecto depende del apoyo de personas como tu.",
"skipSupportingFluffyChat": "Omitir apoyo a FluffyChat",
"iDoNotWantToSupport": "No quiero apoyar",
"iAlreadySupportFluffyChat": "Ya apoyo FluffyChat",
"setLowPriority": "Colocar baja prioridad",
"unsetLowPriority": "Desactivar baja prioridad",
"removeCallFromChat": "Remover llamadas del chat",
"removeCallFromChatDescription": "Deseas remover la llamada del chat para todos los miembros?",
"removeCallForEveryone": "Remover llamadas para todos",
"startVoiceCall": "Iniciar llamada",
"startVideoCall": "Iniciar videollamada",
"joinVoiceCall": "Ingresar a llamada",
"joinVideoCall": "Ingresar a videollamada",
"live": "En Vivo",
"playSoundOnNotification": "Sonido en notificación",
"addTag": "Agregar etiqueta",
"removeTag": "Remover etiqueta",
"tagName": "Nombre de etiqueta",
"createNewTag": "Crear nueva etiqueta"
}

View file

@ -2808,5 +2808,10 @@
"startVideoCall": "Algata videokõne",
"joinVoiceCall": "Liitu häälkõnega",
"joinVideoCall": "Liitu videokõnega",
"live": "Reaalajas"
"live": "Reaalajas",
"playSoundOnNotification": "Lisa teavitusele helimärguanne",
"addTag": "Lisa silt",
"removeTag": "Eemalda silt",
"tagName": "Sildi nimi",
"createNewTag": "Lisa uus silt"
}

View file

@ -2792,5 +2792,22 @@
"description": "This should be a very short string because there is not much space in the button!"
},
"createNewChat": "Sortu txat berria",
"reset": "Berrezarri"
}
"reset": "Berrezarri",
"supportFluffyChat": "Eman babesa FluffyChat-i",
"support": "Lagundu",
"fluffyChatSupportBannerMessage": "FluffyChat-ek ZURE laguntza behar du!\n❤\nFluffyChat beti izango da doakoa, baina garapenak eta ostatatzeak dirua eskatzen du.\nProiektuaren etorkizuna zu bezalako pertsonen babesaren menpe dago.",
"iDoNotWantToSupport": "Ez diot babesik eman nahi",
"iAlreadySupportFluffyChat": "Laguntzen ari naiz dagoeneko FluffyChat",
"setLowPriority": "Ezarri lehentasun baxua",
"unsetLowPriority": "Kendu lehentasun baxua",
"removeCallFromChat": "Kendu deia txatetik",
"removeCallFromChatDescription": "Kide guztientzat kendu nahi al duzu deia txatetik?",
"removeCallForEveryone": "Kendu deia guztientzat",
"startVoiceCall": "Hasi ahots-deia",
"startVideoCall": "Hasi bideo-deia",
"joinVoiceCall": "Batu ahots-deira",
"joinVideoCall": "Batu bideo-deira",
"live": "Zuzenean",
"playSoundOnNotification": "Jo soinua jakinarazpenekin",
"skipSupportingFluffyChat": "Muzin egin FluffyChat-en laguntza eskaerari"
}

View file

@ -2806,5 +2806,18 @@
"iDoNotWantToSupport": "Nílim ag iarraidh tacú leis",
"iAlreadySupportFluffyChat": "Tacaím le FluffyChat cheana féin",
"setLowPriority": "Socraigh tosaíocht íseal",
"unsetLowPriority": "Díshuiteáil tosaíocht íseal"
}
"unsetLowPriority": "Díshuiteáil tosaíocht íseal",
"removeCallFromChat": "Bain glao den chomhrá",
"removeCallFromChatDescription": "Ar mhaith leat an glao a bhaint den chomhrá do gach ball?",
"removeCallForEveryone": "Bain glao do gach duine",
"startVoiceCall": "Tosaigh glao gutha",
"startVideoCall": "Tosaigh glao físe",
"joinVoiceCall": "Glac páirt i nglao gutha",
"joinVideoCall": "Glac páirt i nglao físe",
"live": "Beo",
"playSoundOnNotification": "Seinn fuaim ar fhógra",
"addTag": "Cuir clib leis",
"removeTag": "Bain an chlib",
"tagName": "Ainm an chlib",
"createNewTag": "Cruthaigh clib nua"
}

View file

@ -2808,5 +2808,10 @@
"startVideoCall": "Iniciar chamada de vídeo",
"joinVoiceCall": "Unirse á chamada de voz",
"joinVideoCall": "Unirse á chamada de vídeo",
"live": "En directo"
"live": "En directo",
"playSoundOnNotification": "Reproducir son coas notificacións",
"addTag": "Engadir etiqueta",
"removeTag": "Retirar etiqueta",
"tagName": "Nome da etiqueta",
"createNewTag": "Crear nova etiqueta"
}

View file

@ -230,7 +230,7 @@
"type": "String",
"placeholders": {}
},
"allSpaces": "Visas vietas",
"allSpaces": "Visas kopienas",
"supposedMxid": "Tam būtu jābūt {mxid}",
"@supposedMxid": {
"type": "String",
@ -385,7 +385,7 @@
}
},
"tryAgain": "Jāmēģina vēlreiz",
"areGuestsAllowedToJoin": "Vai vieslietotājiem ir ļauts pievienoties",
"areGuestsAllowedToJoin": "Vai vieslietotāji drīkst pievienoties?",
"@areGuestsAllowedToJoin": {
"type": "String",
"placeholders": {}
@ -458,7 +458,7 @@
"placeholders": {}
},
"link": "Saite",
"newSpaceDescription": "Vietas ļauj apvienot tērzēšanas un būvēt privātas vai publiskas kopienas.",
"newSpaceDescription": "Kopienas ļauj apvienot tērzēšanas un būvēt privātas vai publiskas cilvēku grupas, kurus vieno kaut kas kopīgs, piemēram, zinātne, matemātika, valoda, reliģija, ķīmija, medicīna, kosmoss, datori, ceļošana, grāmatu lasīšana, kriptovalūta, kiberdrošība, aparātprogrammatūra.",
"chatDescription": "Tērzēšanas apraksts",
"next": "Nākamais",
"@next": {
@ -547,7 +547,7 @@
"type": "String",
"placeholders": {}
},
"spaceIsPublic": "Vieta ir publiska",
"spaceIsPublic": "Kopiena ir publiska",
"@spaceIsPublic": {
"type": "String",
"placeholders": {}
@ -790,7 +790,7 @@
}
}
},
"spaceName": "Vietas nosaukums",
"spaceName": "Kopienas nosaukums",
"@spaceName": {
"type": "String",
"placeholders": {}
@ -873,7 +873,7 @@
"type": "String",
"placeholders": {}
},
"logInTo": "PIeteikties {homeserver}",
"logInTo": "Pieteikties {homeserver}",
"@logInTo": {
"type": "String",
"placeholders": {
@ -1195,7 +1195,7 @@
"type": "String",
"placeholders": {}
},
"emoteExists": "Emocija jau pastāv.",
"emoteExists": "Emocija jau pastāv!",
"@emoteExists": {
"type": "String",
"placeholders": {}
@ -1351,7 +1351,7 @@
}
}
},
"addToSpace": "Pievienot vietai",
"addToSpace": "Pievienot kopienai",
"unbanFromChat": "Atcelt liegumu tērzēšanā",
"@unbanFromChat": {
"type": "String",
@ -1646,7 +1646,7 @@
"type": "String",
"placeholders": {}
},
"createNewSpace": "Jauna vieta",
"createNewSpace": "Jauna kopiena",
"@createNewSpace": {
"type": "String",
"placeholders": {}
@ -1725,7 +1725,7 @@
},
"newGroup": "Jauna kopa",
"bundleName": "Komplekta nosaukums",
"removeFromSpace": "Noņemt no vietas",
"removeFromSpace": "Noņemt no kopienas",
"dateAndTimeOfDay": "{date}, {timeOfDay}",
"@dateAndTimeOfDay": {
"type": "String",
@ -2060,7 +2060,7 @@
"type": "String",
"placeholders": {}
},
"newSpace": "Jauna vieta",
"newSpace": "Jauna kopiena",
"changePassword": "Nomainīt paroli",
"@changePassword": {
"type": "String",
@ -2144,7 +2144,7 @@
"type": "String",
"placeholders": {}
},
"pin": "PIN",
"pin": "Piespraust",
"@pin": {
"type": "String",
"placeholders": {}
@ -2203,8 +2203,8 @@
"transparent": "Caurspīdīgs",
"searchForUsers": "Meklēt @lietotājus...",
"pleaseEnterYourCurrentPassword": "Lūgums ievadīt savu pašreizējo paroli",
"publicSpaces": "Publiskas vietas",
"joinSpace": "Pievienoties vietai",
"publicSpaces": "Publiskas kopienas",
"joinSpace": "Pievienoties kopienai",
"createGroupAndInviteUsers": "Izveidot kopu un uzaicināt lietotājus",
"groupCanBeFoundViaSearch": "Kopu var atrast meklēšanā",
"commandHint_sendraw": "Nosūtīt neapstrādātu JSON",
@ -2319,7 +2319,7 @@
}
}
},
"addChatOrSubSpace": "Pievienot tērzēšanu vai apakšvietu",
"addChatOrSubSpace": "Pievienot tērzēšanu vai apakškopienu",
"formattedMessagesDescription": "Attēlot bagātinātu ziņu saturu, piemēram, ar Markdown iezīmētu treknrakstu.",
"sessionLostBody": "Sesija ir zaudēta. Lūgums ziņot par šo kļūdu izstrādātājiem {url}. Kļūdas ziņojums ir: {error}",
"@sessionLostBody": {
@ -2410,7 +2410,7 @@
"changeTheCanonicalRoomAlias": "Mainīt tērzēšanas galveno publisko adresi",
"sendRoomNotifications": "Sūtīt @istaba paziņojumus",
"changeTheDescriptionOfTheGroup": "Mainīt tērzēšanas aprakstu",
"alwaysUse24HourFormat": "",
"alwaysUse24HourFormat": "true",
"@alwaysUse24HourFormat": {
"description": "Set to true to always display time of day in 24 hour format."
},
@ -2423,7 +2423,7 @@
}
}
},
"goToSpace": "Doties uz vietu: {space}",
"goToSpace": "Doties uz kopienu: {space}",
"@goToSpace": {
"type": "String",
"space": {}
@ -2446,8 +2446,8 @@
"changelog": "Izmaiņu žurnāls",
"noMoreChatsFound": "Vairs netika atrasta neviena tērzēšana...",
"unread": "Nelasītas",
"space": "Vieta",
"spaces": "Vietas",
"space": "Kopiena",
"spaces": "Kopienas",
"markAsUnread": "Atzīmēt kā nelasītu",
"sendingAttachment": "Nosūta pielikumu...",
"generatingVideoThumbnail": "Izveido video sīktēlu...",
@ -2562,7 +2562,7 @@
"more": "Vairāk",
"roomNotificationSettings": "Istabu paziņojumu iestatījumi",
"notificationRuleEncrypted": "Šifrēts",
"notificationRuleJitsi": "Jitsi",
"notificationRuleJitsi": "Jitsi videozvani",
"notificationRuleIsUserMention": "Lietotāja pieminēšana",
"notificationRuleIsRoomMentionDescription": "Paziņo lietotājam, kad tiek pieminēta istaba.",
"notificationRuleMessageDescription": "Paziņo lietotājam par vispārējām ziņām.",
@ -2645,7 +2645,7 @@
"longPressToRecordVoiceMessage": "Ilga piespiešana, lai ierakstītu balss ziņu.",
"pause": "Apturēt",
"resume": "Atsākt",
"removeFromSpaceDescription": "Tērzēšana tiks noņemta no vietas, bet tā joprojām būs redzama tērzēšanu sarakstā.",
"removeFromSpaceDescription": "Tērzēšana tiks noņemta no kopienas, bet tā joprojām būs redzama tērzēšanu sarakstā.",
"countChats": "{chats} tērzēšanas",
"@countChats": {
"type": "String",
@ -2759,4 +2759,4 @@
"signUpGreeting": "FluffyChat ir decentralizēta. Jāatlasa serveris, kurā ir vēlēšanās izveidot savu kontu, un aiziet!",
"signInGreeting": "Jau ir Matrix konts? Laipni lūdzam atpakaļ! Jāatlasa savs mājasserveris un jāpiesakās.",
"appIntro": "Ar FluffyChat vari tērzēt ar saviem draugiem. Tā ir droša un decentralizēta [matrix] ziņapmaiņas lietotne. Vairāk var uzzināt https://matrix.org, ja ir vēlēšanās, vai vienkārši jāpiesakās."
}
}

View file

@ -2815,5 +2815,10 @@
"startVideoCall": "Start videosamtale",
"joinVoiceCall": "Bli med i lydsamtale",
"joinVideoCall": "Bli med i videosamtale",
"live": "Direkte"
"live": "Direkte",
"playSoundOnNotification": "Spill av lyd ved varsling",
"addTag": "Legg til emneknagg",
"removeTag": "Fjern emneknagg",
"tagName": "Navn på emneknagg",
"createNewTag": "Opprett ny emneknagg"
}

View file

@ -1953,7 +1953,7 @@
"unsupportedAndroidVersion": "Niet-ondersteunde Android-versie",
"unsupportedAndroidVersionLong": "Voor deze functie is een nieuwe Android-versie verplicht. Controleer je updates of Lineage OS-ondersteuning.",
"videoCallsBetaWarning": "Houd er rekening mee dat videogesprekken momenteel in bèta zijn. Ze werken misschien niet zoals je verwacht of werken niet op alle platformen.",
"voiceCall": "Spraakoproep",
"voiceCall": "Spraakgesprek",
"confirmEventUnpin": "Weet je zeker dat je de gebeurtenis definitief wilt losmaken?",
"experimentalVideoCalls": "Videogesprekken (experimenteel)",
"youAcceptedTheInvitation": "👍 Je hebt de uitnodiging geaccepteerd",
@ -2128,7 +2128,7 @@
"replace": "Vervang",
"report": "Rapporteer",
"reportErrorDescription": "😭 Oh nee. Er is iets misgegaan. Probeer het later nog eens. Als je wilt, kun je de bug rapporteren aan de ontwikkelaars.",
"sendTypingNotifications": "Typemeldingen verzenden",
"sendTypingNotifications": "Toon 'aan het typen'-meldingen",
"chatPermissions": "Chatrechten",
"chatDescription": "Onderwerp",
"chatDescriptionHasBeenChanged": "Onderwerp gewijzigd",
@ -2799,5 +2799,18 @@
"supportFluffyChat": "FluffyChat steunen",
"support": "Steunen",
"setLowPriority": "Lage prioriteit instellen",
"unsetLowPriority": "Lage prioriteit uitschakelen"
}
"unsetLowPriority": "Lage prioriteit uitschakelen",
"removeCallFromChat": "Verwijder oproep van chat",
"removeCallFromChatDescription": "Wil je de oproep voor iedereen in de chat verwijderen?",
"removeCallForEveryone": "Verwijder oproep voor iedereen",
"live": "Live",
"startVoiceCall": "Start audio-gesprek",
"startVideoCall": "Start video-gesprek",
"joinVoiceCall": "Audio-gesprek opnemen",
"joinVideoCall": "Deelnemen aan video-gesprek",
"playSoundOnNotification": "Meldingsgeluid afspelen",
"addTag": "Tag toevoegen",
"removeTag": "Tag verwijderen",
"tagName": "Tagnaam",
"createNewTag": "Nieuwe tag maken"
}

View file

@ -83,7 +83,7 @@
"type": "String",
"placeholders": {}
},
"areGuestsAllowedToJoin": "Разрешено ли гостям присоединяться",
"areGuestsAllowedToJoin": "Разрешено ли гостям присоединяться?",
"@areGuestsAllowedToJoin": {
"type": "String",
"placeholders": {}
@ -2105,8 +2105,8 @@
"notAnImage": "Это не картинка.",
"importNow": "Импортировать сейчас",
"importEmojis": "Импортировать эмодзи",
"importFromZipFile": "Импортировать из ZIP-файла",
"exportEmotePack": "Экспортировать набор эмодзи как ZIP",
"importFromZipFile": "Импортировать из zip-файла",
"exportEmotePack": "Экспортировать набор эмодзи как zip",
"replace": "Заменить",
"googlyEyesContent": "{senderName} выпучил глаза",
"@googlyEyesContent": {
@ -2471,7 +2471,7 @@
"boldText": "Жирный шрифт",
"strikeThrough": "Перечёркнутый",
"pleaseFillOut": "Пожалуйста, заполните",
"invalidUrl": "Не верный URL",
"invalidUrl": "Неверный url-адрес",
"addLink": "Добавить ссылку",
"italicText": "Italic",
"unableToJoinChat": "Невозможно присоединиться к чату. Возможно, другая сторона уже закончила разговор.",
@ -2482,7 +2482,7 @@
"manageAccount": "Управление аккаунтом",
"contactServerAdmin": "Админ сервера",
"contactServerSecurity": "Безопасность контактов сервера",
"supportPage": "Поддержка",
"supportPage": "Страница поддержки",
"name": "Имя",
"version": "Версия",
"website": "Сайт",
@ -2699,5 +2699,124 @@
"type": "int"
}
}
}
},
"versionWithNumber": "Версия: {version}",
"@versionWithNumber": {
"type": "String",
"placeholders": {
"version": {
"type": "String"
}
}
},
"logs": "Архивные записи",
"advancedConfigs": "Расширенные Настройки",
"advancedConfigurations": "Расширенные конфигурации",
"signUpGreeting": "FluffyChat децентрализорован! Выберите сервер, где вы хотите сделать свой аккаунт и заходите!",
"signInGreeting": "У вас есть уже аккаунт в Matrix? Добро пожаловать! Выберите свой сервер и войдите.",
"appIntro": "С FluffyChat'ом вы можете говорить со своими друзьями. Это защищённый децентрализорованный [matrix] мессенджер! Узнайте больше на https://matrix.org, если вам нравится или просто зарегистрироваться.",
"join": "Присоединиться",
"countFiles": "{count} файлов",
"@countFiles": {
"placeholders": {
"count": {
"type": "int"
}
}
},
"unreadChatsInApp": "{appname}: {unread} непрочитанных чатов",
"@unreadChatsInApp": {
"type": "String",
"placeholders": {
"appname": {
"type": "String"
},
"unread": {
"type": "String"
}
}
},
"thereAreCountUsersBlocked": "Сейчас {count} пользователей заблокировано.",
"@thereAreCountUsersBlocked": {
"type": "String",
"count": {}
},
"serverLimitReached": "Достигнут серверный лимит! Подождите {seconds} секунд...",
"@serverLimitReached": {
"type": "integer",
"placeholders": {
"seconds": {
"type": "int"
}
}
},
"countVotes": "{count, plural, =1{Один голос} other{{count} голоса(-ов)}}",
"@countVotes": {
"type": "int",
"placeholders": {
"count": {
"type": "int"
}
}
},
"countReplies": "{count, plural, =1{Один ответ} other{{count} ответа(-ов)}}",
"@countReplies": {
"type": "int",
"placeholders": {
"count": {
"type": "int"
}
}
},
"chatSearchedUntil": "Чат индексируется до {time}",
"@chatSearchedUntil": {
"type": "String",
"placeholders": {
"time": {
"type": "String"
}
}
},
"federationBaseUrl": "Основной URL федерации",
"clientWellKnownInformation": "Client-Well-Known Информация:",
"baseUrl": "Базовый URL",
"identityServer": "Сервер профилей:",
"signIn": "Войти",
"searchOrEnterHomeserverAddress": "Поищите или введите адрес домашнего сервера",
"matrixId": "Matrix ID",
"setPowerLevel": "Установить уровень возможностей",
"makeModerator": "Назначить модератором",
"makeAdmin": "Назначить администратором",
"removeModeratorRights": "Удалить права модератора",
"removeAdminRights": "Удалить права администратора",
"powerLevel": "Уровень энергии",
"setPowerLevelDescription": "Уровни прав определяют, что участнику разрешено делать в этой комнате, и обычно варьируются от 0 до 100.",
"owner": "Владелец",
"mute": "Мут",
"@mute": {
"description": "This should be a very short string because there is not much space in the button!"
},
"createNewChat": "Создать новый чат",
"reset": "Сброс",
"supportFluffyChat": "Поддержите FluffyChat",
"support": "Поддержка",
"fluffyChatSupportBannerMessage": "FluffyChat нуждается в ВАШЕЙ помощи!\n❤\nFluffyChat всегда будет бесплатным, но разработка и хостинг всё равно требуют затрат.\nБудущее проекта зависит от поддержки таких людей, как вы.",
"skipSupportingFluffyChat": "Пропустить помощь FluffyChat",
"iAlreadySupportFluffyChat": "Я уже поддерживаю FluffyChat",
"iDoNotWantToSupport": "Я не хочу поддерживать",
"setLowPriority": "Установить низкий приоритет",
"unsetLowPriority": "Неопределенный приоритет",
"removeCallFromChat": "Удалить сообщение из чата",
"removeCallFromChatDescription": "Вы хотите удалить это сообщение из чата для всех участников?",
"removeCallForEveryone": "Отменить вызов для всех",
"startVoiceCall": "Начать голосовой вызов",
"startVideoCall": "Начать видеозвонок",
"joinVoiceCall": "Присоединиться к голосовому звонку",
"joinVideoCall": "Присоединиться к видеозвонку",
"live": "Прямой эфир",
"playSoundOnNotification": "Воспроизвести звук при получении уведомления",
"addTag": "Добавить тег",
"removeTag": "Удалить тег",
"tagName": "Название тега",
"createNewTag": "Создать новый тег"
}

View file

@ -72,7 +72,7 @@
"type": "String",
"placeholders": {}
},
"areGuestsAllowedToJoin": "Får gästanvändare gå med",
"areGuestsAllowedToJoin": "Får gästanvändare gå med?",
"@areGuestsAllowedToJoin": {
"type": "String",
"placeholders": {}
@ -2040,7 +2040,7 @@
},
"screenSharingTitle": "skärmdelning",
"noKeyForThisMessage": "Detta kan hända om meddelandet skickades innan du loggade in på ditt konto i den här enheten.\n\nDet kan också vara så att avsändaren har blockerat din enhet eller att något gick fel med internetanslutningen.\n\nKan du läsa meddelandet i en annan session? I sådana fall kan du överföra meddelandet från den sessionen! Gå till Inställningar > Enhet och säkerställ att dina enheter har verifierat varandra. När du öppnar rummet nästa gång och båda sessionerna är i förgrunden, så kommer nycklarna att överföras automatiskt.\n\nVill du inte förlora nycklarna vid utloggning eller när du byter enhet? Säkerställ att du har aktiverat säkerhetskopiering för chatten i inställningarna.",
"fileIsTooBigForServer": "Gick inte att skicka! Servern stödjer endast bilagor upp till{max}.",
"fileIsTooBigForServer": "Gick inte att skicka! Servern stödjer endast bilagor upp till {max}.",
"deviceKeys": "Enhetsnycklar:",
"commandHint_googly": "Skicka några googly ögon",
"commandHint_cuddle": "Skicka en omfamning",
@ -2333,7 +2333,7 @@
"stickers": "Klistermärken",
"discover": "Upptäck",
"ignoreUser": "Ignorera användare",
"aboutHomeserver": "Om{homeserver}",
"aboutHomeserver": "Om {homeserver}",
"@aboutHomeserver": {
"type": "String",
"placeholders": {
@ -2395,7 +2395,7 @@
}
}
},
"invitedBy": "📩Inbjuden av{user}",
"invitedBy": "📩Inbjuden av {user}",
"@invitedBy": {
"placeholders": {
"user": {
@ -2644,4 +2644,4 @@
"logs": "Loggar",
"signIn": "Logga in",
"createNewAccount": "Skapa nytt konto"
}
}

View file

@ -2775,5 +2775,6 @@
"signUpGreeting": "FluffyChat децентралізований! Виберіть сервер, на якому ви хочете створити свій обліковий запис, і почнімо!",
"signInGreeting": "Ви вже маєте обліковий запис у Matrix? Ласкаво просимо! Виберіть свій домашній сервер і ввійдіть.",
"appIntro": "За допомогою FluffyChat ви можете спілкуватися зі своїми друзями. Це безпечний децентралізований месенджер [matrix]! Дізнайтеся більше на сайті https://matrix.org або просто зареєструйтеся.",
"theProcessWasCanceled": "Процес скасовано."
}
"theProcessWasCanceled": "Процес скасовано.",
"join": "Приєднатись"
}

View file

@ -2808,5 +2808,10 @@
"startVideoCall": "开始视频通话",
"joinVoiceCall": "加入语音通话",
"joinVideoCall": "加入视频通话",
"live": "实时"
"live": "实时",
"playSoundOnNotification": "播放通知声音",
"addTag": "添加标签",
"removeTag": "删除标签",
"tagName": "标签名",
"createNewTag": "创建新标签"
}

View file

@ -48,6 +48,7 @@ class ArchiveController extends State<Archive> {
OkCancelResult.ok) {
return;
}
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
futureWithProgress: (onProgress) async {

View file

@ -382,6 +382,7 @@ class BootstrapDialogState extends State<BootstrapDialog> {
).wrongRecoveryKey,
);
} catch (e, s) {
if (!context.mounted) return;
ErrorReporter(
context,
'Unable to open SSSS with recovery key',
@ -425,6 +426,7 @@ class BootstrapDialogState extends State<BootstrapDialog> {
cancelLabel: L10n.of(context).cancel,
);
if (consent != OkCancelResult.ok) return;
if (!context.mounted) return;
final req = await showFutureLoadingDialog(
context: context,
delay: false,
@ -435,11 +437,12 @@ class BootstrapDialogState extends State<BootstrapDialog> {
},
);
if (req.error != null) return;
if (!context.mounted) return;
final success = await KeyVerificationDialog(
request: req.result!,
).show(context);
if (success != true) return;
if (!mounted) return;
if (!context.mounted) return;
final result = await showFutureLoadingDialog(
context: context,

View file

@ -212,6 +212,7 @@ class ChatController extends State<ChatPageWithRoom>
context: context,
future: room.leave,
);
if (!mounted) return;
if (success.error != null) return;
context.go('/rooms');
}
@ -463,21 +464,25 @@ class ChatController extends State<ChatPageWithRoom>
scrollUpBannerEventId = eventId;
});
bool firstUpdateReceived = false;
String? animateInEventId;
void _insert(int index) {
if (index > 0) return;
animateInEventId = timeline?.events.firstOrNull?.eventId;
}
void updateView() {
if (!mounted) return;
setReadMarker();
setState(() {
firstUpdateReceived = true;
});
setState(() {});
}
Future<void>? loadTimelineFuture;
Future<void> _getTimeline({String? eventContextId}) async {
await Matrix.of(context).client.roomsLoading;
await Matrix.of(context).client.accountDataLoading;
final matrix = Matrix.of(context);
await matrix.client.roomsLoading;
await matrix.client.accountDataLoading;
if (eventContextId != null &&
(!eventContextId.isValidMatrixId || eventContextId.sigil != '\$')) {
eventContextId = null;
@ -486,6 +491,7 @@ class ChatController extends State<ChatPageWithRoom>
timeline?.cancelSubscriptions();
timeline = await room.getTimeline(
onUpdate: updateView,
onInsert: _insert,
eventContextId: eventContextId,
);
} catch (e, s) {
@ -633,6 +639,7 @@ class ChatController extends State<ChatPageWithRoom>
Future<void> sendFileAction({FileType type = FileType.any}) async {
final files = await selectFiles(context, allowMultiple: true, type: type);
if (files.isEmpty) return;
if (!mounted) return;
await showAdaptiveDialog(
context: context,
builder: (c) => SendFileDialog(
@ -669,6 +676,7 @@ class ChatController extends State<ChatPageWithRoom>
FocusScope.of(context).requestFocus(FocusNode());
final file = await ImagePicker().pickImage(source: ImageSource.camera);
if (file == null) return;
if (!mounted) return;
await showAdaptiveDialog(
context: context,
@ -690,6 +698,7 @@ class ChatController extends State<ChatPageWithRoom>
maxDuration: const Duration(minutes: 1),
);
if (file == null) return;
if (!mounted) return;
await showAdaptiveDialog(
context: context,
@ -732,26 +741,27 @@ class ChatController extends State<ChatPageWithRoom>
mimeType: mimeType,
);
room
.sendFileEvent(
file,
inReplyTo: replyEvent,
threadRootEventId: activeThreadId,
extraContent: {
'info': {...file.info, 'duration': duration},
'org.matrix.msc3245.voice': {},
'org.matrix.msc1767.audio': {
'duration': duration,
'waveform': waveform,
},
try {
await room.sendFileEvent(
file,
inReplyTo: replyEvent,
threadRootEventId: activeThreadId,
extraContent: {
'info': {...file.info, 'duration': duration},
'org.matrix.msc3245.voice': {},
'org.matrix.msc1767.audio': {
'duration': duration,
'waveform': waveform,
},
)
.catchError((e) {
scaffoldMessenger.showSnackBar(
SnackBar(content: Text((e as Object).toLocalizedString(context))),
);
return null;
});
},
);
} catch (e) {
if (!mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(e.toLocalizedString(context))),
);
return;
}
setState(() {
replyEvent = null;
});
@ -813,29 +823,30 @@ class ChatController extends State<ChatPageWithRoom>
Future<void> reportEventAction() async {
final event = selectedEvents.single;
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final score = await showModalActionPopup<int>(
context: context,
title: L10n.of(context).reportMessage,
message: L10n.of(context).howOffensiveIsThisContent,
cancelLabel: L10n.of(context).cancel,
title: l10n.reportMessage,
message: l10n.howOffensiveIsThisContent,
cancelLabel: l10n.cancel,
actions: [
AdaptiveModalAction(
value: -100,
label: L10n.of(context).extremeOffensive,
),
AdaptiveModalAction(value: -50, label: L10n.of(context).offensive),
AdaptiveModalAction(value: 0, label: L10n.of(context).inoffensive),
AdaptiveModalAction(value: -100, label: l10n.extremeOffensive),
AdaptiveModalAction(value: -50, label: l10n.offensive),
AdaptiveModalAction(value: 0, label: l10n.inoffensive),
],
);
if (score == null) return;
if (!mounted) return;
final reason = await showTextInputDialog(
context: context,
title: L10n.of(context).whyDoYouWantToReportThis,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
hintText: L10n.of(context).reason,
title: l10n.whyDoYouWantToReportThis,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: l10n.reason,
);
if (reason == null || reason.isEmpty) return;
if (!mounted) return;
final result = await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(context).client.reportEvent(
@ -846,12 +857,13 @@ class ChatController extends State<ChatPageWithRoom>
),
);
if (result.error != null) return;
if (!mounted) return;
setState(() {
showEmojiPicker = false;
selectedEvents.clear();
});
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).contentHasBeenReported)),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.contentHasBeenReported)),
);
}
@ -867,6 +879,7 @@ class ChatController extends State<ChatPageWithRoom>
}
setState(selectedEvents.clear);
} catch (e, s) {
if (!mounted) return;
ErrorReporter(
context,
'Error while delete error events action',
@ -891,6 +904,7 @@ class ChatController extends State<ChatPageWithRoom>
: null;
if (reasonInput == null) return;
final reason = reasonInput.isEmpty ? null : reasonInput;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
futureWithProgress: (onProgress) async {
@ -1147,7 +1161,7 @@ class ChatController extends State<ChatPageWithRoom>
true,
false,
);
users.sort((a, b) => a.powerLevel.compareTo(b.powerLevel));
users.sort((a, b) => a.powerLevel.level.compareTo(b.powerLevel.level));
final via = users
.map((user) => user.id.domain)
.whereType<String>()
@ -1248,6 +1262,7 @@ class ChatController extends State<ChatPageWithRoom>
okLabel: L10n.of(context).unpin,
cancelLabel: L10n.of(context).cancel,
);
if (!mounted) return;
if (response == OkCancelResult.ok) {
final events = room.pinnedEventIds
..removeWhere((oldEvent) => oldEvent == eventId);
@ -1337,17 +1352,18 @@ class ChatController extends State<ChatPageWithRoom>
Future<void> onPhoneButtonTap() async {
// VoIP required Android SDK 21
if (PlatformInfos.isAndroid) {
DeviceInfoPlugin().androidInfo.then((value) {
if (value.version.sdkInt < 21) {
Navigator.pop(context);
showOkAlertDialog(
context: context,
title: L10n.of(context).unsupportedAndroidVersion,
message: L10n.of(context).unsupportedAndroidVersionLong,
okLabel: L10n.of(context).close,
);
}
});
final androidInfo = await DeviceInfoPlugin().androidInfo;
if (!mounted) return;
if (androidInfo.version.sdkInt < 21) {
Navigator.pop(context);
await showOkAlertDialog(
context: context,
title: L10n.of(context).unsupportedAndroidVersion,
message: L10n.of(context).unsupportedAndroidVersionLong,
okLabel: L10n.of(context).close,
);
return;
}
}
final callType = await showModalActionPopup<CallType>(
context: context,
@ -1368,11 +1384,13 @@ class ChatController extends State<ChatPageWithRoom>
],
);
if (callType == null) return;
if (!mounted) return;
final voipPlugin = Matrix.of(context).voipPlugin;
try {
await voipPlugin!.voip.inviteToCall(room, callType);
} catch (e) {
if (!mounted) return;
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text(e.toLocalizedString(context))));

View file

@ -77,7 +77,7 @@ class ChatEventList extends StatelessWidget {
return Column(
mainAxisSize: .min,
children: [
SeenByRow(event: events.first),
if (events.isNotEmpty) SeenByRow(event: events.first),
TypingIndicators(controller),
],
);
@ -117,9 +117,7 @@ class ChatEventList extends StatelessWidget {
// The message at this index:
final event = events[i];
final animateIn =
event.eventId == timeline.events.first.eventId &&
controller.firstUpdateReceived;
final animateIn = event.eventId == controller.animateInEventId;
final nextEvent = i + 1 < events.length ? events[i + 1] : null;
final previousEvent = i > 0 ? events[i - 1] : null;

View file

@ -189,6 +189,7 @@ class AudioPlayerState extends State<AudioPlayerWidget> {
});
} catch (e, s) {
Logs().v('Could not download audio file', e, s);
if (!mounted) rethrow;
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text(e.toLocalizedString(context))));
@ -208,6 +209,7 @@ class AudioPlayerState extends State<AudioPlayerWidget> {
),
);
}
if (!mounted) return;
audioPlayer.play().onError(
ErrorReporter(context, 'Unable to play audio message').onErrorCallback,

View file

@ -50,6 +50,7 @@ class _CuteContentState extends State<CuteContent> {
Future<void> addOverlay() async {
_isOverlayShown = true;
await Future.delayed(const Duration(milliseconds: 50));
if (!mounted) return;
OverlayEntry? overlay;
overlay = OverlayEntry(

View file

@ -1,4 +1,5 @@
import 'package:collection/collection.dart';
import 'package:fluffychat/config/setting_keys.dart';
import 'package:fluffychat/utils/code_highlight_theme.dart';
import 'package:fluffychat/utils/event_checkbox_extension.dart';
import 'package:fluffychat/widgets/avatar.dart';
@ -509,11 +510,15 @@ class HtmlMessage extends StatelessWidget {
@override
Widget build(BuildContext context) {
final element = parser.parse(html).body ?? dom.Element.html('');
final configuredMaxLines = AppSettings.messagePreviewMaxLines.value;
final maxLines = !limitHeight || configuredMaxLines <= 0
? null
: configuredMaxLines;
return Text.rich(
_renderHtml(element, context),
style: TextStyle(fontSize: fontSize, color: textColor),
maxLines: limitHeight ? 64 : null,
overflow: TextOverflow.fade,
maxLines: maxLines,
overflow: maxLines == null ? TextOverflow.visible : TextOverflow.fade,
selectionColor: textColor.withAlpha(128),
);
}

View file

@ -205,6 +205,8 @@ class Message extends StatelessWidget {
final enterThread = this.enterThread;
final sender = event.senderFromMemoryOrFallback;
final fileSendingStatus = event.fileSendingStatus;
return _AnimateIn(
animateIn: animateIn,
child: Center(
@ -318,9 +320,33 @@ class Message extends StatelessWidget {
height: 16,
child: event.status == EventStatus.error
? const Icon(Icons.error, color: Colors.red)
: event.fileSendingStatus != null
? const CircularProgressIndicator.adaptive(
strokeWidth: 1,
: fileSendingStatus != null
? Stack(
children: [
Center(
child: switch (fileSendingStatus) {
FileSendingStatus
.generatingThumbnail =>
Icon(
Icons.compress_outlined,
size: 14,
),
FileSendingStatus.encrypting =>
Icon(
Icons.lock_outlined,
size: 14,
),
FileSendingStatus.uploading =>
Icon(
Icons.upload_outlined,
size: 14,
),
},
),
const CircularProgressIndicator.adaptive(
strokeWidth: 1,
),
],
)
: null,
),
@ -361,17 +387,20 @@ class Message extends StatelessWidget {
? const SizedBox(height: 12)
: Row(
children: [
if (sender.powerLevel >= 50)
if (sender.powerLevel.role !=
PowerLevelRole.user)
Padding(
padding: const EdgeInsets.only(
right: 2.0,
),
child: Icon(
sender.powerLevel >= 100
sender.powerLevel.role ==
PowerLevelRole
.moderator
? Icons
.admin_panel_settings
.add_moderator_outlined
: Icons
.add_moderator_outlined,
.admin_panel_settings,
size: 14,
color: theme
.colorScheme
@ -432,147 +461,161 @@ class Message extends StatelessWidget {
HapticFeedback.heavyImpact();
onSelect(event);
},
child: Container(
decoration: BoxDecoration(
color: noBubble
? Colors.transparent
: color,
borderRadius: borderRadius,
),
clipBehavior: Clip.antiAlias,
child: BubbleBackground(
colors: colors,
ignore:
noBubble ||
!ownMessage ||
MediaQuery.highContrastOf(context),
scrollController: scrollController,
child: Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(
AppConfig.borderRadius,
child: AnimatedOpacity(
duration: FluffyThemes.animationDuration,
curve: FluffyThemes.animationCurve,
opacity:
event.status.isSending ||
event.type == EventTypes.Encrypted
? 0.5
: 1,
child: Container(
decoration: BoxDecoration(
color: noBubble
? Colors.transparent
: color,
borderRadius: borderRadius,
),
clipBehavior: Clip.antiAlias,
child: BubbleBackground(
colors: colors,
ignore:
noBubble ||
!ownMessage ||
MediaQuery.highContrastOf(context),
scrollController: scrollController,
child: Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(
AppConfig.borderRadius,
),
),
),
constraints: const BoxConstraints(
maxWidth:
FluffyThemes.columnWidth * 1.5,
),
child: Column(
mainAxisSize: .min,
crossAxisAlignment:
CrossAxisAlignment.start,
children: <Widget>[
if (event.inReplyToEventId(
includingFallback: false,
) !=
null)
FutureBuilder<Event?>(
future: event.getReplyEvent(
timeline,
),
builder: (BuildContext context, snapshot) {
final replyEvent =
snapshot.hasData
? snapshot.data!
: Event(
eventId:
event
.inReplyToEventId() ??
'\$fake_event_id',
content: {
'msgtype': 'm.text',
'body': '...',
},
senderId:
event.senderId,
type:
'm.room.message',
room: event.room,
status:
EventStatus.sent,
originServerTs:
DateTime.now(),
);
return Padding(
padding:
const EdgeInsets.only(
left: 16,
right: 16,
top: 8,
),
child: Material(
color: Colors.transparent,
borderRadius: ReplyContent
.borderRadius,
child: InkWell(
constraints: const BoxConstraints(
maxWidth:
FluffyThemes.columnWidth * 1.5,
),
child: Column(
mainAxisSize: .min,
crossAxisAlignment:
CrossAxisAlignment.start,
children: <Widget>[
if (event.inReplyToEventId(
includingFallback: false,
) !=
null)
FutureBuilder<Event?>(
future: event.getReplyEvent(
timeline,
),
builder: (BuildContext context, snapshot) {
final replyEvent =
snapshot.hasData
? snapshot.data!
: Event(
eventId:
event
.inReplyToEventId() ??
'\$fake_event_id',
content: {
'msgtype':
'm.text',
'body': '...',
},
senderId:
event.senderId,
type:
'm.room.message',
room: event.room,
status: EventStatus
.sent,
originServerTs:
DateTime.now(),
);
return Padding(
padding:
const EdgeInsets.only(
left: 16,
right: 16,
top: 8,
),
child: Material(
color:
Colors.transparent,
borderRadius:
ReplyContent
.borderRadius,
onTap: () =>
scrollToEventId(
replyEvent
.eventId,
child: InkWell(
borderRadius:
ReplyContent
.borderRadius,
onTap: () =>
scrollToEventId(
replyEvent
.eventId,
),
child: AbsorbPointer(
child: ReplyContent(
replyEvent,
ownMessage:
ownMessage,
timeline:
timeline,
),
child: AbsorbPointer(
child: ReplyContent(
replyEvent,
ownMessage:
ownMessage,
timeline: timeline,
),
),
),
),
);
},
),
MessageContent(
displayEvent,
textColor: textColor,
linkColor: linkColor,
onInfoTab: onInfoTab,
borderRadius: borderRadius,
timeline: timeline,
selected: selected,
bigEmojis: bigEmojis,
),
if (event.hasAggregatedEvents(
timeline,
RelationshipTypes.edit,
))
Padding(
padding: const EdgeInsets.only(
bottom: 8.0,
left: 16.0,
right: 16.0,
);
},
),
child: Row(
mainAxisSize:
MainAxisSize.min,
spacing: 4.0,
children: [
Icon(
Icons.edit_outlined,
color: textColor
.withAlpha(164),
size: 14,
),
Text(
displayEvent
.originServerTs
.localizedTimeShort(
context,
),
style: TextStyle(
MessageContent(
displayEvent,
textColor: textColor,
linkColor: linkColor,
onInfoTab: onInfoTab,
borderRadius: borderRadius,
timeline: timeline,
selected: selected,
bigEmojis: bigEmojis,
),
if (event.hasAggregatedEvents(
timeline,
RelationshipTypes.edit,
))
Padding(
padding:
const EdgeInsets.only(
bottom: 8.0,
left: 16.0,
right: 16.0,
),
child: Row(
mainAxisSize:
MainAxisSize.min,
spacing: 4.0,
children: [
Icon(
Icons.edit_outlined,
color: textColor
.withAlpha(164),
fontSize: 11,
size: 14,
),
),
],
Text(
displayEvent
.originServerTs
.localizedTimeShort(
context,
),
style: TextStyle(
color: textColor
.withAlpha(164),
fontSize: 11,
),
),
],
),
),
),
],
],
),
),
),
),
@ -959,15 +1002,10 @@ class __AnimateInState extends State<_AnimateIn> {
});
});
}
return AnimatedOpacity(
return AnimatedSize(
duration: FluffyThemes.animationDuration,
curve: FluffyThemes.animationCurve,
opacity: _animationFinished ? 1 : 0,
child: AnimatedSize(
duration: FluffyThemes.animationDuration,
curve: FluffyThemes.animationCurve,
child: _animationFinished ? widget.child : const SizedBox.shrink(),
),
child: _animationFinished ? widget.child : const SizedBox.shrink(),
);
}
}

View file

@ -61,9 +61,11 @@ class MessageContent extends StatelessWidget {
final client = Matrix.of(context).client;
final state = await client.getCryptoIdentityState();
if (!state.connected) {
if (!context.mounted) return;
final success = await context.push('/backup');
if (success != true) return;
}
if (!context.mounted) return;
event.requestKey();
final sender = event.senderFromMemoryOrFallback;
await showAdaptiveBottomSheet(

View file

@ -392,6 +392,10 @@ class InputBar extends StatelessWidget {
controller: controller,
focusNode: focusNode,
readOnly: readOnly,
onEditingComplete: () {
// To not lose focus on iOS:
// https://github.com/krille-chan/fluffychat/issues/2784
},
contextMenuBuilder: (c, e) => MarkdownContextBuilder(
editableTextState: e,
controller: controller,

View file

@ -15,6 +15,7 @@ class PinnedEvents extends StatelessWidget {
const PinnedEvents(this.controller, {super.key});
Future<void> _displayPinnedEventsDialog(BuildContext context) async {
final l10n = L10n.of(context);
final eventsResult = await showFutureLoadingDialog(
context: context,
future: () => Future.wait(
@ -25,13 +26,14 @@ class PinnedEvents extends StatelessWidget {
);
final events = eventsResult.result;
if (events == null) return;
if (!context.mounted) return;
final eventId = events.length == 1
? events.single?.eventId
: await showModalActionPopup<String>(
context: context,
title: L10n.of(context).pin,
cancelLabel: L10n.of(context).cancel,
title: l10n.pin,
cancelLabel: l10n.cancel,
actions: events
.map(
(event) => AdaptiveModalAction(
@ -39,7 +41,7 @@ class PinnedEvents extends StatelessWidget {
icon: const Icon(Icons.push_pin_outlined),
label:
event?.calcLocalizedBodyFallback(
MatrixLocals(L10n.of(context)),
MatrixLocals(l10n),
withSenderNamePrefix: true,
hideReply: true,
) ??

View file

@ -44,6 +44,7 @@ class RecordingViewModelState extends State<RecordingViewModel> {
room.client.getConfig(); // Preload server file configuration.
if (PlatformInfos.isAndroid) {
final info = await DeviceInfoPlugin().androidInfo;
if (!mounted) return;
if (info.version.sdkInt < 19) {
showOkAlertDialog(
context: context,
@ -76,6 +77,7 @@ class RecordingViewModelState extends State<RecordingViewModel> {
final result = await audioRecorder.hasPermission();
if (result != true) {
if (!mounted) return;
showOkAlertDialog(
context: context,
title: L10n.of(context).oopsSomethingWentWrong,
@ -97,10 +99,12 @@ class RecordingViewModelState extends State<RecordingViewModel> {
),
path: path ?? '',
);
if (!mounted) return;
setState(() => duration = Duration.zero);
_subscribe();
} catch (e, s) {
Logs().w('Unable to start voice message recording', e, s);
if (!mounted) return;
showOkAlertDialog(
context: context,
title: L10n.of(context).oopsSomethingWentWrong,

View file

@ -146,6 +146,7 @@ class SendFileDialogState extends State<SendFileDialog> {
scaffoldMessenger.clearSnackBars();
} catch (e) {
scaffoldMessenger.clearSnackBars();
if (!mounted || !widget.outerContext.mounted) rethrow;
final theme = Theme.of(context);
scaffoldMessenger.showSnackBar(
SnackBar(

View file

@ -81,6 +81,7 @@ class SendLocationDialogState extends State<SendLocationDialog> {
context: context,
future: () => widget.room.sendLocation(body, uri),
);
if (!mounted) return;
Navigator.of(context, rootNavigator: false).pop();
}

View file

@ -44,6 +44,7 @@ class _StartPollBottomSheetState extends State<StartPollBottomSheet> {
maxSelections: _allowMultipleAnswers ? _answers.length : 1,
txid: _txid,
);
if (!mounted) return;
Navigator.of(context).pop();
} catch (e, s) {
Logs().w('Unable to create poll', e, s);

View file

@ -160,6 +160,7 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
}
Future<void> updateRoomAction() async {
final l10n = L10n.of(context);
final roomVersion = room
.getState(EventTypes.RoomCreate)!
.content
@ -170,10 +171,11 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
);
final capabilities = capabilitiesResult.result;
if (capabilities == null) return;
if (!mounted) return;
final newVersion = await showModalActionPopup<String>(
context: context,
title: L10n.of(context).replaceRoomWithNewerVersion,
cancelLabel: L10n.of(context).cancel,
title: l10n.replaceRoomWithNewerVersion,
cancelLabel: l10n.cancel,
actions: capabilities.mRoomVersions!.available.entries
.where((r) => r.key != roomVersion)
.map(
@ -185,18 +187,20 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
)
.toList(),
);
if (newVersion == null ||
OkCancelResult.cancel ==
await showOkCancelAlertDialog(
context: context,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).cancel,
title: L10n.of(context).areYouSure,
message: L10n.of(context).roomUpgradeDescription,
isDestructive: true,
)) {
if (newVersion == null) return;
if (!mounted) return;
final confirmUpgrade = await showOkCancelAlertDialog(
context: context,
okLabel: l10n.yes,
cancelLabel: l10n.cancel,
title: l10n.areYouSure,
message: l10n.roomUpgradeDescription,
isDestructive: true,
);
if (confirmUpgrade == OkCancelResult.cancel) {
return;
}
if (!mounted) return;
final result = await showFutureLoadingDialog(
context: context,
futureWithProgress: (onProgress) async {
@ -243,6 +247,7 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
}
Future<void> addAlias() async {
final l10n = L10n.of(context);
final domain = room.client.userID?.domain;
if (domain == null) {
throw Exception('userID or domain is null! This should never happen.');
@ -250,11 +255,12 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
final input = await showTextInputDialog(
context: context,
title: L10n.of(context).editRoomAliases,
title: l10n.editRoomAliases,
prefixText: '#',
suffixText: domain,
hintText: L10n.of(context).alias,
hintText: l10n.alias,
);
if (!mounted) return;
final aliasLocalpart = input?.trim();
if (aliasLocalpart == null || aliasLocalpart.isEmpty) return;
final alias = '#$aliasLocalpart:$domain';
@ -264,17 +270,19 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
future: () => room.client.setRoomAlias(alias, room.id),
);
if (result.error != null) return;
if (!mounted) return;
setState(() {});
if (!room.canChangeStateEvent(EventTypes.RoomCanonicalAlias)) return;
final canonicalAliasConsent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).setAsCanonicalAlias,
title: l10n.setAsCanonicalAlias,
message: alias,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).no,
okLabel: l10n.yes,
cancelLabel: l10n.no,
);
if (!mounted) return;
final altAliases =
room

View file

@ -37,69 +37,78 @@ class ChatDetailsController extends State<ChatDetails> {
String? get roomId => widget.roomId;
Future<void> setDisplaynameAction() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final room = Matrix.of(context).client.getRoomById(roomId!)!;
final input = await showTextInputDialog(
context: context,
title: L10n.of(context).changeTheNameOfTheGroup,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
initialText: room.getLocalizedDisplayname(MatrixLocals(L10n.of(context))),
title: l10n.changeTheNameOfTheGroup,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
initialText: room.getLocalizedDisplayname(MatrixLocals(l10n)),
);
if (input == null) return;
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => room.setName(input),
);
if (!mounted) return;
if (success.error == null) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).displaynameHasBeenChanged)),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.displaynameHasBeenChanged)),
);
}
}
Future<void> setTopicAction() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final room = Matrix.of(context).client.getRoomById(roomId!)!;
final input = await showTextInputDialog(
context: context,
title: L10n.of(context).setChatDescription,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
hintText: L10n.of(context).noChatDescriptionYet,
title: l10n.setChatDescription,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: l10n.noChatDescriptionYet,
initialText: room.topic,
minLines: 4,
maxLines: 8,
);
if (input == null) return;
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => room.setDescription(input),
);
if (!mounted) return;
if (success.error == null) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).chatDescriptionHasBeenChanged)),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.chatDescriptionHasBeenChanged)),
);
}
}
Future<void> setAvatarAction() async {
final l10n = L10n.of(context);
final room = Matrix.of(context).client.getRoomById(roomId!);
final actions = [
if (PlatformInfos.isMobile)
AdaptiveModalAction(
value: AvatarAction.camera,
label: L10n.of(context).openCamera,
label: l10n.openCamera,
isDefaultAction: true,
icon: const Icon(Icons.camera_alt_outlined),
),
AdaptiveModalAction(
value: AvatarAction.file,
label: L10n.of(context).openGallery,
label: l10n.openGallery,
icon: const Icon(Icons.photo_outlined),
),
if (room?.avatar != null)
AdaptiveModalAction(
value: AvatarAction.remove,
label: L10n.of(context).delete,
label: l10n.delete,
isDestructive: true,
icon: const Icon(Icons.delete_outlined),
),
@ -108,11 +117,12 @@ class ChatDetailsController extends State<ChatDetails> {
? actions.single.value
: await showModalActionPopup<AvatarAction>(
context: context,
title: L10n.of(context).editRoomAvatar,
cancelLabel: L10n.of(context).cancel,
title: l10n.editRoomAvatar,
cancelLabel: l10n.cancel,
actions: actions,
);
if (action == null) return;
if (!mounted) return;
if (action == AvatarAction.remove) {
await showFutureLoadingDialog(
context: context,
@ -131,6 +141,7 @@ class ChatDetailsController extends State<ChatDetails> {
if (result == null) return;
file = MatrixFile(bytes: await result.readAsBytes(), name: result.path);
} else {
if (!mounted) return;
final picked = await selectFiles(
context,
allowMultiple: false,
@ -143,6 +154,7 @@ class ChatDetailsController extends State<ChatDetails> {
name: pickedFile.name,
);
}
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => room!.setAvatar(file),

View file

@ -44,7 +44,7 @@ class ChatDetailsView extends StatelessWidget {
),
builder: (context, snapshot) {
var members = room.getParticipants().toList()
..sort((b, a) => a.powerLevel.compareTo(b.powerLevel));
..sort((b, a) => a.powerLevel.level.compareTo(b.powerLevel.level));
members = members.take(10).toList();
final actualMembersCount =
(room.summary.mInvitedMemberCount ?? 0) +

View file

@ -23,13 +23,16 @@ class ParticipantListItem extends StatelessWidget {
Membership.leave => L10n.of(context).leftTheChat,
};
final permissionBatch = user.room.creatorUserIds.contains(user.id)
? L10n.of(context).owner
: user.powerLevel >= 100
? L10n.of(context).admin
: user.powerLevel >= 50
? L10n.of(context).moderator
: '';
final permissionBatch = switch (user.powerLevel.role) {
PowerLevelRole.user => '',
PowerLevelRole.moderator => L10n.of(context).moderator,
PowerLevelRole.admin => L10n.of(context).admin,
PowerLevelRole.owner => L10n.of(context).owner,
};
final isAdminOrOwner =
user.powerLevel.role == PowerLevelRole.admin ||
user.powerLevel.role == PowerLevelRole.owner;
return ListTile(
onTap: () => showMemberActionsPopupMenu(context: context, user: user),
@ -45,7 +48,7 @@ class ParticipantListItem extends StatelessWidget {
Container(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
decoration: BoxDecoration(
color: user.powerLevel >= 100
color: isAdminOrOwner
? theme.colorScheme.tertiary
: theme.colorScheme.tertiaryContainer,
borderRadius: BorderRadius.circular(AppConfig.borderRadius),
@ -53,7 +56,7 @@ class ParticipantListItem extends StatelessWidget {
child: Text(
permissionBatch,
style: theme.textTheme.labelSmall?.copyWith(
color: user.powerLevel >= 100
color: isAdminOrOwner
? theme.colorScheme.onTertiary
: theme.colorScheme.onTertiaryContainer,
),

View file

@ -30,38 +30,40 @@ class ChatEncryptionSettingsController extends State<ChatEncryptionSettings> {
}
Future<void> enableEncryption(_) async {
final l10n = L10n.of(context);
if (room.encrypted) {
showOkAlertDialog(
context: context,
title: L10n.of(context).sorryThatsNotPossible,
message: L10n.of(context).disableEncryptionWarning,
title: l10n.sorryThatsNotPossible,
message: l10n.disableEncryptionWarning,
);
return;
}
if (room.joinRules == JoinRules.public) {
showOkAlertDialog(
context: context,
title: L10n.of(context).sorryThatsNotPossible,
message: L10n.of(context).noEncryptionForPublicRooms,
title: l10n.sorryThatsNotPossible,
message: l10n.noEncryptionForPublicRooms,
);
return;
}
if (!room.canChangeStateEvent(EventTypes.Encryption)) {
showOkAlertDialog(
context: context,
title: L10n.of(context).sorryThatsNotPossible,
message: L10n.of(context).noPermission,
title: l10n.sorryThatsNotPossible,
message: l10n.noPermission,
);
return;
}
final consent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
message: L10n.of(context).enableEncryptionWarning,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).cancel,
title: l10n.areYouSure,
message: l10n.enableEncryptionWarning,
okLabel: l10n.yes,
cancelLabel: l10n.cancel,
);
if (consent != OkCancelResult.ok) return;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => room.enableEncryption(),
@ -69,14 +71,16 @@ class ChatEncryptionSettingsController extends State<ChatEncryptionSettings> {
}
Future<void> startVerification() async {
final l10n = L10n.of(context);
final consent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).verifyOtherUser,
message: L10n.of(context).verifyOtherUserDescription,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.verifyOtherUser,
message: l10n.verifyOtherUserDescription,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
);
if (consent != OkCancelResult.ok) return;
if (!mounted) return;
final req = await room.client.userDeviceKeys[room.directChatMatrixID]!
.startVerification();
req.onUpdate = () {
@ -84,6 +88,7 @@ class ChatEncryptionSettingsController extends State<ChatEncryptionSettings> {
setState(() {});
}
};
if (!mounted) return;
await KeyVerificationDialog(request: req).show(context);
}

View file

@ -1,6 +1,6 @@
import 'dart:async';
import 'dart:developer';
import 'package:collection/collection.dart';
import 'package:cross_file/cross_file.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/l10n/l10n.dart';
@ -30,7 +30,7 @@ import '../../config/setting_keys.dart';
import '../../utils/url_launcher.dart';
import '../../widgets/matrix.dart';
enum ActiveFilter { allChats, messages, groups, unread, spaces }
enum ActiveFilter { allChats, messages, groups, unread, spaces, tag }
extension LocalizedActiveFilter on ActiveFilter {
String toLocalizedString(BuildContext context) {
@ -45,6 +45,8 @@ extension LocalizedActiveFilter on ActiveFilter {
return L10n.of(context).groups;
case ActiveFilter.spaces:
return L10n.of(context).spaces;
case ActiveFilter.tag:
throw 'Tags should not directly be displayed!';
}
}
}
@ -73,6 +75,7 @@ class ChatListController extends State<ChatList>
StreamSubscription? _intentFileStreamSubscription;
late ActiveFilter activeFilter;
String? activeTag;
String? _activeSpaceId;
String? get activeSpaceId => _activeSpaceId;
@ -90,6 +93,8 @@ class ChatListController extends State<ChatList>
});
Future<void> onChatTap(Room room) async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
if (room.membership == Membership.invite) {
final joinResult = await showFutureLoadingDialog(
context: context,
@ -105,10 +110,11 @@ class ChatListController extends State<ChatList>
);
if (joinResult.error != null) return;
}
if (!mounted) return;
if (room.membership == Membership.ban) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).youHaveBeenBannedFromThisChat)),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.youHaveBeenBannedFromThisChat)),
);
return;
}
@ -138,6 +144,8 @@ class ChatListController extends State<ChatList>
return (room) => room.isUnreadOrInvited;
case ActiveFilter.spaces:
return (room) => room.isSpace;
case ActiveFilter.tag:
return (room) => room.tags.keys.contains(activeTag);
}
}
@ -156,23 +164,25 @@ class ChatListController extends State<ChatList>
static const String _serverStoreNamespace = 'im.fluffychat.search.server';
Future<void> setServer() async {
final matrix = Matrix.of(context);
final l10n = L10n.of(context);
final newServer = await showTextInputDialog(
useRootNavigator: false,
title: L10n.of(context).changeTheHomeserver,
title: l10n.changeTheHomeserver,
context: context,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
prefixText: 'https://',
hintText: Matrix.of(context).client.homeserver?.host,
hintText: matrix.client.homeserver?.host,
initialText: searchServer,
keyboardType: TextInputType.url,
autocorrect: false,
validator: (server) => server.contains('.') == true
? null
: L10n.of(context).invalidServerName,
validator: (server) =>
server.contains('.') == true ? null : l10n.invalidServerName,
);
if (newServer == null) return;
Matrix.of(context).store.setString(_serverStoreNamespace, newServer);
if (!mounted) return;
matrix.store.setString(_serverStoreNamespace, newServer);
setState(() {
searchServer = newServer;
});
@ -185,6 +195,7 @@ class ChatListController extends State<ChatList>
Future<void> _search() async {
final client = Matrix.of(context).client;
final scaffoldMessenger = ScaffoldMessenger.of(context);
if (!isSearching) {
setState(() {
isSearching = true;
@ -227,9 +238,10 @@ class ChatListController extends State<ChatList>
);
} catch (e, s) {
Logs().w('Searching has crashed', e, s);
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text(e.toLocalizedString(context))));
if (!mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(e.toLocalizedString(context))),
);
}
if (!isSearchMode) return;
setState(() {
@ -293,9 +305,8 @@ class ChatListController extends State<ChatList>
Future<void> editSpace(BuildContext context, String spaceId) async {
await Matrix.of(context).client.getRoomById(spaceId)!.postLoad();
if (mounted) {
context.push('/rooms/$spaceId/details');
}
if (!context.mounted) return;
context.push('/rooms/$spaceId/details');
}
// Needs to match GroupsSpacesEntry for 'separate group' checking.
@ -305,11 +316,10 @@ class ChatListController extends State<ChatList>
String? get activeChat => widget.activeChat;
void _processIncomingSharedMedia(List<SharedMediaFile> files) {
files.removeWhere(
(file) => file.path.startsWith(AppConfig.deepLinkPrefix) == true,
);
if (files.isEmpty) return;
inspect(files);
if (files.singleOrNull?.path.startsWith(AppConfig.deepLinkPrefix) == true) {
return;
}
showScaffoldDialog(
context: context,
@ -353,9 +363,10 @@ class ChatListController extends State<ChatList>
}
}
StreamSubscription? _onRoomTagUpdate;
@override
void initState() {
activeFilter = ActiveFilter.allChats;
_initReceiveSharingIntent();
_activeSpaceId = widget.activeSpace;
@ -378,6 +389,32 @@ class ChatListController extends State<ChatList>
);
});
_updateRoomTags();
_onRoomTagUpdate = Matrix.of(context).client.onSync.stream
.where(
(syncUpdate) =>
syncUpdate.rooms?.join?.values.any(
(roomUpdate) =>
roomUpdate.accountData?.any(
(accountData) => accountData.type == 'm.tag',
) ??
false,
) ??
false,
)
.listen(_updateRoomTags);
if (roomTags.containsKey(AppSettings.chatFilter.value)) {
activeFilter = ActiveFilter.tag;
activeTag = AppSettings.chatFilter.value;
} else {
activeFilter =
ActiveFilter.values.singleWhereOrNull(
(filter) => AppSettings.chatFilter.value == filter.name,
) ??
ActiveFilter.allChats;
}
super.initState();
}
@ -385,6 +422,7 @@ class ChatListController extends State<ChatList>
void dispose() {
_intentDataStreamSubscription?.cancel();
_intentFileStreamSubscription?.cancel();
_onRoomTagUpdate?.cancel();
scrollController.removeListener(_onScroll);
super.dispose();
}
@ -613,6 +651,30 @@ class ChatListController extends State<ChatList>
],
),
),
if (activeTag == null)
PopupMenuItem(
value: ChatContextAction.addTag,
child: Row(
mainAxisSize: .min,
children: [
Icon(Icons.bookmark_add_outlined),
const SizedBox(width: 12),
Text(L10n.of(context).addTag),
],
),
)
else
PopupMenuItem(
value: ChatContextAction.removeTag,
child: Row(
mainAxisSize: .min,
children: [
Icon(Icons.bookmark_remove_outlined),
const SizedBox(width: 12),
Text(L10n.of(context).removeTag),
],
),
),
if (spacesWithPowerLevels.isNotEmpty)
PopupMenuItem(
value: ChatContextAction.addToSpace,
@ -742,6 +804,7 @@ class ChatListController extends State<ChatList>
.toList(),
);
if (space == null) return;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => space.setSpaceChild(room.id),
@ -752,9 +815,68 @@ class ChatListController extends State<ChatList>
future: () => room.setLowPriority(!room.isLowPriority),
);
return;
case ChatContextAction.addTag:
final existingTags = List.of(roomTags.keys);
existingTags.removeWhere(room.tags.containsKey);
String? tag;
if (existingTags.isNotEmpty) {
tag = await showModalActionPopup<String?>(
context: context,
actions: [
...existingTags.map((tag) {
final displayTag = tag.replaceFirst('u.', '');
return AdaptiveModalAction(
label: displayTag,
value: displayTag,
);
}),
AdaptiveModalAction(
label: L10n.of(context).createNewTag,
value: null,
),
],
);
if (!mounted) return;
}
tag ??= await showTextInputDialog(
context: context,
title: L10n.of(context).addTag,
hintText: L10n.of(context).tagName,
);
final newTag = tag;
if (!mounted) return;
if (newTag == null) return;
await showFutureLoadingDialog(
context: context,
future: () => room.addTag('u.$newTag'),
);
return;
case ChatContextAction.removeTag:
await showFutureLoadingDialog(
context: context,
future: () => room.removeTag(activeTag!),
);
return;
}
}
Map<String, int> roomTags = {};
void _updateRoomTags([_]) {
roomTags.clear();
for (final room in Matrix.of(context).client.rooms) {
for (final tag in room.tags.keys) {
if (tag.startsWith('u.')) roomTags[tag] = (roomTags[tag] ?? 0) + 1;
}
}
setState(() {
if (activeTag != null && !roomTags.keys.contains(activeTag)) {
activeTag = null;
activeFilter = ActiveFilter.allChats;
}
});
}
Future<void> dismissStatusList() async {
final result = await showOkCancelAlertDialog(
title: L10n.of(context).hidePresences,
@ -767,16 +889,18 @@ class ChatListController extends State<ChatList>
}
Future<void> setStatus() async {
final l10n = L10n.of(context);
final client = Matrix.of(context).client;
final currentPresence = await client.fetchCurrentPresence(client.userID!);
if (!mounted) return;
final input = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).setStatus,
message: L10n.of(context).leaveEmptyToClearStatus,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
hintText: L10n.of(context).statusExampleMessage,
title: l10n.setStatus,
message: l10n.leaveEmptyToClearStatus,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: l10n.statusExampleMessage,
maxLines: 6,
minLines: 1,
maxLength: 255,
@ -846,10 +970,17 @@ class ChatListController extends State<ChatList>
}
}
void setActiveFilter(ActiveFilter filter) {
void setActiveFilter(ActiveFilter filter, String? tag) {
if (filter == ActiveFilter.tag && tag == null) {
throw ('Must set a tag when setting filter to tags!');
}
setState(() {
activeTag = tag;
activeFilter = filter;
});
AppSettings.chatFilter.setItem(
filter == ActiveFilter.tag ? tag! : filter.name,
);
}
void setActiveClient(Client client) {
@ -904,18 +1035,21 @@ class ChatListController extends State<ChatList>
if (action == null) return;
switch (action) {
case EditBundleAction.addToBundle:
if (!mounted) return;
final bundle = await showTextInputDialog(
context: context,
title: l10n.bundleName,
hintText: l10n.bundleName,
);
if (bundle == null || bundle.isEmpty || bundle.isEmpty) return;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => client.setAccountBundle(bundle),
);
break;
case EditBundleAction.removeFromBundle:
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => client.removeFromAccountBundle(activeBundle!),
@ -962,6 +1096,8 @@ enum ChatContextAction {
goToSpace,
favorite,
lowPriority,
addTag,
removeTag,
markUnread,
mute,
leave,

View file

@ -134,37 +134,40 @@ class ChatListViewBody extends StatelessWidget {
padding: const EdgeInsets.all(12.0),
shrinkWrap: true,
scrollDirection: Axis.horizontal,
children:
[
ActiveFilter.allChats,
if (spaces.isNotEmpty &&
!AppSettings
.displayNavigationRail
.value &&
!FluffyThemes.isColumnMode(context))
ActiveFilter.spaces,
ActiveFilter.unread,
ActiveFilter.groups,
ActiveFilter.messages,
]
.map(
(filter) => Padding(
padding: const EdgeInsets.symmetric(
horizontal: 4.0,
),
child: FilterChip(
selected:
filter == controller.activeFilter,
onSelected: (_) =>
controller.setActiveFilter(filter),
label: Text(
filter.toLocalizedString(context),
),
children: [
...ActiveFilter.values
.where((filter) => filter != ActiveFilter.tag)
.map(
(filter) => Padding(
padding: const EdgeInsets.symmetric(
horizontal: 4.0,
),
child: FilterChip(
selected: filter == controller.activeFilter,
onSelected: (_) => controller
.setActiveFilter(filter, null),
label: Text(
filter.toLocalizedString(context),
),
),
)
.toList(),
),
),
...controller.roomTags.entries.map(
(entry) => Padding(
padding: const EdgeInsets.symmetric(
horizontal: 4.0,
),
child: FilterChip(
selected: entry.key == controller.activeTag,
onSelected: (_) => controller.setActiveFilter(
ActiveFilter.tag,
entry.key,
),
label: Text(entry.key.replaceFirst('u.', '')),
),
),
),
],
),
),
if (controller.isSearchMode)

View file

@ -219,7 +219,15 @@ class ChatListItem extends StatelessWidget {
room.latestEventReceivedTime.localizedTimeShort(
context,
),
style: TextStyle(fontSize: 11),
style: TextStyle(
fontSize: 11,
fontWeight: room.hasNewMessages
? FontWeight.bold
: null,
color: hasNotifications
? theme.colorScheme.primary
: null,
),
),
),
],

View file

@ -209,6 +209,7 @@ class ClientChooserButton extends StatelessWidget {
cancelLabel: L10n.of(context).cancel,
);
if (consent != OkCancelResult.ok) return;
if (!context.mounted) return;
context.go('/rooms/settings/addaccount');
break;
case SettingsAction.newGroup:

View file

@ -170,14 +170,17 @@ class _SpaceViewState extends State<SpaceView> {
switch (action) {
case SpaceActions.settings:
await space?.postLoad();
if (!mounted) return;
context.push('/rooms/${widget.spaceId}/details');
break;
case SpaceActions.invite:
await space?.postLoad();
if (!mounted) return;
context.push('/rooms/${widget.spaceId}/invite');
break;
case SpaceActions.members:
await space?.postLoad();
if (!mounted) return;
context.push('/rooms/${widget.spaceId}/details/members');
break;
case SpaceActions.leave:
@ -524,14 +527,16 @@ class _SpaceViewState extends State<SpaceView> {
);
}
final item = _discoveredChildren[i];
var joinedRoom = room.client.getRoomById(item.roomId);
final displayname =
item.name ??
item.canonicalAlias ??
joinedRoom?.getLocalizedDisplayname() ??
L10n.of(context).emptyChat;
final avatarUrl = item.avatarUrl ?? joinedRoom?.avatar;
if (!displayname.toLowerCase().contains(filter)) {
return const SizedBox.shrink();
}
var joinedRoom = room.client.getRoomById(item.roomId);
if (joinedRoom?.membership == Membership.leave) {
joinedRoom = null;
}
@ -595,7 +600,7 @@ class _SpaceViewState extends State<SpaceView> {
)
: Avatar(
size: avatarSize,
mxContent: item.avatarUrl,
mxContent: avatarUrl,
name: '#',
backgroundColor:
theme.colorScheme.surfaceContainer,

View file

@ -1,7 +1,6 @@
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:matrix/matrix.dart';
import '../../widgets/matrix.dart';
@ -39,7 +38,7 @@ class ChatMembersController extends State<ChatMembersPage> {
if (filter.isEmpty) {
setState(() {
filteredMembers = members
?..sort((b, a) => a.powerLevel.compareTo(b.powerLevel));
?..sort((b, a) => a.powerLevel.level.compareTo(b.powerLevel.level));
});
return;
}
@ -52,7 +51,7 @@ class ChatMembersController extends State<ChatMembersPage> {
user.id.toLowerCase().contains(filter),
)
.toList()
?..sort((b, a) => a.powerLevel.compareTo(b.powerLevel));
?..sort((b, a) => a.powerLevel.level.compareTo(b.powerLevel.level));
});
}

View file

@ -36,6 +36,7 @@ class ChatPermissionsSettingsController extends State<ChatPermissionsSettings> {
currentLevel: currentLevel,
);
if (newLevel == null) return;
if (!context.mounted) return;
final content = Map<String, dynamic>.from(
room.getState(EventTypes.RoomPowerLevels)!.content,
);

View file

@ -41,12 +41,15 @@ class DevicesSettingsController extends State<DevicesSettings> {
Future<void> _checkChatBackup() async {
final client = Matrix.of(context).client;
final state = await client.getCryptoIdentityState();
if (!mounted) return;
setState(() {
chatBackupEnabled = state.initialized && !state.connected;
});
}
Future<void> removeDevicesAction(List<Device> devices) async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final client = Matrix.of(context).client;
final wellKnown = await Result.capture(client.getWellknown());
@ -57,18 +60,19 @@ class DevicesSettingsController extends State<DevicesSettings> {
launchUrlString(accountManageUrl, mode: LaunchMode.inAppBrowserView);
return;
}
if (!mounted) return;
if (await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
okLabel: L10n.of(context).remove,
cancelLabel: L10n.of(context).cancel,
message: L10n.of(context).removeDevicesDescription,
title: l10n.areYouSure,
okLabel: l10n.remove,
cancelLabel: l10n.cancel,
message: l10n.removeDevicesDescription,
isDestructive: true,
) ==
OkCancelResult.cancel) {
return;
}
final matrix = Matrix.of(context);
if (!mounted) return;
final deviceIds = <String>[];
for (final userDevice in devices) {
deviceIds.add(userDevice.deviceId);
@ -85,19 +89,21 @@ class DevicesSettingsController extends State<DevicesSettings> {
}
Future<void> renameDeviceAction(Device device) async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final displayName = await showTextInputDialog(
context: context,
title: L10n.of(context).changeDeviceName,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.changeDeviceName,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: device.displayName,
);
if (displayName == null) return;
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(
context,
).client.updateDevice(device.deviceId, displayName: displayName),
future: () =>
matrix.client.updateDevice(device.deviceId, displayName: displayName),
);
if (success.error == null) {
reload();
@ -105,17 +111,20 @@ class DevicesSettingsController extends State<DevicesSettings> {
}
Future<void> verifyDeviceAction(Device device) async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final consent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).verifyOtherDevice,
message: L10n.of(context).verifyOtherDeviceDescription,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.verifyOtherDevice,
message: l10n.verifyOtherDeviceDescription,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
);
if (consent != OkCancelResult.ok) return;
final req = await Matrix.of(context)
if (!mounted) return;
final req = await matrix
.client
.userDeviceKeys[Matrix.of(context).client.userID!]!
.userDeviceKeys[matrix.client.userID!]!
.deviceKeys[device.deviceId]!
.startVerification();
req.onUpdate = () {
@ -126,6 +135,7 @@ class DevicesSettingsController extends State<DevicesSettings> {
setState(() {});
}
};
if (!mounted) return;
await KeyVerificationDialog(request: req).show(context);
}

View file

@ -81,10 +81,12 @@ class EventVideoPlayerState extends State<EventVideoPlayer> {
);
});
} on IOException catch (e) {
if (!mounted) return;
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text(e.toLocalizedString(context))));
} catch (e, s) {
if (!mounted) return;
ErrorReporter(context, 'Unable to play video').onErrorCallback(e, s);
}
}

View file

@ -4,6 +4,7 @@ import 'package:fluffychat/widgets/matrix.dart';
import 'package:flutter/material.dart';
Future<void> restoreBackupFlow(BuildContext context) async {
final matrix = Matrix.of(context);
final picked = await selectFiles(context);
final file = picked.firstOrNull;
if (file == null) return;
@ -12,9 +13,9 @@ Future<void> restoreBackupFlow(BuildContext context) async {
await showFutureLoadingDialog(
context: context,
future: () async {
final client = await Matrix.of(context).getLoginClient();
final client = await matrix.getLoginClient();
await client.importDump(String.fromCharCodes(await file.readAsBytes()));
Matrix.of(context).initMatrix();
matrix.initMatrix();
},
);
}

View file

@ -166,6 +166,7 @@ class IntroPage extends StatelessWidget {
final client = await Matrix.of(
context,
).getLoginClient();
if (!context.mounted) return;
context.go(
'${GoRouterState.of(context).uri.path}/login',
extra: client,

View file

@ -75,7 +75,8 @@ class _IntroPagePresenterState extends State<IntroPagePresenter> {
final client = await Matrix.of(context).getLoginClient();
await client.checkHomeserver(homeserverUrl);
await client.oidcLogin(session: session, code: code, state: state);
if (context.mounted) context.go('/backup');
if (!mounted) return;
context.go('/backup');
} catch (e, s) {
Logs().w('Unable to login via OIDC', e, s);
if (mounted) {

View file

@ -54,6 +54,8 @@ class InvitationSelectionController extends State<InvitationSelection> {
String id,
String displayname,
) async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final room = Matrix.of(context).client.getRoomById(roomId!)!;
final success = await showFutureLoadingDialog(
@ -61,10 +63,9 @@ class InvitationSelectionController extends State<InvitationSelection> {
future: () => room.invite(id),
);
if (success.error == null) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(L10n.of(context).contactHasBeenInvitedToTheGroup),
),
if (!context.mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.contactHasBeenInvitedToTheGroup)),
);
}
}
@ -91,6 +92,7 @@ class InvitationSelectionController extends State<InvitationSelection> {
try {
response = await matrix.client.searchUserDirectory(text, limit: 10);
} catch (e) {
if (!context.mounted) return;
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text((e).toLocalizedString(context))));

View file

@ -82,6 +82,7 @@ class KeyVerificationPageState extends State<KeyVerificationDialog> {
},
);
if (valid.error != null) {
if (!mounted) return;
await showOkAlertDialog(
useRootNavigator: false,
context: context,
@ -178,9 +179,10 @@ class KeyVerificationPageState extends State<KeyVerificationDialog> {
);
buttons.add(
AdaptiveDialogAction(
onPressed: () => widget.request.rejectVerification().then(
(_) => Navigator.of(context, rootNavigator: false).pop(false),
),
onPressed: () => widget.request.rejectVerification().then((_) {
if (!context.mounted) return;
Navigator.of(context, rootNavigator: false).pop(false);
}),
child: Text(
L10n.of(context).reject,
style: TextStyle(color: theme.colorScheme.error),

View file

@ -130,15 +130,19 @@ class LoginController extends State<Login> {
Logs().v(
'$newDomain is not running a homeserver, asking to use $oldHomeserver',
);
if (!mounted) return;
final l10n = L10n.of(context);
final dialogResult = await showOkCancelAlertDialog(
context: context,
useRootNavigator: false,
title: L10n.of(
context,
).noMatrixServer(newDomain.toString(), oldHomeserver.toString()),
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.noMatrixServer(
newDomain.toString(),
oldHomeserver.toString(),
),
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
);
if (!mounted) return;
if (dialogResult == OkCancelResult.ok) {
if (mounted) setState(() => usernameError = null);
} else {
@ -156,26 +160,30 @@ class LoginController extends State<Login> {
}
} catch (e) {
widget.client.homeserver = oldHomeserver;
if (!mounted) return;
usernameError = e.toLocalizedString(context);
if (mounted) setState(() {});
}
}
Future<void> passwordForgotten() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final input = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).passwordForgotten,
message: L10n.of(context).enterAnEmailAddress,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.passwordForgotten,
message: l10n.enterAnEmailAddress,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
initialText: usernameController.text.isEmail
? usernameController.text
: '',
hintText: L10n.of(context).enterAnEmailAddress,
hintText: l10n.enterAnEmailAddress,
keyboardType: TextInputType.emailAddress,
);
if (input == null) return;
if (!mounted) return;
final clientSecret = DateTime.now().millisecondsSinceEpoch.toString();
final response = await showFutureLoadingDialog(
context: context,
@ -186,27 +194,30 @@ class LoginController extends State<Login> {
),
);
if (response.error != null) return;
if (!mounted) return;
final password = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).passwordForgotten,
message: L10n.of(context).chooseAStrongPassword,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.passwordForgotten,
message: l10n.chooseAStrongPassword,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: '******',
obscureText: true,
minLines: 1,
maxLines: 1,
);
if (password == null) return;
if (!mounted) return;
final ok = await showOkAlertDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).weSentYouAnEmail,
message: L10n.of(context).pleaseClickOnLink,
okLabel: L10n.of(context).iHaveClickedOnLink,
title: l10n.weSentYouAnEmail,
message: l10n.pleaseClickOnLink,
okLabel: l10n.iHaveClickedOnLink,
);
if (ok != OkCancelResult.ok) return;
if (!mounted) return;
final data = <String, dynamic>{
'new_password': password,
'logout_devices': false,
@ -226,9 +237,10 @@ class LoginController extends State<Login> {
data: data,
),
);
if (!mounted) return;
if (success.error == null) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).passwordHasBeenChanged)),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.passwordHasBeenChanged)),
);
usernameController.text = input;
passwordController.text = password;

View file

@ -81,17 +81,19 @@ class NewPrivateChatController extends State<NewPrivateChat> {
void inviteAction() => FluffyShare.shareInviteLink(context);
Future<void> openScannerAction() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
if (PlatformInfos.isAndroid) {
final info = await DeviceInfoPlugin().androidInfo;
if (!mounted) return;
if (info.version.sdkInt < 21) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(L10n.of(context).unsupportedAndroidVersionLong),
),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.unsupportedAndroidVersionLong)),
);
return;
}
}
if (!mounted) return;
await showAdaptiveBottomSheet(
context: context,
builder: (_) => QrScannerModal(
@ -101,12 +103,15 @@ class NewPrivateChatController extends State<NewPrivateChat> {
}
Future<void> copyUserId() async {
final scaffoldMessenger = ScaffoldMessenger.of(context);
final l10n = L10n.of(context);
await Clipboard.setData(
ClipboardData(text: Matrix.of(context).client.userID!),
);
ScaffoldMessenger.of(
context,
).showSnackBar(SnackBar(content: Text(L10n.of(context).copiedToClipboard)));
if (!mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.copiedToClipboard)),
);
}
void openUserModal(Profile profile) =>

View file

@ -66,6 +66,7 @@ class QrScannerModalState extends State<QrScannerModal> {
late StreamSubscription sub;
sub = controller.scannedDataStream.listen((scanData) {
sub.cancel();
if (!mounted) return;
Navigator.of(context).pop();
final data = scanData.code;
if (data != null) widget.onScan(data);

View file

@ -37,18 +37,20 @@ class SettingsController extends State<Settings> {
});
Future<void> setDisplaynameAction() async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final profile = await profileFuture;
if (!mounted) return;
final input = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).editDisplayname,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
initialText:
profile?.displayName ?? Matrix.of(context).client.userID!.localpart,
title: l10n.editDisplayname,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
initialText: profile?.displayName ?? matrix.client.userID!.localpart,
);
if (input == null) return;
final matrix = Matrix.of(context);
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => matrix.client.setProfileField(
@ -63,19 +65,19 @@ class SettingsController extends State<Settings> {
}
Future<void> logoutAction() async {
if (await showOkCancelAlertDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).areYouSureYouWantToLogout,
message: L10n.of(context).noBackupWarning,
isDestructive: cryptoIdentityConnected == false,
okLabel: L10n.of(context).logout,
cancelLabel: L10n.of(context).cancel,
) ==
OkCancelResult.cancel) {
return;
}
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final consent = await showOkCancelAlertDialog(
useRootNavigator: false,
context: context,
title: l10n.areYouSureYouWantToLogout,
message: l10n.noBackupWarning,
isDestructive: cryptoIdentityConnected == false,
okLabel: l10n.logout,
cancelLabel: l10n.cancel,
);
if (consent != OkCancelResult.ok) return;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => matrix.client.logout(),
@ -83,24 +85,27 @@ class SettingsController extends State<Settings> {
}
Future<void> setAvatarAction() async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final profile = await profileFuture;
if (!mounted) return;
final actions = [
if (PlatformInfos.isMobile)
AdaptiveModalAction(
value: AvatarAction.camera,
label: L10n.of(context).openCamera,
label: l10n.openCamera,
isDefaultAction: true,
icon: const Icon(Icons.camera_alt_outlined),
),
AdaptiveModalAction(
value: AvatarAction.file,
label: L10n.of(context).openGallery,
label: l10n.openGallery,
icon: const Icon(Icons.photo_outlined),
),
if (profile?.avatarUrl != null)
AdaptiveModalAction(
value: AvatarAction.remove,
label: L10n.of(context).removeYourAvatar,
label: l10n.removeYourAvatar,
isDestructive: true,
icon: const Icon(Icons.delete_outlined),
),
@ -109,12 +114,12 @@ class SettingsController extends State<Settings> {
? actions.single.value
: await showModalActionPopup<AvatarAction>(
context: context,
title: L10n.of(context).changeYourAvatar,
cancelLabel: L10n.of(context).cancel,
title: l10n.changeYourAvatar,
cancelLabel: l10n.cancel,
actions: actions,
);
if (action == null) return;
final matrix = Matrix.of(context);
if (!mounted) return;
if (action == AvatarAction.remove) {
final success = await showFutureLoadingDialog(
context: context,
@ -139,12 +144,14 @@ class SettingsController extends State<Settings> {
bytes = await result.readAsBytes();
name = result.path;
} else {
if (!mounted) return;
final result = await selectFiles(context, type: FileType.image);
final pickedFile = result.firstOrNull;
if (pickedFile == null) return;
bytes = await pickedFile.readAsBytes();
name = pickedFile.name;
}
if (!mounted) return;
final cropped = await showDialog<Uint8List>(
context: context,
builder: (contect) => AvatarCropDialog(image: bytes),
@ -155,6 +162,7 @@ class SettingsController extends State<Settings> {
bytes: bytes,
name: name,
);
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => matrix.client.setAvatar(file),
@ -181,6 +189,7 @@ class SettingsController extends State<Settings> {
}
final state = await client.getCryptoIdentityState();
if (!mounted) return;
setState(() {
cryptoIdentityConnected = state.initialized && state.connected;
});

View file

@ -19,41 +19,48 @@ class Settings3Pid extends StatefulWidget {
class Settings3PidController extends State<Settings3Pid> {
Future<void> add3PidAction() async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
final input = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).enterAnEmailAddress,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
hintText: L10n.of(context).enterAnEmailAddress,
title: l10n.enterAnEmailAddress,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
hintText: l10n.enterAnEmailAddress,
keyboardType: TextInputType.emailAddress,
);
if (input == null) return;
if (!mounted) return;
final clientSecret = DateTime.now().millisecondsSinceEpoch.toString();
final response = await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(context).client.requestTokenToRegisterEmail(
future: () => matrix.client.requestTokenToRegisterEmail(
clientSecret,
input,
Settings3Pid.sendAttempt++,
),
);
if (response.error != null) return;
if (!mounted) return;
final ok = await showOkAlertDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).weSentYouAnEmail,
message: L10n.of(context).pleaseClickOnLink,
okLabel: L10n.of(context).iHaveClickedOnLink,
title: l10n.weSentYouAnEmail,
message: l10n.pleaseClickOnLink,
okLabel: l10n.iHaveClickedOnLink,
);
if (ok != OkCancelResult.ok) return;
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
delay: false,
future: () => Matrix.of(context).client.uiaRequestBackground(
(auth) => Matrix.of(
context,
).client.add3PID(clientSecret, response.result!.sid, auth: auth),
future: () => matrix.client.uiaRequestBackground(
(auth) => matrix.client.add3PID(
clientSecret,
response.result!.sid,
auth: auth,
),
),
);
if (success.error != null) return;
@ -63,21 +70,25 @@ class Settings3PidController extends State<Settings3Pid> {
Future<List<ThirdPartyIdentifier>?>? request;
Future<void> delete3Pid(ThirdPartyIdentifier identifier) async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
if (await showOkCancelAlertDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).areYouSure,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).cancel,
title: l10n.areYouSure,
okLabel: l10n.yes,
cancelLabel: l10n.cancel,
) !=
OkCancelResult.ok) {
return;
}
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(
context,
).client.delete3pidFromAccount(identifier.address, identifier.medium),
future: () => matrix.client.delete3pidFromAccount(
identifier.address,
identifier.medium,
),
);
if (success.error != null) return;
setState(() => request = null);

View file

@ -91,6 +91,7 @@ class _ImportEmoteArchiveDialogState extends State<ImportEmoteArchiveDialog> {
}
Future<void> _addEmotePack() async {
final matrix = Matrix.of(context);
setState(() {
_loading = true;
_progress = 0;
@ -148,7 +149,7 @@ class _ImportEmoteArchiveDialogState extends State<ImportEmoteArchiveDialog> {
} else {
mxcFile = thumbnail;
}
final uri = await Matrix.of(context).client.uploadContent(
final uri = await matrix.client.uploadContent(
mxcFile.bytes,
filename: mxcFile.name,
contentType: mxcFile.mimeType,
@ -178,6 +179,7 @@ class _ImportEmoteArchiveDialogState extends State<ImportEmoteArchiveDialog> {
}
}
if (!mounted) return;
await widget.controller.save(context);
_importMap.removeWhere(
(key, value) => successfulUploads.contains(key.name),

View file

@ -293,6 +293,7 @@ class EmotesSettingsController extends State<EmotesSettings> {
}
Future<void> createStickers() async {
final matrix = Matrix.of(context);
final pickedFiles = await selectFiles(
context,
type: FileType.image,
@ -315,7 +316,7 @@ class EmotesSettingsController extends State<EmotesSettings> {
nativeImplementations: ClientManager.nativeImplementations,
) ??
file;
final uri = await Matrix.of(context).client.uploadContent(
final uri = await matrix.client.uploadContent(
file.bytes,
filename: file.name,
contentType: file.mimeType,
@ -361,6 +362,7 @@ class EmotesSettingsController extends State<EmotesSettings> {
final buffer = InputMemoryStream(await result.single.readAsBytes());
final archive = ZipDecoder().decodeStream(buffer);
if (!mounted) return;
await showDialog(
context: context,
@ -375,7 +377,7 @@ class EmotesSettingsController extends State<EmotesSettings> {
Future<void> exportAsZip() async {
final client = Matrix.of(context).client;
await showFutureLoadingDialog(
final result = await showFutureLoadingDialog<MatrixFile>(
context: context,
future: () async {
final pack = _getPack();
@ -397,11 +399,12 @@ class EmotesSettingsController extends State<EmotesSettings> {
'${pack.pack.displayName ?? client.userID?.localpart ?? 'emotes'}.zip';
final output = ZipEncoder().encode(archive);
MatrixFile(
name: fileName,
bytes: Uint8List.fromList(output),
).save(context);
return MatrixFile(name: fileName, bytes: Uint8List.fromList(output));
},
);
final file = result.result;
if (file == null) return;
if (!mounted) return;
file.save(context);
}
}

View file

@ -40,6 +40,7 @@ class SettingsNotificationsController extends State<SettingsNotifications> {
],
);
if (delete != true) return;
if (!mounted) return;
final success = await showFutureLoadingDialog(
context: context,

View file

@ -1,7 +1,10 @@
import 'package:fluffychat/config/setting_keys.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/pages/settings_notifications/push_rule_extensions.dart';
import 'package:fluffychat/widgets/layouts/max_width_body.dart';
import 'package:fluffychat/widgets/settings_switch_list_tile.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:matrix/matrix.dart';
@ -47,6 +50,11 @@ class SettingsNotificationsView extends StatelessWidget {
return SelectionArea(
child: Column(
children: [
if (kIsWeb)
SettingsSwitchListTile.adaptive(
title: L10n.of(context).playSoundOnNotification,
setting: AppSettings.webNotificationSound,
),
if (pushRules != null)
for (final category in pushCategories) ...[
ListTile(

View file

@ -24,6 +24,8 @@ class SettingsPasswordController extends State<SettingsPassword> {
bool loading = false;
Future<void> changePassword() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
setState(() {
oldPasswordError = newPassword1Error = newPassword2Error = null;
});
@ -51,13 +53,13 @@ class SettingsPasswordController extends State<SettingsPassword> {
loading = true;
});
try {
final scaffoldMessenger = ScaffoldMessenger.of(context);
await Matrix.of(context).client.changePassword(
newPassword1Controller.text,
oldPassword: oldPasswordController.text,
);
if (!mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(L10n.of(context).passwordHasBeenChanged)),
SnackBar(content: Text(l10n.passwordHasBeenChanged)),
);
if (mounted) context.pop();
} catch (e) {

View file

@ -19,20 +19,21 @@ class SettingsSecurity extends StatefulWidget {
class SettingsSecurityController extends State<SettingsSecurity> {
Future<void> setAppLockAction() async {
final l10n = L10n.of(context);
if (AppLock.of(context).isActive) {
AppLock.of(context).showLockScreen();
}
final newLock = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).pleaseChooseAPasscode,
message: L10n.of(context).pleaseEnter4Digits,
cancelLabel: L10n.of(context).cancel,
title: l10n.pleaseChooseAPasscode,
message: l10n.pleaseEnter4Digits,
cancelLabel: l10n.cancel,
validator: (text) {
if (text.isEmpty || (text.length == 4 && int.tryParse(text)! >= 0)) {
return null;
}
return L10n.of(context).pleaseEnter4Digits;
return l10n.pleaseEnter4Digits;
},
keyboardType: TextInputType.number,
obscureText: true,
@ -41,53 +42,55 @@ class SettingsSecurityController extends State<SettingsSecurity> {
maxLength: 4,
);
if (newLock != null) {
if (!mounted) return;
await AppLock.of(context).changePincode(newLock);
}
}
Future<void> deleteAccountAction() async {
final l10n = L10n.of(context);
final matrix = Matrix.of(context);
if (await showOkCancelAlertDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).warning,
message: L10n.of(context).deactivateAccountWarning,
okLabel: L10n.of(context).ok,
cancelLabel: L10n.of(context).cancel,
title: l10n.warning,
message: l10n.deactivateAccountWarning,
okLabel: l10n.ok,
cancelLabel: l10n.cancel,
isDestructive: true,
) ==
OkCancelResult.cancel) {
return;
}
final supposedMxid = Matrix.of(context).client.userID!;
if (!mounted) return;
final supposedMxid = matrix.client.userID!;
final mxid = await showTextInputDialog(
useRootNavigator: false,
context: context,
title: L10n.of(context).confirmMatrixId,
validator: (text) => text == supposedMxid
? null
: L10n.of(context).supposedMxid(supposedMxid),
title: l10n.confirmMatrixId,
validator: (text) =>
text == supposedMxid ? null : l10n.supposedMxid(supposedMxid),
isDestructive: true,
okLabel: L10n.of(context).delete,
cancelLabel: L10n.of(context).cancel,
okLabel: l10n.delete,
cancelLabel: l10n.cancel,
);
if (mxid == null || mxid.isEmpty || mxid != supposedMxid) {
return;
}
if (!mounted) return;
final resp = await showFutureLoadingDialog(
context: context,
delay: false,
future: () =>
Matrix.of(context).client.uiaRequestBackground<IdServerUnbindResult?>(
(auth) => Matrix.of(
context,
).client.deactivateAccount(auth: auth, erase: true),
),
future: () => matrix.client.uiaRequestBackground<IdServerUnbindResult?>(
(auth) => matrix.client.deactivateAccount(auth: auth, erase: true),
),
);
if (!resp.isError) {
if (!mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(context).client.logout(),
future: () => matrix.client.logout(),
);
}
}

View file

@ -29,6 +29,7 @@ class SettingsStyleController extends State<SettingsStyle> {
final picked = await selectFiles(context, type: FileType.image);
final pickedFile = picked.firstOrNull;
if (pickedFile == null) return;
if (!mounted) return;
await showFutureLoadingDialog(
context: context,

View file

@ -108,6 +108,7 @@ class SignInPage extends StatelessWidget {
SizedBox.square(
dimension: 32,
child: IconButton(
tooltip: website,
icon: const Icon(
Icons.open_in_new_outlined,
size: 16,

View file

@ -122,7 +122,7 @@ class BackgroundPush {
//<GOOGLE_SERVICES>firebase.setListeners(
//<GOOGLE_SERVICES> onMessage: (message) => pushHelper(
//<GOOGLE_SERVICES> PushNotification.fromJson(
//<GOOGLE_SERVICES> Map<String, dynamic>.from(message['data'] ?? message),
//<GOOGLE_SERVICES> message.tryGetMap<String, Object>('data') ?? message,
//<GOOGLE_SERVICES> ),
//<GOOGLE_SERVICES> client: client,
//<GOOGLE_SERVICES> l10n: l10n,
@ -351,6 +351,9 @@ class BackgroundPush {
Future<void> setupFirebase() async {
Logs().v('Setup firebase');
if (_fcmToken?.isEmpty ?? true) {
if (PlatformInfos.isIOS) {
//<GOOGLE_SERVICES>await firebase.requestPermission();
}
try {
//<GOOGLE_SERVICES>_fcmToken = await firebase.getToken();
if (_fcmToken == null) throw ('PushToken is null');

View file

@ -13,7 +13,7 @@ Future<List<XFile>> selectFiles(
final result = await AppLock.of(context).pauseWhile(
showFutureLoadingDialog(
context: context,
future: () => FilePicker.platform.pickFiles(
future: () => FilePicker.pickFiles(
compressionQuality: 0,
allowMultiple: allowMultiple,
type: type,

View file

@ -12,6 +12,8 @@ abstract class FluffyShare {
BuildContext context, {
bool copyOnly = false,
}) async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
if (PlatformInfos.isMobile && !copyOnly) {
final box = context.findRenderObject() as RenderBox;
await SharePlus.instance.share(
@ -24,21 +26,20 @@ abstract class FluffyShare {
}
await Clipboard.setData(ClipboardData(text: text));
if (!PlatformInfos.isMobile) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
showCloseIcon: true,
content: Text(L10n.of(context).copiedToClipboard),
),
scaffoldMessenger.showSnackBar(
SnackBar(showCloseIcon: true, content: Text(l10n.copiedToClipboard)),
);
}
return;
}
static Future<void> shareInviteLink(BuildContext context) async {
final l10n = L10n.of(context);
final client = Matrix.of(context).client;
final ownProfile = await client.fetchOwnProfile();
if (!context.mounted) return;
await FluffyShare.share(
L10n.of(context).inviteText(
l10n.inviteText(
ownProfile.displayName ?? client.userID!,
'https://matrix.to/#/${client.userID}?client=im.fluffychat',
),

View file

@ -25,12 +25,14 @@ extension LocalizedBody on Event {
Future<void> saveFile(BuildContext context) async {
final matrixFile = await _getFile(context);
if (!context.mounted) return;
matrixFile.result?.save(context);
}
Future<void> shareFile(BuildContext context) async {
final matrixFile = await _getFile(context);
if (!context.mounted) return;
matrixFile.result?.share(context);
}

View file

@ -9,7 +9,7 @@ extension MatrixFileExtension on MatrixFile {
Future<void> save(BuildContext context) async {
final scaffoldMessenger = ScaffoldMessenger.of(context);
final l10n = L10n.of(context);
final downloadPath = await FilePicker.platform.saveFile(
final downloadPath = await FilePicker.saveFile(
dialogTitle: l10n.saveFile,
fileName: name,
type: filePickerFileType,

View file

@ -47,15 +47,17 @@ abstract class PlatformInfos {
}
static Future<void> showDialog(BuildContext context) async {
final l10n = L10n.of(context);
final version = await PlatformInfos.getVersion();
if (!context.mounted) return;
showAboutDialog(
context: context,
children: [
Text(L10n.of(context).versionWithNumber(version)),
Text(l10n.versionWithNumber(version)),
TextButton.icon(
onPressed: () => launchUrlString(AppConfig.sourceCodeUrl),
icon: const Icon(Icons.source_outlined),
label: Text(L10n.of(context).sourceCode),
label: Text(l10n.sourceCode),
),
Builder(
builder: (innerContext) {
@ -65,7 +67,7 @@ abstract class PlatformInfos {
Navigator.of(innerContext).pop();
},
icon: const Icon(Icons.list_outlined),
label: Text(L10n.of(context).logs),
label: Text(l10n.logs),
);
},
),
@ -77,7 +79,7 @@ abstract class PlatformInfos {
Navigator.of(innerContext).pop();
},
icon: const Icon(Icons.settings_applications_outlined),
label: Text(L10n.of(context).advancedConfigs),
label: Text(l10n.advancedConfigs),
);
},
),

View file

@ -17,6 +17,8 @@ import 'package:flutter_shortcuts_new/flutter_shortcuts_new.dart';
import 'package:matrix/matrix.dart';
const notificationAvatarDimension = 128;
const String groupKey = 'im.fluffychat.messages';
const int summaryId = -1;
Future<void> pushHelper(
PushNotification notification, {
@ -53,6 +55,7 @@ Future<void> pushHelper(
AppSettings.applicationName.value,
(notification.counts?.unread ?? 0).toString(),
),
groupKey: groupKey,
importance: Importance.high,
priority: Priority.max,
shortcutId: notification.roomId,
@ -252,7 +255,7 @@ Future<void> _tryPushHelper(
),
importance: Importance.high,
priority: Priority.max,
groupKey: event.room.spaceParents.firstOrNull?.roomId ?? 'rooms',
groupKey: groupKey,
actions: event.type == EventTypes.RoomMember || !useNotificationActions
? null
: <AndroidNotificationAction>[
@ -300,6 +303,41 @@ Future<void> _tryPushHelper(
event.eventId,
).toString(),
);
// Send summary notification on Android
if (PlatformInfos.isAndroid) {
final activeNotifications =
(await flutterLocalNotificationsPlugin.getActiveNotifications())
.where((n) => n.groupKey == groupKey)
.toList();
if (activeNotifications.isEmpty) {
return;
}
final title = l10n.unreadChatsInApp(
AppSettings.applicationName.value,
activeNotifications.length.toString(),
);
await flutterLocalNotificationsPlugin.show(
id: summaryId,
notificationDetails: NotificationDetails(
android: AndroidNotificationDetails(
AppConfig.pushNotificationsChannelId,
l10n.incomingMessages,
groupKey: groupKey,
setAsGroupSummary: true,
styleInformation: InboxStyleInformation(
activeNotifications.map((n) => n.body ?? '').toList(),
contentTitle: title,
summaryText: title,
),
autoCancel: false,
),
),
);
}
Logs().v('Push helper has been completed!');
}

View file

@ -10,6 +10,7 @@ abstract class UpdateNotifier {
static Future<void> showUpdateSnackBar(BuildContext context) async {
final scaffoldMessenger = ScaffoldMessenger.of(context);
final l10n = L10n.of(context);
final currentVersion = await PlatformInfos.getVersion();
final store = await SharedPreferences.getInstance();
final storedVersion = store.getString(versionStoreKey);
@ -20,9 +21,9 @@ abstract class UpdateNotifier {
SnackBar(
duration: const Duration(seconds: 30),
showCloseIcon: true,
content: Text(L10n.of(context).updateInstalled(currentVersion)),
content: Text(l10n.updateInstalled(currentVersion)),
action: SnackBarAction(
label: L10n.of(context).changelog,
label: l10n.changelog,
onPressed: () => launchUrlString(AppConfig.changelogUrl),
),
),

View file

@ -38,6 +38,7 @@ Future<void> connectToHomeserverFlow(
if ((kIsWeb || PlatformInfos.isLinux) &&
(supportsSso || authMetadata != null || (signUp && regLink != null))) {
if (!context.mounted) return;
final consent = await showOkCancelAlertDialog(
context: context,
title: l10n.appWantsToUseForLogin(homeserverInput),
@ -45,7 +46,9 @@ Future<void> connectToHomeserverFlow(
okLabel: l10n.continueText,
);
if (consent != OkCancelResult.ok) return;
if (!context.mounted) return;
}
if (!context.mounted) return;
if (authMetadata != null && AppSettings.enableMatrixNativeOIDC.value) {
await oidcLoginFlow(client, context, signUp);
@ -55,6 +58,7 @@ Future<void> connectToHomeserverFlow(
if (signUp && regLink != null) {
await launchUrlString(regLink);
}
if (!context.mounted) return;
final pathSegments = List.of(
GoRouter.of(context).routeInformationProvider.value.uri.pathSegments,
);

View file

@ -27,6 +27,8 @@ class UrlLauncher {
const UrlLauncher(this.context, this.url, [this.name]);
Future<void> launchUrl() async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
if (url!.toLowerCase().startsWith(AppConfig.deepLinkPrefix) ||
url!.toLowerCase().startsWith(AppConfig.inviteLinkPrefix) ||
{'#', '@', '!', '+', '\$'}.contains(url![0]) ||
@ -36,8 +38,8 @@ class UrlLauncher {
final uri = Uri.tryParse(url!);
if (uri == null) {
// we can't open this thing
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).cantOpenUri(url!))),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.cantOpenUri(url!))),
);
return;
}
@ -47,10 +49,10 @@ class UrlLauncher {
// that the user can see the actual url before opening the browser.
final consent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).openLinkInBrowser,
title: l10n.openLinkInBrowser,
message: url,
okLabel: L10n.of(context).open,
cancelLabel: L10n.of(context).cancel,
okLabel: l10n.open,
cancelLabel: l10n.cancel,
);
if (consent != OkCancelResult.ok) return;
}
@ -90,8 +92,8 @@ class UrlLauncher {
return;
}
if (uri.host.isEmpty) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).cantOpenUri(url!))),
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.cantOpenUri(url!))),
);
return;
}
@ -161,6 +163,7 @@ class UrlLauncher {
}
}
servers.addAll(identityParts.via);
if (!context.mounted) return;
if (room != null) {
if (room.isSpace) {
// TODO: Implement navigate to space
@ -178,6 +181,7 @@ class UrlLauncher {
}
return;
} else {
if (!context.mounted) return;
await showAdaptiveDialog(
context: context,
builder: (c) =>
@ -185,6 +189,7 @@ class UrlLauncher {
);
}
if (roomIdOrAlias.sigil == '!') {
if (!context.mounted) return;
if (await showOkCancelAlertDialog(
useRootNavigator: false,
context: context,
@ -192,6 +197,7 @@ class UrlLauncher {
) ==
OkCancelResult.ok) {
roomId = roomIdOrAlias;
if (!context.mounted) return;
final response = await showFutureLoadingDialog(
context: context,
future: () => matrix.client.joinRoom(
@ -200,11 +206,13 @@ class UrlLauncher {
),
);
if (response.error != null) return;
if (!context.mounted) return;
// wait for two seconds so that it probably came down /sync
await showFutureLoadingDialog(
context: context,
future: () => Future.delayed(const Duration(seconds: 2)),
);
if (!context.mounted) return;
if (event != null) {
context.go(
Uri(
@ -228,6 +236,7 @@ class UrlLauncher {
return Profile(userId: userId);
}),
);
if (!context.mounted) return;
await UserDialog.show(
context: context,
profile: profileResult.result!,

View file

@ -25,6 +25,7 @@ class PublicRoomDialog extends StatelessWidget {
const PublicRoomDialog({super.key, this.roomAlias, this.chunk, this.via});
Future<void> _joinRoom(BuildContext context) async {
final l10n = L10n.of(context);
final client = Matrix.of(context).client;
final chunk = this.chunk;
final knock = chunk?.joinRule == 'knock';
@ -48,12 +49,13 @@ class PublicRoomDialog extends StatelessWidget {
);
final roomId = result.result;
if (roomId == null) return;
if (!context.mounted) return;
if (knock && client.getRoomById(roomId) == null) {
Navigator.of(context).pop<bool>(true);
await showOkAlertDialog(
context: context,
title: L10n.of(context).youHaveKnocked,
message: L10n.of(context).pleaseWaitUntilInvited,
title: l10n.youHaveKnocked,
message: l10n.pleaseWaitUntilInvited,
);
return;
}
@ -73,6 +75,7 @@ class PublicRoomDialog extends StatelessWidget {
bool _testRoom(PublishedRoomsChunk r) => r.canonicalAlias == roomAlias;
Future<PublishedRoomsChunk> _search(BuildContext context) async {
final l10n = L10n.of(context);
final chunk = this.chunk;
if (chunk != null) return chunk;
final query = await Matrix.of(context).client.queryPublicRooms(
@ -80,7 +83,7 @@ class PublicRoomDialog extends StatelessWidget {
filter: PublicRoomQueryFilter(genericSearchTerm: roomAlias),
);
if (!query.chunk.any(_testRoom)) {
throw (L10n.of(context).noRoomsFound);
throw (l10n.noRoomsFound);
}
return query.chunk.firstWhere(_testRoom);
}
@ -248,6 +251,7 @@ class PublicRoomDialog extends StatelessWidget {
hintText: L10n.of(context).reason,
);
if (reason == null || reason.isEmpty) return;
if (!context.mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(context).client.reportRoom(

View file

@ -220,6 +220,7 @@ class UserDialog extends StatelessWidget {
hintText: L10n.of(context).reason,
);
if (reason == null || reason.isEmpty) return;
if (!context.mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => Matrix.of(

View file

@ -53,16 +53,18 @@ class ChatSettingsPopupMenuState extends State<ChatSettingsPopupMenu> {
onSelected: (choice) async {
switch (choice) {
case ChatPopupMenuActions.leave:
final l10n = L10n.of(context);
final router = GoRouter.of(context);
final confirmed = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
message: L10n.of(context).archiveRoomDescription,
okLabel: L10n.of(context).leave,
cancelLabel: L10n.of(context).cancel,
title: l10n.areYouSure,
message: l10n.archiveRoomDescription,
okLabel: l10n.leave,
cancelLabel: l10n.cancel,
isDestructive: true,
);
if (confirmed != OkCancelResult.ok) return;
if (!context.mounted) return;
final result = await showFutureLoadingDialog(
context: context,
future: () => widget.room.leave(),

View file

@ -43,7 +43,7 @@ class FluffyChatApp extends StatelessWidget {
// Pass deep links to app:
if (state.uri.toString().startsWith(AppConfig.deepLinkPrefix)) {
return '/rooms/newprivatechat?deeplink=${state.uri}';
return '/rooms/newprivatechat#${state.uri}';
}
return null;
},

View file

@ -5,6 +5,7 @@ import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/utils/localized_exception_extension.dart';
import 'package:fluffychat/widgets/adaptive_dialogs/adaptive_dialog_action.dart';
import 'package:flutter/material.dart';
import 'package:matrix/matrix_api_lite/utils/logs.dart';
/// Displays a loading dialog which reacts to the given [future]. The dialog
/// will be dismissed and the value will be returned when the future completes.
@ -40,6 +41,15 @@ Future<Result<T>> showFutureLoadingDialog<T>({
}
}
if (!context.mounted) {
Logs().e(
'Unable to show loading dialog!',
Exception('The BuildContext is not mounted!'),
StackTrace.current,
);
return Result.capture(futureExec);
}
final result = await showAdaptiveDialog<Result<T>>(
context: context,
barrierDismissible: barrierDismissible,

View file

@ -16,7 +16,12 @@ import 'package:matrix/matrix.dart';
import 'package:universal_html/html.dart' as html;
extension LocalNotificationsExtension on MatrixState {
static final html.AudioElement _audioPlayer = html.AudioElement()
..src = 'assets/assets/sounds/notification.ogg'
..load();
Future<void> showLocalNotification(Event event) async {
final l10n = L10n.of(context);
final roomId = event.room.id;
if (activeRoomId == roomId) {
if (WidgetsBinding.instance.lifecycleState == AppLifecycleState.resumed) {
@ -68,6 +73,8 @@ extension LocalNotificationsExtension on MatrixState {
);
}
if (AppSettings.webNotificationSound.value) _audioPlayer.play();
html.Notification(
title,
body: body,
@ -114,11 +121,11 @@ extension LocalNotificationsExtension on MatrixState {
actions: [
NotificationAction(
DesktopNotificationActions.openChat.name,
L10n.of(context).openChat,
l10n.openChat,
),
NotificationAction(
DesktopNotificationActions.seen.name,
L10n.of(context).markAsRead,
l10n.markAsRead,
),
],
hints: hints,

View file

@ -265,12 +265,19 @@ class MatrixState extends State<Matrix> with WidgetsBindingObserver {
InitWithRestoreExtension.deleteSessionBackup(name);
if (loggedInWithMultipleClients) {
final snackbarContext =
FluffyChatApp
.router
.routerDelegate
.navigatorKey
.currentContext ??
context;
if (!snackbarContext.mounted) return;
final l10n = L10n.of(snackbarContext);
ScaffoldMessenger.of(
FluffyChatApp.router.routerDelegate.navigatorKey.currentContext ??
context,
).showSnackBar(
SnackBar(content: Text(L10n.of(context).oneClientLoggedOut)),
);
snackbarContext,
).showSnackBar(SnackBar(content: Text(l10n.oneClientLoggedOut)));
return;
}
FluffyChatApp.router.go('/');
@ -382,15 +389,17 @@ class MatrixState extends State<Matrix> with WidgetsBindingObserver {
}
Future<void> dehydrateAction(BuildContext context) async {
final l10n = L10n.of(context);
final response = await showOkCancelAlertDialog(
context: context,
isDestructive: true,
title: L10n.of(context).dehydrate,
message: L10n.of(context).dehydrateWarning,
title: l10n.dehydrate,
message: l10n.dehydrateWarning,
);
if (response != OkCancelResult.ok) {
return;
}
if (!context.mounted) return;
final result = await showFutureLoadingDialog(
context: context,
future: client.exportDump,
@ -404,6 +413,7 @@ class MatrixState extends State<Matrix> with WidgetsBindingObserver {
'fluffychat-export-${DateFormat(DateFormat.YEAR_MONTH_DAY).format(DateTime.now())}.fluffybackup';
final file = MatrixFile(bytes: exportBytes, name: exportFileName);
if (!context.mounted) return;
file.save(context);
}
}

View file

@ -14,6 +14,8 @@ Future<void> showMemberActionsPopupMenu({
required User user,
void Function()? onMention,
}) async {
final l10n = L10n.of(context);
final scaffoldMessenger = ScaffoldMessenger.of(context);
final theme = Theme.of(context);
final displayname = user.calcDisplayname();
final isMe = user.room.client.userID == user.id;
@ -79,7 +81,7 @@ Future<void> showMemberActionsPopupMenu({
),
),
if (user.canChangeUserPowerLevel) ...[
if (user.powerLevel < 100)
if (user.powerLevel.level < 100)
PopupMenuItem(
value: _MemberActions.makeAdmin,
child: Row(
@ -90,7 +92,7 @@ Future<void> showMemberActionsPopupMenu({
],
),
),
if (user.powerLevel < 50)
if (user.powerLevel.level < 50)
PopupMenuItem(
value: _MemberActions.makeModerator,
child: Row(
@ -101,7 +103,7 @@ Future<void> showMemberActionsPopupMenu({
],
),
),
if (user.powerLevel >= 100)
if (user.powerLevel.role == PowerLevelRole.admin)
PopupMenuItem(
value: _MemberActions.removeAdmin,
child: Row(
@ -112,7 +114,7 @@ Future<void> showMemberActionsPopupMenu({
],
),
)
else if (user.powerLevel >= 50)
else if (user.powerLevel.role == PowerLevelRole.moderator)
PopupMenuItem(
value: _MemberActions.removeModerator,
child: Row(
@ -125,7 +127,7 @@ Future<void> showMemberActionsPopupMenu({
),
],
if (user.canChangeUserPowerLevel ||
!defaultPowerLevels.contains(user.powerLevel))
!defaultPowerLevels.contains(user.powerLevel.level))
PopupMenuItem(
value: _MemberActions.setPowerLevel,
enabled: user.canChangeUserPowerLevel,
@ -138,7 +140,7 @@ Future<void> showMemberActionsPopupMenu({
? L10n.of(context).setPowerLevel
: L10n.of(context).powerLevel,
),
if (!defaultPowerLevels.contains(user.powerLevel))
if (!defaultPowerLevels.contains(user.powerLevel.level))
Text(' (${user.powerLevel})'),
],
),
@ -217,8 +219,8 @@ Future<void> showMemberActionsPopupMenu({
case _MemberActions.setPowerLevel:
final power = await showPermissionChooser(
context,
currentLevel: user.powerLevel,
maxLevel: user.room.ownPowerLevel,
currentLevel: user.powerLevel.level,
maxLevel: user.room.ownPowerLevel.level,
);
if (power == null) return;
if (!context.mounted) return;
@ -245,12 +247,13 @@ Future<void> showMemberActionsPopupMenu({
case _MemberActions.kick:
if (await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).no,
message: L10n.of(context).kickUserDescription,
title: l10n.areYouSure,
okLabel: l10n.yes,
cancelLabel: l10n.no,
message: l10n.kickUserDescription,
) ==
OkCancelResult.ok) {
if (!context.mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => user.kick(),
@ -260,12 +263,13 @@ Future<void> showMemberActionsPopupMenu({
case _MemberActions.ban:
if (await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).no,
message: L10n.of(context).banUserDescription,
title: l10n.areYouSure,
okLabel: l10n.yes,
cancelLabel: l10n.no,
message: l10n.banUserDescription,
) ==
OkCancelResult.ok) {
if (!context.mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => user.ban(),
@ -275,20 +279,22 @@ Future<void> showMemberActionsPopupMenu({
case _MemberActions.report:
final reason = await showTextInputDialog(
context: context,
title: L10n.of(context).whyDoYouWantToReportThis,
okLabel: L10n.of(context).report,
cancelLabel: L10n.of(context).cancel,
hintText: L10n.of(context).reason,
title: l10n.whyDoYouWantToReportThis,
okLabel: l10n.report,
cancelLabel: l10n.cancel,
hintText: l10n.reason,
);
if (reason == null || reason.isEmpty) return;
if (!context.mounted) return;
final result = await showFutureLoadingDialog(
context: context,
future: () => user.room.client.reportUser(user.id, reason),
);
if (result.error != null) return;
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(L10n.of(context).contentHasBeenReported)),
if (!context.mounted) return;
scaffoldMessenger.showSnackBar(
SnackBar(content: Text(l10n.contentHasBeenReported)),
);
return;
case _MemberActions.info:
@ -304,19 +310,20 @@ Future<void> showMemberActionsPopupMenu({
case _MemberActions.unban:
if (await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,
okLabel: L10n.of(context).yes,
cancelLabel: L10n.of(context).no,
message: L10n.of(context).unbanUserDescription,
title: l10n.areYouSure,
okLabel: l10n.yes,
cancelLabel: l10n.no,
message: l10n.unbanUserDescription,
) ==
OkCancelResult.ok) {
if (!context.mounted) return;
await showFutureLoadingDialog(
context: context,
future: () => user.unban(),
);
}
case _MemberActions.makeAdmin:
if (user.room.ownPowerLevel <= 100) {
if (user.room.ownPowerLevel.level <= 100) {
final consent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context).areYouSure,

View file

@ -54,11 +54,11 @@ class SpacesNavigationRail extends StatelessWidget {
isSelected: activeSpaceId == null,
onTap: onGoToChats,
icon: const Padding(
padding: EdgeInsets.all(10.0),
padding: EdgeInsets.all(8.0),
child: Icon(Icons.forum_outlined),
),
selectedIcon: const Padding(
padding: EdgeInsets.all(10.0),
padding: EdgeInsets.all(8.0),
child: Icon(Icons.forum),
),
toolTip: L10n.of(context).chats,
@ -71,7 +71,7 @@ class SpacesNavigationRail extends StatelessWidget {
isSelected: false,
onTap: () => context.go('/rooms/newspace'),
icon: const Padding(
padding: EdgeInsets.all(8.0),
padding: EdgeInsets.all(6.0),
child: Icon(Icons.add),
),
toolTip: L10n.of(context).createNewSpace,
@ -94,6 +94,7 @@ class SpacesNavigationRail extends StatelessWidget {
icon: Avatar(
mxContent: allSpaces[i].avatar,
name: displayname,
size: 36,
shapeBorder: RoundedSuperellipseBorder(
side: BorderSide(
width: 1,

View file

@ -5,26 +5,26 @@ packages:
dependency: transitive
description:
name: _fe_analyzer_shared
sha256: c209688d9f5a5f26b2fb47a188131a6fb9e876ae9e47af3737c0b4f58a93470d
sha256: "8d7ff3948166b8ec5da0fbb5962000926b8e02f2ed9b3e51d1738905fbd4c98d"
url: "https://pub.dev"
source: hosted
version: "91.0.0"
version: "93.0.0"
analyzer:
dependency: transitive
description:
name: analyzer
sha256: f51c8499b35f9b26820cfe914828a6a98a94efd5cc78b37bb7d03debae3a1d08
sha256: de7148ed2fcec579b19f122c1800933dfa028f6d9fd38a152b04b1516cec120b
url: "https://pub.dev"
source: hosted
version: "8.4.1"
version: "10.0.1"
analyzer_plugin:
dependency: transitive
description:
name: analyzer_plugin
sha256: "825071d553c4aef2252196d46a665fbd8e0cb06de07725f25d1b29bd18d65fff"
sha256: "7df504f0c9d6891bacc9f73a5a8c5f6fe4fc49c90ec8e3379916372906ba0b32"
url: "https://pub.dev"
source: hosted
version: "0.13.6"
version: "0.14.1"
ansicolor:
dependency: transitive
description:
@ -53,10 +53,10 @@ packages:
dependency: "direct main"
description:
name: async
sha256: "758e6d74e971c3e5aceb4110bfd6698efc7f501675bcfe0c775459a8140750eb"
sha256: e2eb0491ba5ddb6177742d2da23904574082139b07c1e33b8503b9f46f3e1a37
url: "https://pub.dev"
source: hosted
version: "2.13.0"
version: "2.13.1"
audio_session:
dependency: transitive
description:
@ -69,10 +69,10 @@ packages:
dependency: "direct main"
description:
name: badges
sha256: a7b6bbd60dce418df0db3058b53f9d083c22cdb5132a052145dc267494df0b84
sha256: cf1c88fb3777df69ccd630b80de5267f54efa4a39381b0404a7c03d56cb7c041
url: "https://pub.dev"
source: hosted
version: "3.1.2"
version: "3.2.0"
barbecue:
dependency: transitive
description:
@ -149,10 +149,10 @@ packages:
dependency: "direct main"
description:
name: chewie
sha256: "44bcfc5f0dfd1de290c87c9d86a61308b3282a70b63435d5557cfd60f54a69ca"
sha256: "53dadd2c5b6748742d7744072b38a417ad22691ca55715850300ee793dc7cb27"
url: "https://pub.dev"
source: hosted
version: "1.13.0"
version: "1.13.1"
cli_config:
dependency: transitive
description:
@ -261,10 +261,10 @@ packages:
dependency: "direct dev"
description:
name: dart_code_linter
sha256: "1b53722d9933a5f5d4580acc29c7f16b1fde66d21d1ecf7bb2a811caf3a42b42"
sha256: "8ece88f710621ca1c40b6c344b316d78bb2269d728d37d2a44f19a81d9d2cb93"
url: "https://pub.dev"
source: hosted
version: "3.2.1"
version: "4.0.2"
dart_earcut:
dependency: transitive
description:
@ -285,18 +285,18 @@ packages:
dependency: transitive
description:
name: dart_style
sha256: a9c30492da18ff84efe2422ba2d319a89942d93e58eb0b73d32abe822ef54b7b
sha256: "29f7ecc274a86d32920b1d9cfc7502fa87220da41ec60b55f329559d5732e2b2"
url: "https://pub.dev"
source: hosted
version: "3.1.3"
version: "3.1.7"
dart_webrtc:
dependency: transitive
description:
name: dart_webrtc
sha256: "4ed7b9fa9924e5a81eb39271e2c2356739dd1039d60a13b86ba6c5f448625086"
sha256: f6d615bddea5e458ce180a914f3055c234ffb52fb7397a51b3491e76d6d7edb2
url: "https://pub.dev"
source: hosted
version: "1.7.0"
version: "1.8.1"
dbus:
dependency: transitive
description:
@ -309,10 +309,10 @@ packages:
dependency: "direct main"
description:
name: desktop_drop
sha256: e70b46b2d61f1af7a81a40d1f79b43c28a879e30a4ef31e87e9c27bea4d784e8
sha256: aa1e797255bfbc76f9eb5aa4f61e5b68dbf69962ab1be6495816d2f251bc0d1f
url: "https://pub.dev"
source: hosted
version: "0.7.0"
version: "0.7.1"
desktop_notifications:
dependency: "direct main"
description:
@ -389,10 +389,10 @@ packages:
dependency: "direct main"
description:
name: file_picker
sha256: "57d9a1dd5063f85fa3107fb42d1faffda52fdc948cefd5fe5ea85267a5fc7343"
sha256: f13a03000d942e476bc1ff0a736d2e9de711d2f89a95cd4c1d88f861c3348387
url: "https://pub.dev"
source: hosted
version: "10.3.10"
version: "11.0.2"
file_selector:
dependency: "direct main"
description:
@ -479,10 +479,10 @@ packages:
dependency: "direct main"
description:
name: flutter_foreground_task
sha256: "1903697944a31f596622e51a6af55e3a9dfb27762f9763ab2841184098c6b0ba"
sha256: fc5c01a5e1b8f7bb51d0c737714f0c50440dbdf1aeddc5f8cbba313aa6fd4856
url: "https://pub.dev"
source: hosted
version: "9.2.1"
version: "9.2.2"
flutter_linkify:
dependency: "direct main"
description:
@ -540,10 +540,10 @@ packages:
dependency: "direct main"
description:
name: flutter_map
sha256: "391e7dc95cc3f5190748210a69d4cfeb5d8f84dcdfa9c3235d0a9d7742ccb3f8"
sha256: "03b71c02806ff20c3718d108cbbb3638142ebafe368d8ce2dd22a33344bcb02b"
url: "https://pub.dev"
source: hosted
version: "8.2.2"
version: "8.3.0"
flutter_native_splash:
dependency: "direct dev"
description:
@ -649,10 +649,10 @@ packages:
dependency: "direct main"
description:
name: flutter_web_auth_2
sha256: "432ff8c7b2834eaeec3378d99e24a0210b9ac2f453b3f7a7d739a5c09069fba3"
sha256: d354998934ddc338e69b999b2abaeb33c6fd09999d3a5f92ead1a6b49b49712e
url: "https://pub.dev"
source: hosted
version: "5.0.1"
version: "5.0.2"
flutter_web_auth_2_platform_interface:
dependency: transitive
description:
@ -670,10 +670,10 @@ packages:
dependency: "direct main"
description:
name: flutter_webrtc
sha256: c549ea8ffb20167110ad0a28e5f17a2650b5bea8837d984898cd9b0ffd5fa78b
sha256: c7b0a67ca2c878575fc5c146d801cd874f58f5f1ef5fa6e8eb0c93d413beb948
url: "https://pub.dev"
source: hosted
version: "1.3.1"
version: "1.4.1"
frontend_server_client:
dependency: transitive
description:
@ -763,10 +763,10 @@ packages:
dependency: "direct main"
description:
name: go_router
sha256: "7974313e217a7771557add6ff2238acb63f635317c35fa590d348fb238f00896"
sha256: "08b742eef4f71c9df5af543751cd0b7f1c679c4088488f4223ecaddc1a813b79"
url: "https://pub.dev"
source: hosted
version: "17.1.0"
version: "17.2.2"
gsettings:
dependency: transitive
description:
@ -1088,10 +1088,10 @@ packages:
dependency: "direct main"
description:
name: matrix
sha256: "5bb38e98212bc4c3244c762a1af787f7239a38d2cfdf44488258283ff899f77c"
sha256: "0da5f65016c704bda81eae807cdadc18a046a444fa7e5cec83e60e5d04006d17"
url: "https://pub.dev"
source: hosted
version: "6.2.0"
version: "7.0.0"
media_kit:
dependency: transitive
description:
@ -1208,10 +1208,10 @@ packages:
dependency: "direct main"
description:
name: package_info_plus
sha256: f69da0d3189a4b4ceaeb1a3defb0f329b3b352517f52bed4290f83d4f06bc08d
sha256: "468c26b4254ab01979fa5e4a98cb343ea3631b9acee6f21028997419a80e1a20"
url: "https://pub.dev"
source: hosted
version: "9.0.0"
version: "9.0.1"
package_info_plus_platform_interface:
dependency: transitive
description:
@ -1600,10 +1600,10 @@ packages:
dependency: "direct main"
description:
name: share_plus
sha256: "14c8860d4de93d3a7e53af51bff479598c4e999605290756bbbe45cf65b37840"
sha256: "223873d106614442ea6f20db5a038685cc5b32a2fba81cdecaefbbae0523f7fa"
url: "https://pub.dev"
source: hosted
version: "12.0.1"
version: "12.0.2"
share_plus_platform_interface:
dependency: transitive
description:
@ -1616,10 +1616,10 @@ packages:
dependency: "direct main"
description:
name: shared_preferences
sha256: "2939ae520c9024cb197fc20dee269cd8cdbf564c8b5746374ec6cacdc5169e64"
sha256: c3025c5534b01739267eb7d76959bbc25a6d10f6988e1c2a3036940133dd10bf
url: "https://pub.dev"
source: hosted
version: "2.5.4"
version: "2.5.5"
shared_preferences_android:
dependency: transitive
description:
@ -2133,10 +2133,10 @@ packages:
dependency: "direct main"
description:
name: wakelock_plus
sha256: e4e125b7c1a2f0e491e5452afdc0e25ab77b2d2775a7caa231fcc1c1f2162c47
sha256: ddf3db70eaa10c37558ff817519b85d527dbd21034fd5d8e1c2e85f31588f1c1
url: "https://pub.dev"
source: hosted
version: "1.5.0"
version: "1.5.2"
wakelock_plus_platform_interface:
dependency: transitive
description:
@ -2214,10 +2214,10 @@ packages:
dependency: "direct main"
description:
name: webrtc_interface
sha256: ad0e5786b2acd3be72a3219ef1dde9e1cac071cf4604c685f11b61d63cdd6eb3
sha256: c6f100eac5057d9a817a60473126f9828c796d42884d498af4f339c97b21014f
url: "https://pub.dev"
source: hosted
version: "1.4.0"
version: "1.5.1"
win32:
dependency: transitive
description:

View file

@ -4,43 +4,43 @@ publish_to: none
# On version bump please also increase:
# 1. The build number (for F-Droid)
# 2. The version in /snap/snapcraft.yaml
version: 2.5.0+3550
version: 2.5.1+3551
environment:
sdk: ">=3.11.1 <4.0.0"
dependencies:
archive: ^4.0.7
async: ^2.11.0
badges: ^3.1.2
async: ^2.13.1
badges: ^3.2.0
blurhash_dart: ^1.2.1
chewie: ^1.13.0
chewie: ^1.13.1
collection: ^1.18.0
crop_image: ^1.0.17
cross_file: ^0.3.5
desktop_drop: ^0.7.0
desktop_drop: ^0.7.1
desktop_notifications: ^0.6.3
device_info_plus: ^12.3.0
dynamic_color: ^1.8.1
emoji_picker_flutter: ^4.4.0
file_picker: ^10.3.10
file_picker: ^11.0.2
file_selector: ^1.1.0
flutter:
sdk: flutter
flutter_foreground_task: ^9.2.1
flutter_foreground_task: ^9.2.2
flutter_linkify: ^6.0.0
flutter_local_notifications: ^21.0.0
flutter_localizations:
sdk: flutter
flutter_map: ^8.2.2
flutter_map: ^8.3.0
flutter_new_badger: ^1.1.1
flutter_secure_storage: ^10.0.0
flutter_shortcuts_new: ^2.0.0
flutter_vodozemac: ^0.5.0
flutter_web_auth_2: ^5.0.1
flutter_webrtc: ^1.3.1
flutter_web_auth_2: ^5.0.2
flutter_webrtc: ^1.4.1
geolocator: ^14.0.2
go_router: ^17.1.0
go_router: ^17.2.2
handy_window: ^0.4.2
highlight: ^0.7.0
html: ^0.15.4
@ -52,13 +52,13 @@ dependencies:
just_audio_media_kit: ^2.1.0
latlong2: ^0.9.1
linkify: ^5.0.0
matrix: ^6.2.0
matrix: ^7.0.0
media_kit_libs_linux: ^1.2.1
media_kit_libs_windows_video: ^1.0.11
mime: ^2.0.0
native_imaging: ^0.4.0
opus_caf_converter_dart: ^1.0.1
package_info_plus: ^9.0.0
package_info_plus: ^9.0.1
particles_network: ^1.9.3
pasteboard: ^0.5.0
path: ^1.9.0
@ -69,10 +69,10 @@ dependencies:
qr_code_scanner_plus: ^2.1.1
qr_image: ^1.0.0
receive_sharing_intent: ^1.8.1
record: ^6.1.2
record: ^6.2.0
scroll_to_index: ^3.0.1
share_plus: ^12.0.1
shared_preferences: ^2.5.4 # Pinned because https://github.com/flutter/flutter/issues/118401
share_plus: ^12.0.2
shared_preferences: ^2.5.5 # Pinned because https://github.com/flutter/flutter/issues/118401
slugify: ^2.0.0
sqflite_common_ffi: ^2.3.7+1
sqlcipher_flutter_libs: ^0.6.8
@ -84,11 +84,11 @@ dependencies:
video_compress: ^3.1.4
video_player: ^2.11.1
video_player_media_kit: ^2.0.0
wakelock_plus: ^1.5.0
webrtc_interface: ^1.3.0
wakelock_plus: ^1.5.2
webrtc_interface: ^1.5.1
dev_dependencies:
dart_code_linter: ^3.2.1
dart_code_linter: ^4.0.2
flutter_lints: ^6.0.0
flutter_native_splash: ^2.4.7
flutter_test:

View file

@ -1,6 +1,6 @@
#!/usr/bin/env bash
flutter pub add fcm_shared_isolate:0.2.0
flutter pub add fcm_shared_isolate
flutter pub get
if [[ "$OSTYPE" == "darwin"* ]]; then

View file

@ -1,8 +1,8 @@
#!/bin/sh -ve
# Compile Vodozemac for web
version=$(yq ".dependencies.flutter_vodozemac" < pubspec.yaml | tr -d '"')
version=$(expr "$version" : '\^*\(.*\)')
version=$(yq ".dependencies.flutter_vodozemac" < pubspec.yaml)
version=$(printf "%s" "$version" | tr -d '"^')
git clone https://github.com/famedly/dart-vodozemac.git -b ${version} .vodozemac
cd .vodozemac
cargo install flutter_rust_bridge_codegen
@ -15,8 +15,8 @@ flutter pub get
dart compile js ./web/native_executor.dart -o ./web/native_executor.js -m
# Download native_imaging for web:
version=$(yq ".dependencies.native_imaging" < pubspec.yaml | tr -d '"')
version=$(expr "$version" : '\^*\(.*\)')
version=$(yq ".dependencies.native_imaging" < pubspec.yaml)
version=$(printf "%s" "$version" | tr -d '"^')
curl -L "https://github.com/famedly/dart_native_imaging/releases/download/v${version}/native_imaging.zip" > native_imaging.zip
unzip native_imaging.zip
mv js/* web/

View file

@ -1,5 +1,5 @@
#!/bin/sh -ve
flutter pub add fcm_shared_isolate:0.2.0
flutter pub add fcm_shared_isolate
sed -i '' 's,//<GOOGLE_SERVICES>,,g' lib/utils/background_push.dart
flutter clean
flutter pub get

View file

@ -1,7 +1,7 @@
name: fluffychat
title: FluffyChat
base: core24
version: 2.5.0
version: 2.5.1
license: AGPL-3.0
summary: The cutest messenger in the Matrix network
description: |
@ -53,7 +53,7 @@ platforms:
parts:
flutter-git:
source: https://github.com/flutter/flutter.git
source-tag: 3.41.5
source-tag: 3.41.7
source-depth: 1
plugin: nil
override-build: |