From 460601e9a0c4faa3072be6e7f9402d1f1ac25dd0 Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Mon, 1 Mar 2021 11:29:41 +0300 Subject: [PATCH 01/18] Removed channel last_activity display if it's 0 or null --- lib/utils/dateformatter.dart | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/utils/dateformatter.dart b/lib/utils/dateformatter.dart index bb533ac5..f38907aa 100644 --- a/lib/utils/dateformatter.dart +++ b/lib/utils/dateformatter.dart @@ -4,7 +4,7 @@ class DateFormatter { /// Function to get a verbose representation of timestamp, /// like [just now] or [today] static String getVerboseDateTime(int timestamp) { - if (timestamp == null) return ''; + if (timestamp == null || timestamp == 0) return ''; final dateTime = DateTime.fromMillisecondsSinceEpoch(timestamp); final localDT = dateTime.toLocal(); // if timestamp is less than a minute old return 'Now' From c22bfe03f2160ed4b36914a246609a0da37638f9 Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Mon, 1 Mar 2021 14:28:39 +0300 Subject: [PATCH 02/18] Added new feature: username/password saving after unsuccessful auth attempt --- lib/blocs/auth_bloc/auth_bloc.dart | 10 ++++++++-- lib/blocs/auth_bloc/auth_state.dart | 14 +++++++++----- lib/blocs/messages_bloc/messages_bloc.dart | 8 ++++---- lib/widgets/auth/auth_form.dart | 7 ++++++- 4 files changed, 27 insertions(+), 12 deletions(-) diff --git a/lib/blocs/auth_bloc/auth_bloc.dart b/lib/blocs/auth_bloc/auth_bloc.dart index 2a52b295..677dd480 100644 --- a/lib/blocs/auth_bloc/auth_bloc.dart +++ b/lib/blocs/auth_bloc/auth_bloc.dart @@ -123,10 +123,16 @@ class AuthBloc extends Bloc { ); switch (res) { case AuthResult.WrongCredentials: - yield WrongCredentials(); + yield WrongCredentials( + username: event.username, + password: event.password, + ); break; case AuthResult.NetworkError: - yield AuthenticationError(); + yield AuthenticationError( + username: event.username, + password: event.password, + ); break; default: final InitData initData = await initMain(); diff --git a/lib/blocs/auth_bloc/auth_state.dart b/lib/blocs/auth_bloc/auth_state.dart index cf1253cb..96e30707 100644 --- a/lib/blocs/auth_bloc/auth_state.dart +++ b/lib/blocs/auth_bloc/auth_state.dart @@ -14,15 +14,18 @@ class AuthInitializing extends AuthState { class Unauthenticated extends AuthState { final String message; + final String username; + final String password; - const Unauthenticated({this.message}); + const Unauthenticated({this.message, this.username, this.password}); @override - List get props => [message]; + List get props => [username, password]; } class WrongCredentials extends Unauthenticated { - const WrongCredentials(); + const WrongCredentials({String username, String password}) + : super(username: username, password: password); @override List get props => []; @@ -62,8 +65,9 @@ class PasswordReset extends AuthState { List get props => [link]; } -class AuthenticationError extends AuthState { - const AuthenticationError(); +class AuthenticationError extends Unauthenticated { + const AuthenticationError({String username, String password}) + : super(username: username, password: password); @override List get props => []; diff --git a/lib/blocs/messages_bloc/messages_bloc.dart b/lib/blocs/messages_bloc/messages_bloc.dart index 4b59f849..75a0d7f1 100644 --- a/lib/blocs/messages_bloc/messages_bloc.dart +++ b/lib/blocs/messages_bloc/messages_bloc.dart @@ -206,7 +206,7 @@ class MessagesBloc parentChannel: selectedChannel, ); yield newState; - _updateParentChannel(); + _updateParentChannel(event.channelId); } } else if (event is ModifyResponsesCount) { var thread = await repository.updateResponsesCount(event.threadId); @@ -287,7 +287,7 @@ class MessagesBloc message.lastName = ProfileBloc.lastName; this.repository.items.add(message); this.add(FinishLoadingMessages()); - _updateParentChannel(); + // _updateParentChannel(); }, ); this.repository.items.add(tempItem); @@ -342,10 +342,10 @@ class MessagesBloc return map; } - void _updateParentChannel() { + void _updateParentChannel([String channelId]) { channelsBloc.add(ModifyMessageCount( workspaceId: ProfileBloc.selectedWorkspace, - channelId: selectedChannel.id, + channelId: channelId ?? selectedChannel.id, companyId: ProfileBloc.selectedCompany, totalModifier: 1, )); diff --git a/lib/widgets/auth/auth_form.dart b/lib/widgets/auth/auth_form.dart index d607f42b..9a179567 100644 --- a/lib/widgets/auth/auth_form.dart +++ b/lib/widgets/auth/auth_form.dart @@ -4,7 +4,7 @@ import 'package:twake/blocs/auth_bloc/auth_bloc.dart'; import 'package:twake/blocs/connection_bloc/connection_bloc.dart' as cb; import 'package:twake/config/styles_config.dart'; import 'package:twake/config/dimensions_config.dart' show Dim; -import 'package:twake/utils/navigation.dart'; +// import 'package:twake/utils/navigation.dart'; class AuthForm extends StatefulWidget { final Function onConfigurationOpen; @@ -44,6 +44,11 @@ class _AuthFormState extends State { super.initState(); _usernameController.addListener(onUsernameSaved); _passwordController.addListener(onPasswordSaved); + final state = BlocProvider.of(context).state; + if (state is Unauthenticated) { + _usernameController.text = state.username; + _passwordController.text = state.username; + } } @override From b4760b5e96e4cf1c57fa24fdf9561d38c220f14a Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Mon, 1 Mar 2021 14:54:22 +0300 Subject: [PATCH 03/18] Direct channels highlights --- lib/widgets/channel/direct_tile.dart | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/lib/widgets/channel/direct_tile.dart b/lib/widgets/channel/direct_tile.dart index 85a8e87a..7565a907 100644 --- a/lib/widgets/channel/direct_tile.dart +++ b/lib/widgets/channel/direct_tile.dart @@ -1,11 +1,11 @@ import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; -import 'package:twake/blocs/channels_bloc/channels_bloc.dart'; -import 'package:twake/blocs/directs_bloc/directs_bloc.dart'; +// import 'package:twake/blocs/channels_bloc/channels_bloc.dart'; +// import 'package:twake/blocs/directs_bloc/directs_bloc.dart'; import 'package:twake/blocs/draft_bloc/draft_bloc.dart'; import 'package:twake/config/dimensions_config.dart' show Dim; import 'package:twake/models/direct.dart'; -import 'package:twake/pages/messages_page.dart'; +// import 'package:twake/pages/messages_page.dart'; import 'package:twake/repositories/draft_repository.dart'; import 'package:twake/utils/dateformatter.dart'; import 'package:twake/utils/navigation.dart'; @@ -20,7 +20,6 @@ class DirectTile extends StatelessWidget { Widget build(BuildContext context) { return InkWell( onTap: () { - var draftType = DraftType.direct; final id = direct.id; // Load draft from local storage @@ -47,7 +46,9 @@ class DirectTile extends StatelessWidget { textAlign: TextAlign.start, style: TextStyle( fontSize: 16.0, - fontWeight: FontWeight.w600, + fontWeight: direct.hasUnread == 1 + ? FontWeight.w900 + : FontWeight.w400, color: Color(0xff444444), ), ), From 8b723436fc81c5f52b03bc1cb04b3b43b74c3a0e Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Tue, 2 Mar 2021 12:45:34 +0300 Subject: [PATCH 04/18] Added new local notifications handler --- .../notification_bloc/notification_bloc.dart | 2 +- lib/pages/server_configuration.dart | 6 +- lib/services/notifications.dart | 57 ++++++++++++++++++- pubspec.yaml | 2 + 4 files changed, 61 insertions(+), 6 deletions(-) diff --git a/lib/blocs/notification_bloc/notification_bloc.dart b/lib/blocs/notification_bloc/notification_bloc.dart index fdae13ab..52505ea6 100644 --- a/lib/blocs/notification_bloc/notification_bloc.dart +++ b/lib/blocs/notification_bloc/notification_bloc.dart @@ -95,8 +95,8 @@ class NotificationBloc extends Bloc { socketConnectionState = SocketConnectionState.CONNECTED; while (socketConnectionState != SocketConnectionState.AUTHENTICATED) { if (socket.disconnected) socket = socket.connect(); - await Future.delayed(Duration(seconds: 2)); socket.emit(SocketIOEvent.AUTHENTICATE, {'token': this.token}); + await Future.delayed(Duration(seconds: 5)); print('WAITING FOR SOCKET AUTH'); } }); diff --git a/lib/pages/server_configuration.dart b/lib/pages/server_configuration.dart index a7a7cd2e..09004b68 100644 --- a/lib/pages/server_configuration.dart +++ b/lib/pages/server_configuration.dart @@ -4,7 +4,7 @@ import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:twake/blocs/auth_bloc/auth_bloc.dart'; import 'package:twake/blocs/configuration_cubit/configuration_cubit.dart'; import 'package:twake/blocs/configuration_cubit/configuration_state.dart'; -import 'package:twake/services/api.dart'; +// import 'package:twake/services/api.dart'; import 'package:twake/utils/extensions.dart'; class ServerConfiguration extends StatefulWidget { @@ -107,7 +107,9 @@ class _ServerConfigurationState extends State { current is ConfigurationError, listener: (context, state) { if (state is ConfigurationSaved) { - context.read().add(ValidateHost(_controller.text)); + context + .read() + .add(ValidateHost(_controller.text)); } else if (state is ConfigurationError) { Scaffold.of(context).showSnackBar(SnackBar( content: Text( diff --git a/lib/services/notifications.dart b/lib/services/notifications.dart index 48fdc751..30832fd6 100644 --- a/lib/services/notifications.dart +++ b/lib/services/notifications.dart @@ -5,6 +5,7 @@ import 'package:firebase_messaging/firebase_messaging.dart'; // import 'package:twake/blocs/profile_bloc/profile_bloc.dart'; import 'package:twake/models/notification.dart'; import 'package:twake/services/service_bundle.dart'; +import 'package:flutter_local_notifications/flutter_local_notifications.dart'; class Notifications { final logger = Logger(); @@ -13,7 +14,35 @@ class Notifications { final Function(MessageNotification) onResumeCallback; final Function(MessageNotification) onLaunchCallback; FirebaseMessaging _fcm = FirebaseMessaging(); + FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = + FlutterLocalNotificationsPlugin(); // final _api = Api(); +// Future onDidReceiveLocalNotification( + // int id, String title, String body, String payload) async { + // // display a dialog with the notification details, tap ok to go to another page + // showDialog( + // context: context, + // builder: (BuildContext context) => CupertinoAlertDialog( + // title: Text(title), + // content: Text(body), + // actions: [ + // CupertinoDialogAction( + // isDefaultAction: true, + // child: Text('Ok'), + // onPressed: () async { + // Navigator.of(context, rootNavigator: true).pop(); + // await Navigator.push( + // context, + // MaterialPageRoute( + // builder: (context) => SecondScreen(payload), + // ), + // ); + // }, + // ) + // ], + // ), + // ); +// } Notifications({ this.onMessageCallback, @@ -34,6 +63,18 @@ class Notifications { onResume: onResume, onLaunch: onLaunch, ); + + const AndroidInitializationSettings initializationSettingsAndroid = + AndroidInitializationSettings('logo'); + final IOSInitializationSettings initializationSettingsIOS = + IOSInitializationSettings(); + final InitializationSettings initializationSettings = + InitializationSettings( + android: initializationSettingsAndroid, + iOS: initializationSettingsIOS, + ); + flutterLocalNotificationsPlugin.initialize(initializationSettings, + onSelectNotification: (paload) async {}); } // Future checkWhatsNew(String workspaceId) async { @@ -50,9 +91,19 @@ class Notifications { // } Future onMessage(Map message) async { - // logger.d('GOT MESSAGE FROM FIREBASE: $message'); - // final notification = messageParse(message); - // onMessageCallback(notification); + logger.d('GOT MESSAGE FROM FIREBASE: $message'); + final notification = messageParse(message); + const AndroidNotificationDetails androidPlatformChannelSpecifics = + AndroidNotificationDetails( + 'your channel id', 'your channel name', 'your channel description', + importance: Importance.max, + priority: Priority.high, + showWhen: false); + const NotificationDetails platformChannelSpecifics = + NotificationDetails(android: androidPlatformChannelSpecifics); + await flutterLocalNotificationsPlugin.show( + 0, 'plain title', 'plain body', platformChannelSpecifics, + payload: 'item x'); } NotificationData messageParse(Map message) { diff --git a/pubspec.yaml b/pubspec.yaml index adf44b85..1806b79f 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -56,6 +56,8 @@ dependencies: flutter_markdown: git: https://github.com/bobs4462/flutter_markdown.git + flutter_local_notifications: ^4.0.1+1 + # The following adds the Cupertino Icons font to your application. From e486c466d5073085945738e21f285d1786764d0b Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Tue, 2 Mar 2021 16:16:37 +0300 Subject: [PATCH 05/18] Local notifications handling on Android is working now --- android/app/proguard-rules.pro | 23 +++++++++++++++ lib/blocs/channels_bloc/channels_bloc.dart | 3 ++ lib/blocs/messages_bloc/messages_bloc.dart | 3 +- .../notification_bloc/notification_bloc.dart | 16 +++++++++-- lib/blocs/profile_bloc/profile_bloc.dart | 11 ++++++++ lib/repositories/profile_repository.dart | 4 +++ lib/services/notifications.dart | 28 +++++++++++++++---- 7 files changed, 79 insertions(+), 9 deletions(-) create mode 100644 android/app/proguard-rules.pro diff --git a/android/app/proguard-rules.pro b/android/app/proguard-rules.pro new file mode 100644 index 00000000..0f96ad8a --- /dev/null +++ b/android/app/proguard-rules.pro @@ -0,0 +1,23 @@ +## Gson rules +# Gson uses generic type information stored in a class file when working with fields. Proguard +# removes such information by default, so configure it to keep all of it. +-keepattributes Signature + +# For using GSON @Expose annotation +-keepattributes *Annotation* + +# Gson specific classes +-dontwarn sun.misc.** +#-keep class com.google.gson.stream.** { *; } + +# Prevent proguard from stripping interface information from TypeAdapter, TypeAdapterFactory, +# JsonSerializer, JsonDeserializer instances (so they can be used in @JsonAdapter) +-keep class * extends com.google.gson.TypeAdapter +-keep class * implements com.google.gson.TypeAdapterFactory +-keep class * implements com.google.gson.JsonSerializer +-keep class * implements com.google.gson.JsonDeserializer + +# Prevent R8 from leaving Data object members always null +-keepclassmembers,allowobfuscation class * { + @com.google.gson.annotations.SerializedName ; +} diff --git a/lib/blocs/channels_bloc/channels_bloc.dart b/lib/blocs/channels_bloc/channels_bloc.dart index b0a62b86..6aa6d4ce 100644 --- a/lib/blocs/channels_bloc/channels_bloc.dart +++ b/lib/blocs/channels_bloc/channels_bloc.dart @@ -2,6 +2,7 @@ import 'dart:async'; import 'package:twake/blocs/base_channel_bloc/base_channel_bloc.dart'; import 'package:twake/blocs/notification_bloc/notification_bloc.dart'; +import 'package:twake/blocs/profile_bloc/profile_bloc.dart'; import 'package:twake/blocs/workspaces_bloc/workspaces_bloc.dart'; import 'package:twake/blocs/channels_bloc/channel_event.dart'; import 'package:twake/models/channel.dart'; @@ -116,6 +117,8 @@ class ChannelsBloc extends BaseChannelBloc { selected: repository.selected, hasUnread: repository.selected.hasUnread, ); + ProfileBloc.selectedChannel = event.channelId; + ProfileBloc.selectedThread = null; yield newState; } else if (event is ModifyMessageCount) { await this.updateMessageCount(event); diff --git a/lib/blocs/messages_bloc/messages_bloc.dart b/lib/blocs/messages_bloc/messages_bloc.dart index 75a0d7f1..b4ca7986 100644 --- a/lib/blocs/messages_bloc/messages_bloc.dart +++ b/lib/blocs/messages_bloc/messages_bloc.dart @@ -307,7 +307,8 @@ class MessagesBloc parentChannel: selectedChannel, ); } else if (event is SelectMessage) { - print('$T MESSAGE SELECTED'); + // print('$T MESSAGE SELECTED'); + ProfileBloc.selectedThread = event.messageId; repository.select(event.messageId); yield MessageSelected( threadMessage: repository.selected, diff --git a/lib/blocs/notification_bloc/notification_bloc.dart b/lib/blocs/notification_bloc/notification_bloc.dart index 52505ea6..294fe25a 100644 --- a/lib/blocs/notification_bloc/notification_bloc.dart +++ b/lib/blocs/notification_bloc/notification_bloc.dart @@ -52,6 +52,7 @@ class NotificationBloc extends Bloc { onMessageCallback: onMessageCallback, onResumeCallback: onResumeCallback, onLaunchCallback: onLaunchCallback, + shouldNotify: shouldNotify, ); print('TOKEN: $token\nHOST: $socketIOHost'); socket = IO.io( @@ -192,8 +193,14 @@ class NotificationBloc extends Bloc { } } + bool shouldNotify(MessageNotification data) { + if (data.channelId == ProfileBloc.selectedChannel && + (ProfileBloc.selectedThread == data.threadId || + ProfileBloc.selectedThread == null)) return false; + return true; + } + void onMessageCallback(NotificationData data) { - logger.d('ON message callback: ${data is MessageNotification}'); if (data is MessageNotification) { if (data.threadId.isNotEmpty && data.threadId != data.messageId) { logger.d('adding ThreadMessageEvent'); @@ -206,6 +213,7 @@ class NotificationBloc extends Bloc { this.add(ChannelMessageEvent(data)); } } + navigate(data); // } else if (data is WhatsNewItem) { // if (data.workspaceId == null) { // this.add(UpdateDirectChannel(data)); @@ -215,8 +223,11 @@ class NotificationBloc extends Bloc { // } } - Future onResumeCallback(MessageNotification data) async { + onResumeCallback(MessageNotification data) { onMessageCallback(data); + } + + void navigate(MessageNotification data) { navigator.currentState.popUntil( ModalRoute.withName('/'), ); // navigator.popAndPushNamed( @@ -246,7 +257,6 @@ class NotificationBloc extends Bloc { ), ); } - // logger.w('ON RESUME HERE IS the notification\n$data'); } void onLaunchCallback(NotificationData data) { diff --git a/lib/blocs/profile_bloc/profile_bloc.dart b/lib/blocs/profile_bloc/profile_bloc.dart index 1cf15517..d76748e1 100644 --- a/lib/blocs/profile_bloc/profile_bloc.dart +++ b/lib/blocs/profile_bloc/profile_bloc.dart @@ -32,6 +32,9 @@ class ProfileBloc extends Bloc { static String get selectedCompany => repository.selectedCompanyId; static String get selectedWorkspace => repository.selectedWorkspaceId; + static String get selectedChannel => repository.selectedChannelId; + static String get selectedThread => repository.selectedThreadId; + static set selectedCompany(String val) { repository.selectedCompanyId = val; repository.save(); @@ -42,6 +45,14 @@ class ProfileBloc extends Bloc { repository.save(); } + static set selectedChannel(String val) { + repository.selectedChannelId = val; + } + + static set selectedThread(String val) { + repository.selectedThreadId = val; + } + @override Stream mapEventToState(ProfileEvent event) async* { if (event is ReloadProfile) { diff --git a/lib/repositories/profile_repository.dart b/lib/repositories/profile_repository.dart index 486ce931..5564a79c 100644 --- a/lib/repositories/profile_repository.dart +++ b/lib/repositories/profile_repository.dart @@ -40,6 +40,10 @@ class ProfileRepository extends JsonSerializable { String selectedCompanyId; @JsonKey(name: 'selected_workspace_id') String selectedWorkspaceId; + @JsonKey(ignore: true) + String selectedChannelId; + @JsonKey(ignore: true) + String selectedThreadId; // Pseudo constructor for loading profile from storage or api static Future load() async { diff --git a/lib/services/notifications.dart b/lib/services/notifications.dart index 30832fd6..059e39e2 100644 --- a/lib/services/notifications.dart +++ b/lib/services/notifications.dart @@ -13,6 +13,7 @@ class Notifications { final Function(MessageNotification) onMessageCallback; final Function(MessageNotification) onResumeCallback; final Function(MessageNotification) onLaunchCallback; + final bool Function(MessageNotification) shouldNotify; FirebaseMessaging _fcm = FirebaseMessaging(); FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); @@ -48,6 +49,7 @@ class Notifications { this.onMessageCallback, this.onResumeCallback, this.onLaunchCallback, + this.shouldNotify, }) { if (Platform.isAndroid) this.platform = Target.Android; @@ -74,7 +76,18 @@ class Notifications { iOS: initializationSettingsIOS, ); flutterLocalNotificationsPlugin.initialize(initializationSettings, - onSelectNotification: (paload) async {}); + onSelectNotification: (payload) async { + print("PAYLOAD FROM NOTIFY: $payload"); + final map = jsonDecode(payload); + print("NOTIFY CLICK: $map"); + try { + final notification = MessageNotification.fromJson(map); + print("NOTIFICATION PARSED: $notification"); + onMessageCallback(notification); + } catch (e) { + logger.wtf('ERROR PARSING NOTIFY: $e'); + } + }); } // Future checkWhatsNew(String workspaceId) async { @@ -93,20 +106,25 @@ class Notifications { Future onMessage(Map message) async { logger.d('GOT MESSAGE FROM FIREBASE: $message'); final notification = messageParse(message); + if (!shouldNotify(notification)) return; const AndroidNotificationDetails androidPlatformChannelSpecifics = AndroidNotificationDetails( - 'your channel id', 'your channel name', 'your channel description', + 'some-random-text', 'Twake', 'Twake Mobile App', importance: Importance.max, priority: Priority.high, showWhen: false); const NotificationDetails platformChannelSpecifics = NotificationDetails(android: androidPlatformChannelSpecifics); await flutterLocalNotificationsPlugin.show( - 0, 'plain title', 'plain body', platformChannelSpecifics, - payload: 'item x'); + 0, + message['notification']['title'], + message['notification']['body'], + platformChannelSpecifics, + payload: message['data']['notification_data'], + ); } - NotificationData messageParse(Map message) { + MessageNotification messageParse(Map message) { Map data; switch (platform) { case Target.Android: From 85b208d43131476ae22004de579ac6dd5a312581 Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Tue, 2 Mar 2021 19:27:51 +0300 Subject: [PATCH 06/18] Added gradle autosign + changed iOS notifications handling --- .gitignore | 2 + android/app/build.gradle | 18 +++++- lib/blocs/messages_bloc/messages_bloc.dart | 8 +-- lib/services/notifications.dart | 64 +++++++++++++++++++--- 4 files changed, 76 insertions(+), 16 deletions(-) diff --git a/.gitignore b/.gitignore index d48397dd..87ca5f57 100644 --- a/.gitignore +++ b/.gitignore @@ -42,3 +42,5 @@ app.*.symbols app.*.map.json /ios/Runner/GoogleService-Info.plist /android/app/google-services.json +/android/key.properties +/android/android.jks diff --git a/android/app/build.gradle b/android/app/build.gradle index 85f860d4..ee0aeecc 100644 --- a/android/app/build.gradle +++ b/android/app/build.gradle @@ -25,6 +25,12 @@ apply plugin: 'com.android.application' apply plugin: 'kotlin-android' apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle" +def keystoreProperties = new Properties() +def keystorePropertiesFile = rootProject.file('key.properties') +if (keystorePropertiesFile.exists()) { + keystoreProperties.load(new FileInputStream(keystorePropertiesFile)) +} + android { compileSdkVersion 29 @@ -46,11 +52,17 @@ android { versionName flutterVersionName } + signingConfigs { + release { + keyAlias keystoreProperties['keyAlias'] + keyPassword keystoreProperties['keyPassword'] + storeFile keystoreProperties['storeFile'] ? file(keystoreProperties['storeFile']) : null + storePassword keystoreProperties['storePassword'] + } + } buildTypes { release { - // TODO: Add your own signing config for the release build. - // Signing with the debug keys for now, so `flutter run --release` works. - signingConfig signingConfigs.debug + signingConfig signingConfigs.release } } } diff --git a/lib/blocs/messages_bloc/messages_bloc.dart b/lib/blocs/messages_bloc/messages_bloc.dart index b4ca7986..12084497 100644 --- a/lib/blocs/messages_bloc/messages_bloc.dart +++ b/lib/blocs/messages_bloc/messages_bloc.dart @@ -86,16 +86,16 @@ class MessagesBloc if (T == DirectsBloc && state.data.workspaceId == 'direct') { while (selectedChannel.id != state.data.channelId || this.state is! MessagesLoaded) { - print('Waiting for the correct channel loading\n' - 'COND1: ${selectedChannel.id != state.data.channelId}\n' - 'COND2: ${this.state is! MessagesLoaded}'); + // print('Waiting for the correct channel loading\n' + // 'COND1: ${selectedChannel.id != state.data.channelId}\n' + // 'COND2: ${this.state is! MessagesLoaded}'); await Future.delayed(Duration(milliseconds: 500)); } this.add(SelectMessage(state.data.threadId)); } else if (T == ChannelsBloc && state.data.workspaceId != 'direct') { while (selectedChannel.id != state.data.channelId || this.state is! MessagesLoaded) { - print('Waiting for the correct channel loading'); + // print('Waiting for the correct channel loading'); await Future.delayed(Duration(milliseconds: 500)); } this.add(SelectMessage(state.data.threadId)); diff --git a/lib/services/notifications.dart b/lib/services/notifications.dart index 059e39e2..5a486259 100644 --- a/lib/services/notifications.dart +++ b/lib/services/notifications.dart @@ -117,32 +117,78 @@ class Notifications { NotificationDetails(android: androidPlatformChannelSpecifics); await flutterLocalNotificationsPlugin.show( 0, - message['notification']['title'], - message['notification']['body'], + _getTitle(message), + _getBody(message), platformChannelSpecifics, - payload: message['data']['notification_data'], + payload: _getPayload(message), ); } MessageNotification messageParse(Map message) { - Map data; + // logger.d("ok, that's what we have:\n$data"); + final data = jsonDecode(_getPayload(message)); + MessageNotification notification = MessageNotification.fromJson(data); + return notification; + } + + String _getBody(Map message) { + var data; switch (platform) { case Target.Android: // logger.d('Android notification received\n$message'); - data = jsonDecode(message['data']['notification_data']); + data = message['notification']['body']; break; case Target.IOS: // logger.d('iOS notification received\n$message'); - data = message['data']; + data = message['apps']['alert']['body']; break; case Target.Linux: case Target.MacOS: case Target.Windows: throw 'Desktop is not supported'; } - // logger.d("ok, that's what we have:\n$data"); - MessageNotification notification = MessageNotification.fromJson(data); - return notification; + return data; + } + + String _getTitle(Map message) { + var data; + switch (platform) { + case Target.Android: + // logger.d('Android notification received\n$message'); + data = message['notification']['title']; + break; + case Target.IOS: + // logger.d('iOS notification received\n$message'); + data = message['apps']['alert']['title']; + break; + case Target.Linux: + case Target.MacOS: + case Target.Windows: + throw 'Desktop is not supported'; + } + return data; + } + + String _getPayload(Map message) { + var data; + switch (platform) { + case Target.Android: + // logger.d('Android notification received\n$message'); + data = message['data']['notification_data']; + break; + case Target.IOS: + // logger.d('iOS notification received\n$message'); + data = message['notification_data']; + break; + case Target.Linux: + case Target.MacOS: + case Target.Windows: + throw 'Desktop is not supported'; + } + if (data.runtimeType == Map) { + return jsonEncode(data); + } + return data; } Future onResume(Map message) async { From 64ddc70e3192f8e80d69e174386550cccf661665 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 14:28:07 +0300 Subject: [PATCH 07/18] Channels list layout fix. --- ios/Podfile.lock | 6 ++++++ lib/pages/edit_channel.dart | 2 +- lib/widgets/channel/channel_tile.dart | 23 +++++++++++++---------- 3 files changed, 20 insertions(+), 11 deletions(-) diff --git a/ios/Podfile.lock b/ios/Podfile.lock index 0131e946..38bbeea7 100644 --- a/ios/Podfile.lock +++ b/ios/Podfile.lock @@ -47,6 +47,8 @@ PODS: - Flutter (1.0.0) - flutter_inappwebview (0.0.1): - Flutter + - flutter_local_notifications (0.0.1): + - Flutter - FMDB (2.7.5): - FMDB/standard (= 2.7.5) - FMDB/standard (2.7.5) @@ -104,6 +106,7 @@ DEPENDENCIES: - firebase_messaging (from `.symlinks/plugins/firebase_messaging/ios`) - Flutter (from `Flutter`) - flutter_inappwebview (from `.symlinks/plugins/flutter_inappwebview/ios`) + - flutter_local_notifications (from `.symlinks/plugins/flutter_local_notifications/ios`) - image_picker (from `.symlinks/plugins/image_picker/ios`) - package_info (from `.symlinks/plugins/package_info/ios`) - path_provider (from `.symlinks/plugins/path_provider/ios`) @@ -142,6 +145,8 @@ EXTERNAL SOURCES: :path: Flutter flutter_inappwebview: :path: ".symlinks/plugins/flutter_inappwebview/ios" + flutter_local_notifications: + :path: ".symlinks/plugins/flutter_local_notifications/ios" image_picker: :path: ".symlinks/plugins/image_picker/ios" package_info: @@ -170,6 +175,7 @@ SPEC CHECKSUMS: FirebaseMessaging: 5eca4ef173de76253352511aafef774caa1cba2a Flutter: 0e3d915762c693b495b44d77113d4970485de6ec flutter_inappwebview: 69dfbac46157b336ffbec19ca6dfd4638c7bf189 + flutter_local_notifications: 0c0b1ae97e741e1521e4c1629a459d04b9aec743 FMDB: 2ce00b547f966261cd18927a3ddb07cb6f3db82a GoogleDataTransport: f56af7caa4ed338dc8e138a5d7c5973e66440833 GoogleUtilities: 7f2f5a07f888cdb145101d6042bc4422f57e70b3 diff --git a/lib/pages/edit_channel.dart b/lib/pages/edit_channel.dart index a1f1c074..3d800621 100644 --- a/lib/pages/edit_channel.dart +++ b/lib/pages/edit_channel.dart @@ -43,7 +43,7 @@ class _EditChannelState extends State { var _members = []; var _showHistoryForNew = true; - var _canSave = false; + var _canSave = true; var _emojiVisible = false; Channel _channel; diff --git a/lib/widgets/channel/channel_tile.dart b/lib/widgets/channel/channel_tile.dart index 1b397da2..231ca337 100644 --- a/lib/widgets/channel/channel_tile.dart +++ b/lib/widgets/channel/channel_tile.dart @@ -40,16 +40,18 @@ class ChannelTile extends StatelessWidget { Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ - Text( - channel.name, - overflow: TextOverflow.ellipsis, - textAlign: TextAlign.start, - style: TextStyle( - fontSize: 16.0, - fontWeight: channel.hasUnread == 1 - ? FontWeight.w900 - : FontWeight.w400, - color: Color(0xff444444), + Expanded( + child: Text( + channel.name, + overflow: TextOverflow.ellipsis, + textAlign: TextAlign.start, + style: TextStyle( + fontSize: 16.0, + fontWeight: channel.hasUnread == 1 + ? FontWeight.w900 + : FontWeight.w400, + color: Color(0xff444444), + ), ), ), SizedBox(width: 6), @@ -77,6 +79,7 @@ class ChannelTile extends StatelessWidget { ], ), ), + SizedBox(width: 15), Column( crossAxisAlignment: CrossAxisAlignment.end, mainAxisAlignment: MainAxisAlignment.center, From 948d87b5cf9a905c8c4d1df4d7e1b74e71bab2a8 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 14:38:34 +0300 Subject: [PATCH 08/18] MessagesPage title layout fixed to show a long text. --- lib/pages/messages_page.dart | 73 +++++++++++++++++++----------------- 1 file changed, 39 insertions(+), 34 deletions(-) diff --git a/lib/pages/messages_page.dart b/lib/pages/messages_page.dart index 801e7eb7..b3b4983e 100644 --- a/lib/pages/messages_page.dart +++ b/lib/pages/messages_page.dart @@ -93,44 +93,49 @@ class MessagesPage extends StatelessWidget { if (state.parentChannel is Channel) TextAvatar(state.parentChannel.icon), SizedBox(width: 12.0), - Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Row( - children: [ - Text( - state.parentChannel.name, - style: TextStyle( - fontSize: 16.0, - fontWeight: FontWeight.w600, - color: Color(0xff444444), + Expanded( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + children: [ + Expanded( + child: Text( + state.parentChannel.name, + style: TextStyle( + fontSize: 16.0, + fontWeight: FontWeight.w600, + color: Color(0xff444444), + ), + overflow: TextOverflow.ellipsis, + ), ), - overflow: TextOverflow.ellipsis, - ), - SizedBox(width: 6), - if ((state.parentChannel is Channel) && - (state.parentChannel as Channel).visibility != - null && - (state.parentChannel as Channel).visibility == - 'private') - Icon(Icons.lock_outline, - size: 15.0, color: Color(0xff444444)), - ], - ), - if (state.parentChannel is Channel) - Padding( - padding: const EdgeInsets.only(top: 2.0), - child: Text( - '${(state.parentChannel as Channel).membersCount != null && (state.parentChannel as Channel).membersCount > 0 ? (state.parentChannel as Channel).membersCount : 'No'} members', - style: TextStyle( - fontSize: 10.0, - fontWeight: FontWeight.w400, - color: Color(0xff92929C), + SizedBox(width: 6), + if ((state.parentChannel is Channel) && + (state.parentChannel as Channel).visibility != + null && + (state.parentChannel as Channel).visibility == + 'private') + Icon(Icons.lock_outline, + size: 15.0, color: Color(0xff444444)), + ], + ), + if (state.parentChannel is Channel) + Padding( + padding: const EdgeInsets.only(top: 2.0), + child: Text( + '${(state.parentChannel as Channel).membersCount != null && (state.parentChannel as Channel).membersCount > 0 ? (state.parentChannel as Channel).membersCount : 'No'} members', + style: TextStyle( + fontSize: 10.0, + fontWeight: FontWeight.w400, + color: Color(0xff92929C), + ), ), ), - ), - ], + ], + ), ), + SizedBox(width: 15), ], ), ); From 09ccc7e809cd3c9a330a6ebf3f9fe38f7e82c86a Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 15:40:23 +0300 Subject: [PATCH 09/18] Drawer layout fix. --- lib/widgets/drawer/twake_drawer.dart | 52 +++++++++++++++------------- 1 file changed, 28 insertions(+), 24 deletions(-) diff --git a/lib/widgets/drawer/twake_drawer.dart b/lib/widgets/drawer/twake_drawer.dart index 1cc3751a..5ed35b5e 100644 --- a/lib/widgets/drawer/twake_drawer.dart +++ b/lib/widgets/drawer/twake_drawer.dart @@ -107,32 +107,36 @@ class _TwakeDrawerState extends State { height: 30, ), SizedBox(width: 15), - Column( - mainAxisAlignment: - MainAxisAlignment.center, - crossAxisAlignment: - CrossAxisAlignment.start, - children: [ - SizedBox(height: 12), - Text( - state.workspaces[i].name, - style: TextStyle( - fontSize: 16.0, - fontWeight: FontWeight.w600, - color: Color(0xff444444), + Expanded( + child: Column( + mainAxisAlignment: + MainAxisAlignment.center, + crossAxisAlignment: + CrossAxisAlignment.start, + children: [ + SizedBox(height: 12), + Text( + state.workspaces[i].name, + maxLines: 1, + overflow: TextOverflow.ellipsis, + style: TextStyle( + fontSize: 16.0, + fontWeight: FontWeight.w600, + color: Color(0xff444444), + ), ), - ), - SizedBox(height: 4), - Text( - '${state.workspaces[i].totalMembers} members', - style: TextStyle( - fontSize: 12.0, - fontWeight: FontWeight.w400, - color: Color(0xff444444), + SizedBox(height: 4), + Text( + '${state.workspaces[i].totalMembers} members', + style: TextStyle( + fontSize: 12.0, + fontWeight: FontWeight.w400, + color: Color(0xff444444), + ), ), - ), - SizedBox(height: 12), - ], + SizedBox(height: 12), + ], + ), ), ], ), From 9301a8c3831edb298c50d0fd58250758dbcb3de4 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 16:40:36 +0300 Subject: [PATCH 10/18] Keyboard appearance and behavior have been fixed on ServerConfiguration page. --- lib/pages/server_configuration.dart | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/lib/pages/server_configuration.dart b/lib/pages/server_configuration.dart index 09004b68..543d826a 100644 --- a/lib/pages/server_configuration.dart +++ b/lib/pages/server_configuration.dart @@ -125,7 +125,9 @@ class _ServerConfigurationState extends State { buildWhen: (_, current) => current is ConfigurationLoaded, builder: (context, state) { if (state is ConfigurationLoaded) { - _controller.text = state.host; + if (_controller.text.isReallyEmpty) { + _controller.text = state.host; + } print('Server host: ${state.host}'); return TextFormField( @@ -133,7 +135,7 @@ class _ServerConfigurationState extends State { validator: (value) => value.isEmpty ? 'Address cannot be blank' : null, controller: _controller, - keyboardType: TextInputType.emailAddress, + onFieldSubmitted: (_) => _connect(), style: TextStyle( fontSize: 17.0, fontWeight: FontWeight.w400, From 57864f7cd8c700e8fb37729a3b3a7704ccdd16d8 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 16:56:14 +0300 Subject: [PATCH 11/18] ServerConfiguration page layout fix. --- lib/pages/server_configuration.dart | 288 ++++++++++++++-------------- 1 file changed, 146 insertions(+), 142 deletions(-) diff --git a/lib/pages/server_configuration.dart b/lib/pages/server_configuration.dart index 543d826a..cdaaaa75 100644 --- a/lib/pages/server_configuration.dart +++ b/lib/pages/server_configuration.dart @@ -53,168 +53,172 @@ class _ServerConfigurationState extends State { ), ), body: SafeArea( - child: Column( - crossAxisAlignment: CrossAxisAlignment.center, - children: [ - SizedBox(height: 20.0), - Image.asset('assets/images/server.png'), - SizedBox(height: 20.0), - Text( - 'Server connection\npreference', - textAlign: TextAlign.center, - maxLines: 2, - style: TextStyle( - fontSize: 20.0, - fontWeight: FontWeight.w900, - color: Colors.black, - ), - ), - SizedBox(height: 36.0), - Padding( - padding: EdgeInsets.only(left: 16, right: 36.0), - child: Text( - 'Before you can proceed, please, choose a default server connection', + child: GestureDetector( + onTap: () => FocusScope.of(context).requestFocus(FocusNode()), + behavior: HitTestBehavior.opaque, + child: Column( + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + SizedBox(height: 10.0), + Image.asset('assets/images/server.png'), + SizedBox(height: 15.0), + Text( + 'Server connection\npreference', + textAlign: TextAlign.center, + maxLines: 2, style: TextStyle( - fontSize: 15.0, - fontWeight: FontWeight.normal, - color: Colors.black.withOpacity(0.6), + fontSize: 20.0, + fontWeight: FontWeight.w900, + color: Colors.black, ), ), - ), - Padding( - padding: EdgeInsets.fromLTRB(14.0, 12.0, 14.0, 0), - child: BlocListener( - listener: (context, state) { - if (state is HostValidated) { - widget.onConfirm(); - } - if (state is HostInvalid) { - print('HOST INVALID'); - Scaffold.of(context).showSnackBar(SnackBar( - content: Text( - 'Invalid host', - style: TextStyle( - color: Colors.red, + SizedBox(height: 36.0), + Padding( + padding: EdgeInsets.only(left: 16, right: 36.0), + child: Text( + 'Before you can proceed, please, choose a default server connection', + style: TextStyle( + fontSize: 15.0, + fontWeight: FontWeight.normal, + color: Colors.black.withOpacity(0.6), + ), + ), + ), + Padding( + padding: EdgeInsets.fromLTRB(14.0, 12.0, 14.0, 0), + child: BlocListener( + listener: (context, state) { + if (state is HostValidated) { + widget.onConfirm(); + } + if (state is HostInvalid) { + print('HOST INVALID'); + Scaffold.of(context).showSnackBar(SnackBar( + content: Text( + 'Invalid host', + style: TextStyle( + color: Colors.red, + ), ), - ), - duration: Duration(seconds: 2), - )); - } - }, - child: BlocConsumer( - listenWhen: (_, current) => - current is ConfigurationSaved || - current is ConfigurationError, - listener: (context, state) { - if (state is ConfigurationSaved) { - context - .read() - .add(ValidateHost(_controller.text)); - } else if (state is ConfigurationError) { - Scaffold.of(context).showSnackBar(SnackBar( - content: Text( - state.message, - style: TextStyle( - color: Colors.red, + duration: Duration(seconds: 2), + )); + } + }, + child: BlocConsumer( + listenWhen: (_, current) => + current is ConfigurationSaved || + current is ConfigurationError, + listener: (context, state) { + if (state is ConfigurationSaved) { + context + .read() + .add(ValidateHost(_controller.text)); + } else if (state is ConfigurationError) { + Scaffold.of(context).showSnackBar(SnackBar( + content: Text( + state.message, + style: TextStyle( + color: Colors.red, + ), ), - ), - duration: Duration(seconds: 2), - )); - } - }, - buildWhen: (_, current) => current is ConfigurationLoaded, - builder: (context, state) { - if (state is ConfigurationLoaded) { - if (_controller.text.isReallyEmpty) { - _controller.text = state.host; + duration: Duration(seconds: 2), + )); } - print('Server host: ${state.host}'); + }, + buildWhen: (_, current) => current is ConfigurationLoaded, + builder: (context, state) { + if (state is ConfigurationLoaded) { + if (_controller.text.isReallyEmpty) { + _controller.text = state.host; + } + print('Server host: ${state.host}'); - return TextFormField( - key: _formKey, - validator: (value) => - value.isEmpty ? 'Address cannot be blank' : null, - controller: _controller, - onFieldSubmitted: (_) => _connect(), - style: TextStyle( - fontSize: 17.0, - fontWeight: FontWeight.w400, - color: Colors.black, - ), - decoration: InputDecoration( - hintText: 'https://mobile.api.twake.app', - hintStyle: TextStyle( + return TextFormField( + key: _formKey, + validator: (value) => + value.isEmpty ? 'Address cannot be blank' : null, + controller: _controller, + onFieldSubmitted: (_) => _connect(), + style: TextStyle( fontSize: 17.0, fontWeight: FontWeight.w400, - color: Color(0xffc8c8c8), + color: Colors.black, ), - alignLabelWithHint: true, - fillColor: Color(0xfff4f4f4), - filled: true, - suffix: Container( - width: 30, - height: 25, - padding: EdgeInsets.only(left: 10), - child: IconButton( - padding: EdgeInsets.all(0), - onPressed: () => _controller.clear(), - iconSize: 15, - icon: Icon(CupertinoIcons.clear), + decoration: InputDecoration( + hintText: 'https://mobile.api.twake.app', + hintStyle: TextStyle( + fontSize: 17.0, + fontWeight: FontWeight.w400, + color: Color(0xffc8c8c8), ), - ), - border: UnderlineInputBorder( - borderRadius: BorderRadius.circular(10.0), - borderSide: BorderSide( - width: 0.0, - style: BorderStyle.none, + alignLabelWithHint: true, + fillColor: Color(0xfff4f4f4), + filled: true, + suffix: Container( + width: 30, + height: 25, + padding: EdgeInsets.only(left: 10), + child: IconButton( + padding: EdgeInsets.all(0), + onPressed: () => _controller.clear(), + iconSize: 15, + icon: Icon(CupertinoIcons.clear), + ), + ), + border: UnderlineInputBorder( + borderRadius: BorderRadius.circular(10.0), + borderSide: BorderSide( + width: 0.0, + style: BorderStyle.none, + ), ), ), - ), - ); - } else { - return Center(child: CircularProgressIndicator()); - } - }), - ), - ), - Spacer(), - Padding( - padding: EdgeInsets.symmetric(horizontal: 40.0), - child: Text( - 'Tap “Connect” if you don’t know exactly what is this all about', - textAlign: TextAlign.center, - style: TextStyle( - fontSize: 15.0, - fontWeight: FontWeight.normal, - color: Colors.black, + ); + } else { + return Center(child: CircularProgressIndicator()); + } + }), ), ), - ), - Padding( - padding: EdgeInsets.fromLTRB(24.0, 16.0, 24.0, 22.0), - child: TextButton( - onPressed: () => _connect(), - child: Container( - width: Size.infinite.width, - height: 50, - decoration: BoxDecoration( - color: Color(0xff3840f7), - borderRadius: BorderRadius.circular(14.0), + Spacer(), + Padding( + padding: EdgeInsets.symmetric(horizontal: 40.0), + child: Text( + 'Tap “Connect” if you don’t know exactly what is this all about', + textAlign: TextAlign.center, + style: TextStyle( + fontSize: 15.0, + fontWeight: FontWeight.normal, + color: Colors.black, ), - alignment: Alignment.center, - child: Text( - 'Connect', - textAlign: TextAlign.center, - style: TextStyle( - fontSize: 17.0, - fontWeight: FontWeight.bold, - color: Colors.white, + ), + ), + Padding( + padding: EdgeInsets.fromLTRB(24.0, 16.0, 24.0, 22.0), + child: TextButton( + onPressed: () => _connect(), + child: Container( + width: Size.infinite.width, + height: 50, + decoration: BoxDecoration( + color: Color(0xff3840f7), + borderRadius: BorderRadius.circular(14.0), + ), + alignment: Alignment.center, + child: Text( + 'Connect', + textAlign: TextAlign.center, + style: TextStyle( + fontSize: 17.0, + fontWeight: FontWeight.bold, + color: Colors.white, + ), ), ), ), ), - ), - ], + ], + ), ), ), ); From 489452f30a7064d54e7902de408f6cc87414457a Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Wed, 3 Mar 2021 17:09:40 +0300 Subject: [PATCH 12/18] Added channel is read sync to classic channels --- .../src/main/res/drawable-hdpi/logo_blue.png | Bin 0 -> 3687 bytes .../src/main/res/drawable-mdpi/logo_blue.png | Bin 0 -> 3687 bytes .../src/main/res/drawable-v21/logo_blue.png | Bin 0 -> 3687 bytes .../src/main/res/drawable-xhdpi/logo_blue.png | Bin 0 -> 3687 bytes .../main/res/drawable-xxhdpi/logo_blue.png | Bin 0 -> 3687 bytes .../main/res/drawable-xxxhdpi/logo_blue.png | Bin 0 -> 3687 bytes .../main/res/drawable/launch_background.xml | 5 ++++ .../app/src/main/res/drawable/logo_blue.png | Bin 0 -> 3687 bytes .../base_channel_bloc/base_channel_bloc.dart | 8 +++--- lib/blocs/channels_bloc/channels_bloc.dart | 10 +++++++- lib/blocs/messages_bloc/messages_bloc.dart | 23 +++++++++--------- .../single_message_bloc.dart | 2 +- .../single_message_event.dart | 3 ++- lib/blocs/threads_bloc/threads_bloc.dart | 2 ++ lib/repositories/collection_repository.dart | 10 +++++++- lib/repositories/messages_repository.dart | 23 ++++++++++++------ lib/services/endpoints.dart | 2 ++ lib/services/notifications.dart | 2 +- lib/widgets/common/reaction.dart | 8 ++++-- lib/widgets/message/message_tile.dart | 7 +++++- .../sheets/edit/member_management.dart | 8 +++--- 21 files changed, 77 insertions(+), 36 deletions(-) create mode 100644 android/app/src/main/res/drawable-hdpi/logo_blue.png create mode 100644 android/app/src/main/res/drawable-mdpi/logo_blue.png create mode 100644 android/app/src/main/res/drawable-v21/logo_blue.png create mode 100644 android/app/src/main/res/drawable-xhdpi/logo_blue.png create mode 100644 android/app/src/main/res/drawable-xxhdpi/logo_blue.png create mode 100644 android/app/src/main/res/drawable-xxxhdpi/logo_blue.png create mode 100644 android/app/src/main/res/drawable/logo_blue.png diff --git a/android/app/src/main/res/drawable-hdpi/logo_blue.png b/android/app/src/main/res/drawable-hdpi/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable-mdpi/logo_blue.png b/android/app/src/main/res/drawable-mdpi/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable-v21/logo_blue.png b/android/app/src/main/res/drawable-v21/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable-xhdpi/logo_blue.png b/android/app/src/main/res/drawable-xhdpi/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable-xxhdpi/logo_blue.png b/android/app/src/main/res/drawable-xxhdpi/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable-xxxhdpi/logo_blue.png b/android/app/src/main/res/drawable-xxxhdpi/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable/launch_background.xml b/android/app/src/main/res/drawable/launch_background.xml index f114f65c..c974a4f6 100644 --- a/android/app/src/main/res/drawable/launch_background.xml +++ b/android/app/src/main/res/drawable/launch_background.xml @@ -9,4 +9,9 @@ android:gravity="center" android:src="@drawable/logo" /> + + + diff --git a/android/app/src/main/res/drawable/logo_blue.png b/android/app/src/main/res/drawable/logo_blue.png new file mode 100644 index 0000000000000000000000000000000000000000..fb2a466c5331f96063fd2bf24b1c6a0d3c2ca276 GIT binary patch literal 3687 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 FV1j937)t;E literal 0 HcmV?d00001 diff --git a/lib/blocs/base_channel_bloc/base_channel_bloc.dart b/lib/blocs/base_channel_bloc/base_channel_bloc.dart index 4e03d105..57fbc76a 100644 --- a/lib/blocs/base_channel_bloc/base_channel_bloc.dart +++ b/lib/blocs/base_channel_bloc/base_channel_bloc.dart @@ -27,25 +27,25 @@ abstract class BaseChannelBloc extends Bloc { Future updateMessageCount(ModifyMessageCount event) async { final ch = await repository.getItemById(event.channelId); if (ch != null) { + repository.logger.d("UPDATING CHANNEL UNREAD"); // ch.messagesTotal += event.totalModifier ?? 0; ch.hasUnread = 1; ch.messagesUnread += event.unreadModifier ?? 0; ch.lastActivity = event.timeStamp ?? DateTime.now().millisecondsSinceEpoch; repository.saveOne(ch); - } else - return; + } } Future updateChannelState(ModifyChannelState event) async { final ch = await repository.getItemById(event.channelId); + repository.logger.d("UPDATING CHANNEL STATE"); if (ch != null) { ch.hasUnread = 1; if (event.threadId != null || event.messageId != null) { ch.messagesUnread += 1; } repository.saveOne(ch); - } else - return; + } } } diff --git a/lib/blocs/channels_bloc/channels_bloc.dart b/lib/blocs/channels_bloc/channels_bloc.dart index 6aa6d4ce..505f6804 100644 --- a/lib/blocs/channels_bloc/channels_bloc.dart +++ b/lib/blocs/channels_bloc/channels_bloc.dart @@ -10,6 +10,7 @@ import 'package:twake/models/channel.dart'; import 'package:twake/repositories/collection_repository.dart'; import 'package:twake/blocs/channels_bloc/channel_state.dart'; import 'package:twake/blocs/workspaces_bloc/workspace_state.dart'; +import 'package:twake/services/endpoints.dart'; export 'package:twake/blocs/channels_bloc/channel_event.dart'; export 'package:twake/blocs/channels_bloc/channel_state.dart'; @@ -107,7 +108,14 @@ class ChannelsBloc extends BaseChannelBloc { yield ChannelsEmpty(); } else if (event is ChangeSelectedChannel) { repository.logger.w('CHANNEL ${event.channelId} is selected'); - repository.select(event.channelId, saveToStore: false); + repository.select(event.channelId, + saveToStore: false, + apiEndpoint: Endpoint.channelsRead, + params: { + "company_id": ProfileBloc.selectedCompany, + "workspace_id": ProfileBloc.selectedWorkspace, + "channel_id": event.channelId + }); repository.selected.messagesUnread = 0; repository.selected.hasUnread = 0; diff --git a/lib/blocs/messages_bloc/messages_bloc.dart b/lib/blocs/messages_bloc/messages_bloc.dart index 12084497..77ddf2eb 100644 --- a/lib/blocs/messages_bloc/messages_bloc.dart +++ b/lib/blocs/messages_bloc/messages_bloc.dart @@ -197,15 +197,15 @@ class MessagesBloc _makeQueryParams(event), addToItems: event.channelId == selectedChannel.id, ); + _sortItems(); + final newState = MessagesLoaded( + messages: repository.items, + messageCount: repository.itemsCount, + force: DateTime.now().toString(), + parentChannel: selectedChannel, + ); + yield newState; if (updateParent) { - _sortItems(); - final newState = MessagesLoaded( - messages: repository.items, - messageCount: repository.itemsCount, - force: DateTime.now().toString(), - parentChannel: selectedChannel, - ); - yield newState; _updateParentChannel(event.channelId); } } else if (event is ModifyResponsesCount) { @@ -214,8 +214,8 @@ class MessagesBloc if (repository.selected == null) return; if (event.channelId == selectedChannel.id) { - repository.logger - .d('In thread: ${event.threadId == repository.selected.id}'); + // repository.logger + // .d('In thread: ${event.threadId == repository.selected.id}'); thread = event.threadId == repository.selected.id ? thread : repository.selected; @@ -226,7 +226,7 @@ class MessagesBloc parentChannel: selectedChannel, force: DateTime.now().toString(), ); - repository.logger.d('YIELDING STATE: ${newState != this.state}'); + // repository.logger.d('YIELDING STATE: ${newState != this.state}'); yield newState; _updateParentChannel(); } @@ -317,7 +317,6 @@ class MessagesBloc parentChannel: selectedChannel, ); await repository.updateResponsesCount(event.messageId); - print('SELECTED THREAD IS ${repository.selected.id}'); yield MessageSelected( threadMessage: repository.selected, responsesCount: repository.selected.responsesCount, diff --git a/lib/blocs/single_message_bloc/single_message_bloc.dart b/lib/blocs/single_message_bloc/single_message_bloc.dart index 4d723b97..c944f0c2 100644 --- a/lib/blocs/single_message_bloc/single_message_bloc.dart +++ b/lib/blocs/single_message_bloc/single_message_bloc.dart @@ -42,7 +42,7 @@ class SingleMessageBloc extends Bloc { message.updateContent({ 'company_id': ProfileBloc.selectedCompany, 'channel_id': message.channelId, - 'workspace_id': ProfileBloc.selectedWorkspace, + 'workspace_id': event.workspaceId ?? ProfileBloc.selectedWorkspace, 'message_id': message.id, 'thread_id': message.threadId, 'original_str': event.content, diff --git a/lib/blocs/single_message_bloc/single_message_event.dart b/lib/blocs/single_message_bloc/single_message_event.dart index da553945..36706aab 100644 --- a/lib/blocs/single_message_bloc/single_message_event.dart +++ b/lib/blocs/single_message_bloc/single_message_event.dart @@ -6,8 +6,9 @@ abstract class SingleMessageEvent extends Equatable { class UpdateContent extends SingleMessageEvent { final String content; + final String workspaceId; - const UpdateContent(this.content); + const UpdateContent({this.content, this.workspaceId}); @override List get props => [content]; diff --git a/lib/blocs/threads_bloc/threads_bloc.dart b/lib/blocs/threads_bloc/threads_bloc.dart index 0ac248cc..61a76054 100644 --- a/lib/blocs/threads_bloc/threads_bloc.dart +++ b/lib/blocs/threads_bloc/threads_bloc.dart @@ -78,12 +78,14 @@ class ThreadsBloc @override Stream mapEventToState(MessagesEvent event) async* { if (event is LoadMessages) { + print("SELECTED THREAD: ${threadMessage.toJson()}"); if (threadMessage.responsesCount == 0) { repository.clean(); yield MessagesEmpty( threadMessage: threadMessage, parentChannel: parentChannel, ); + repository.logger.d("RETURNING FROM LOADING RESPONSES"); return; } yield MessagesLoading( diff --git a/lib/repositories/collection_repository.dart b/lib/repositories/collection_repository.dart index 80c670f8..2ad34788 100644 --- a/lib/repositories/collection_repository.dart +++ b/lib/repositories/collection_repository.dart @@ -84,7 +84,10 @@ class CollectionRepository { return collection; } - void select(String itemId, {bool saveToStore: true}) { + void select(String itemId, + {bool saveToStore: true, + String apiEndpoint, + Map params}) { // logger.w('BEFORE SELECT $T ${selected.id}'); final item = items.firstWhere((i) => i.id == itemId); var oldSelected = selected; @@ -93,6 +96,11 @@ class CollectionRepository { assert(selected.id == item.id); if (saveToStore) saveOne(item); saveOne(oldSelected); + if (apiEndpoint != null && params != null) + _api.post( + apiEndpoint, + body: params, + ); // logger.w('AFTER SELECT $T ${selected.id}'); } diff --git a/lib/repositories/messages_repository.dart b/lib/repositories/messages_repository.dart index 1445ef73..98f84b66 100644 --- a/lib/repositories/messages_repository.dart +++ b/lib/repositories/messages_repository.dart @@ -157,11 +157,14 @@ class MessagesRepository { if (m == null) return null; // print('BEFORE COUNT: ${m.responsesCount}\nID: $messageId'); final sqlT = 'SELECT count(id) as count FROM message'; - final res = (await _storage.customQuery(sqlT, filters: [ + var res; + res = (await _storage.customQuery(sqlT, filters: [ ['thread_id', '=', messageId] ]))[0]['count']; - m.responsesCount = res; - _storage.store(item: m.toJson(), type: StorageType.Message); + if (res != 0) { + m.responsesCount = res; + await _storage.store(item: m.toJson(), type: StorageType.Message); + } // print('RESPONSES COUNT: $res'); // final sql = 'UPDATE message SET responses_count = ' // '(SELECT count(id) FROM message WHERE thread_id = ?) WHERE id = ?'; @@ -180,7 +183,7 @@ class MessagesRepository { Map queryParams, { bool addToItems = true, }) async { - // logger.d('Pulling item Message from api...\nPARAMS: $queryParams'); + logger.d('Pulling item Message from api...\nPARAMS: $queryParams'); List resp = []; try { resp = (await _api.get(apiEndpoint, params: queryParams)); @@ -190,6 +193,10 @@ class MessagesRepository { } if (resp.isEmpty) return false; var item = Message.fromJson(resp[0]); + var isNew = true; + if (getItemById(queryParams['message_id']) != null) { + isNew = false; + } saveOne(item); if (addToItems) { final query = 'SELECT message.*, ' @@ -219,8 +226,8 @@ class MessagesRepository { this.items.add(message); } - // logger.d('Pulled item: ${item.toJson()}'); - return true; + logger.d('Pulled item: ${item.toJson()}'); + return isNew; } Future pushOne( @@ -251,9 +258,9 @@ class MessagesRepository { if (!forceFromDB) item = items.firstWhere((i) => i.id == id, orElse: () => null); if (item == null) { - // print('GETTING MESSAGE BY ID: $id'); + print('GETTING MESSAGE BY ID: $id'); var map = await _storage.load(type: StorageType.Message, key: id); - // print('MESSAGE: $map'); + print('MESSAGE: $map'); if (map == null) return null; item = Message.fromJson(map); } diff --git a/lib/services/endpoints.dart b/lib/services/endpoints.dart index 0b804ccc..d3f1a647 100644 --- a/lib/services/endpoints.dart +++ b/lib/services/endpoints.dart @@ -19,6 +19,8 @@ class Endpoint { static const workspaceMembers = '/workspaces/members'; // API Endpoint for working with user's channels in a workspace static const channels = '/channels'; + // API Endpoint for marking the channel as read + static const channelsRead = '/channels/read'; // API Endpoint for working with the members of user's channels static const channelMembers = '/channels/members'; // API Endpoint for working with user's direct channels with other users diff --git a/lib/services/notifications.dart b/lib/services/notifications.dart index 5a486259..21390622 100644 --- a/lib/services/notifications.dart +++ b/lib/services/notifications.dart @@ -67,7 +67,7 @@ class Notifications { ); const AndroidInitializationSettings initializationSettingsAndroid = - AndroidInitializationSettings('logo'); + AndroidInitializationSettings('logo_blue'); final IOSInitializationSettings initializationSettingsIOS = IOSInitializationSettings(); final InitializationSettings initializationSettings = diff --git a/lib/widgets/common/reaction.dart b/lib/widgets/common/reaction.dart index d88f9276..6eb2d6ec 100644 --- a/lib/widgets/common/reaction.dart +++ b/lib/widgets/common/reaction.dart @@ -7,14 +7,18 @@ import 'package:twake/config/styles_config.dart'; class Reaction extends StatelessWidget { final String reaction; final int count; - Reaction(this.reaction, this.count); + final String workspaceId; + Reaction(this.reaction, this.count, [this.workspaceId]); @override Widget build(BuildContext context) { return InkWell( onTap: () { BlocProvider.of(context).add( - UpdateReaction(emojiCode: reaction), + UpdateReaction( + emojiCode: reaction, + workspaceId: workspaceId, + ), ); }, child: FittedBox( diff --git a/lib/widgets/message/message_tile.dart b/lib/widgets/message/message_tile.dart index 8769759b..1c5741bc 100644 --- a/lib/widgets/message/message_tile.dart +++ b/lib/widgets/message/message_tile.dart @@ -3,6 +3,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_markdown/flutter_markdown.dart'; import 'package:twake/blocs/base_channel_bloc/base_channel_bloc.dart'; +import 'package:twake/blocs/directs_bloc/directs_bloc.dart'; import 'package:twake/blocs/draft_bloc/draft_bloc.dart'; import 'package:twake/blocs/message_edit_bloc/message_edit_bloc.dart'; import 'package:twake/blocs/messages_bloc/messages_bloc.dart'; @@ -118,7 +119,10 @@ class _MessageTileState onMessageEditComplete: (text) { // smbloc get's closed if // listview disposes of message tile - smbloc.add(UpdateContent(text)); + smbloc.add(UpdateContent( + content: text, + workspaceId: + T == DirectsBloc ? 'direct' : null)); mebloc.add(CancelMessageEdit()); FocusManager.instance.primaryFocus.unfocus(); }), @@ -236,6 +240,7 @@ class _MessageTileState return Reaction( r, messageState.reactions[r]['count'], + T == DirectsBloc ? 'direct' : null, ); }), if (messageState.responsesCount > 0 && diff --git a/lib/widgets/sheets/edit/member_management.dart b/lib/widgets/sheets/edit/member_management.dart index 3071c21d..3aeb1ed4 100644 --- a/lib/widgets/sheets/edit/member_management.dart +++ b/lib/widgets/sheets/edit/member_management.dart @@ -62,7 +62,7 @@ class _MemberManagementState extends State { _channelId = state.channelId; _members = state.members; - print(_members.map((e) => e.email)); + // print(_members.map((e) => e.email)); _himself ??= _members.firstWhere( (m) => m.id == ProfileBloc.userId, @@ -71,9 +71,9 @@ class _MemberManagementState extends State { _members.removeWhere((m) => m.userId == _himself.userId); _members.insert(0, _himself); - print(_members.map((e) => e.email)); - print(_members.map((e) => e.userId)); - print(ProfileBloc.userId); + // print(_members.map((e) => e.email)); + // print(_members.map((e) => e.userId)); + // print(ProfileBloc.userId); } return Column( children: [ From bf4952865cb21a7ffe3f3dee987dfaa8e7cabad1 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Wed, 3 Mar 2021 21:03:27 +0300 Subject: [PATCH 13/18] AddDirectRepository added. --- lib/repositories/add_channel_repository.dart | 2 +- lib/repositories/add_direct_repository.dart | 54 +++++++++++++++++++ lib/repositories/add_direct_repository.g.dart | 24 +++++++++ lib/services/init.dart | 3 +- 4 files changed, 81 insertions(+), 2 deletions(-) create mode 100644 lib/repositories/add_direct_repository.dart create mode 100644 lib/repositories/add_direct_repository.g.dart diff --git a/lib/repositories/add_channel_repository.dart b/lib/repositories/add_channel_repository.dart index 49a189cb..63459bac 100644 --- a/lib/repositories/add_channel_repository.dart +++ b/lib/repositories/add_channel_repository.dart @@ -117,4 +117,4 @@ class AddChannelRepository { String channelId = resp['id']; return channelId; } -} +} \ No newline at end of file diff --git a/lib/repositories/add_direct_repository.dart b/lib/repositories/add_direct_repository.dart new file mode 100644 index 00000000..9211514c --- /dev/null +++ b/lib/repositories/add_direct_repository.dart @@ -0,0 +1,54 @@ +import 'package:json_annotation/json_annotation.dart'; +import 'package:logger/logger.dart'; +import 'package:twake/blocs/profile_bloc/profile_bloc.dart'; +import 'package:twake/services/api.dart'; +import 'package:twake/services/endpoints.dart'; + +part 'add_direct_repository.g.dart'; + +@JsonSerializable(explicitToJson: true) +class AddDirectRepository { + + @JsonKey(required: true, name: 'company_id') + String companyId; + @JsonKey(required: true) + String member; + @JsonKey(required: true, name: 'workspace_id') + String workspaceId = 'direct'; + + @JsonKey(ignore: true) + static final _logger = Logger(); + @JsonKey(ignore: true) + static final _api = Api(); + + AddDirectRepository(); + + factory AddDirectRepository.fromJson(Map json) => + _$AddDirectRepositoryFromJson(json); + + Map toJson() => _$AddDirectRepositoryToJson(this); + + + Future clear() async { + companyId = ''; + member = ''; + } + + Future create() async { + this.companyId = ProfileBloc.selectedCompany; + this.workspaceId = 'direct'; + + final body = this.toJson(); + _logger.d('Direct creation request body: $body'); + Map resp; + try { + resp = await _api.post(Endpoint.directs, body: body); + } catch (error) { + _logger.e('Error while trying to create a direct:\n${error.message}'); + return ''; + } + _logger.d('RESPONSE AFTER DIRECT CREATION: $resp'); + String directId = resp['id']; + return directId; + } +} \ No newline at end of file diff --git a/lib/repositories/add_direct_repository.g.dart b/lib/repositories/add_direct_repository.g.dart new file mode 100644 index 00000000..98289d61 --- /dev/null +++ b/lib/repositories/add_direct_repository.g.dart @@ -0,0 +1,24 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'add_direct_repository.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +AddDirectRepository _$AddDirectRepositoryFromJson(Map json) { + $checkKeys(json, + requiredKeys: const ['company_id', 'member', 'workspace_id']); + return AddDirectRepository() + ..companyId = json['company_id'] as String + ..member = json['member'] as String + ..workspaceId = json['workspace_id'] as String; +} + +Map _$AddDirectRepositoryToJson( + AddDirectRepository instance) => + { + 'company_id': instance.companyId, + 'member': instance.member, + 'workspace_id': instance.workspaceId, + }; diff --git a/lib/services/init.dart b/lib/services/init.dart index d3b6b298..79816c76 100644 --- a/lib/services/init.dart +++ b/lib/services/init.dart @@ -5,10 +5,10 @@ import 'package:twake/models/company.dart'; import 'package:twake/models/direct.dart'; import 'package:twake/models/workspace.dart'; import 'package:twake/repositories/add_channel_repository.dart'; +import 'package:twake/repositories/add_direct_repository.dart'; import 'package:twake/repositories/add_workspace_repository.dart'; import 'package:twake/repositories/auth_repository.dart'; import 'package:twake/repositories/collection_repository.dart'; -import 'package:twake/repositories/configuration_repository.dart'; import 'package:twake/repositories/edit_channel_repository.dart'; import 'package:twake/repositories/fields_repository.dart'; import 'package:twake/repositories/member_repository.dart'; @@ -46,6 +46,7 @@ Future initMain() async { final profile = await ProfileRepository.load(); final sheet = await SheetRepository.load(); final addChannel = await AddChannelRepository.load(); + final addDirect = AddDirectRepository(); final editChannel = await EditChannelRepository.load(); final addWorkspace = AddWorkspaceRepository(); final fields = FieldsRepository(fields: [], data: {}); From a1e60cab7b84fd6a3192ba24fcbdbf1431ecf77e Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Thu, 4 Mar 2021 11:21:14 +0300 Subject: [PATCH 14/18] AddChannelBloc update. --- .../add_channel_bloc/add_channel_bloc.dart | 78 ++++++++++++------- .../add_channel_bloc/add_channel_event.dart | 14 ++++ .../add_channel_bloc/add_channel_state.dart | 21 ++++- lib/pages/initial_page.dart | 5 +- lib/repositories/add_direct_repository.dart | 10 ++- lib/services/init.dart | 3 + 6 files changed, 99 insertions(+), 32 deletions(-) diff --git a/lib/blocs/add_channel_bloc/add_channel_bloc.dart b/lib/blocs/add_channel_bloc/add_channel_bloc.dart index df90467f..a3b899f3 100644 --- a/lib/blocs/add_channel_bloc/add_channel_bloc.dart +++ b/lib/blocs/add_channel_bloc/add_channel_bloc.dart @@ -1,13 +1,18 @@ import 'dart:async'; import 'package:bloc/bloc.dart'; import 'package:twake/repositories/add_channel_repository.dart'; +import 'package:twake/repositories/add_direct_repository.dart'; import 'add_channel_event.dart'; import 'add_channel_state.dart'; class AddChannelBloc extends Bloc { - final AddChannelRepository repository; + final AddChannelRepository channelRepository; + final AddDirectRepository directRepository; - AddChannelBloc(this.repository) : super(AddChannelInitial()); + AddChannelBloc( + this.channelRepository, + this.directRepository, + ) : super(AddChannelInitial()); @override Stream mapEventToState( @@ -16,43 +21,64 @@ class AddChannelBloc extends Bloc { if (event is SetFlowStage) { yield StageUpdated(event.stage); } else if (event is Update) { - repository.icon = event.icon ?? repository.icon; - repository.name = event.name ?? repository.name; - repository.description = event.description ?? repository.description; - repository.channelGroup = event.groupName ?? repository.channelGroup; - repository.type = event.type ?? repository.type ?? ChannelType.public; - repository.members = event.participants ?? repository.members ?? []; - repository.def = event.automaticallyAddNew ?? repository.def ?? true; + channelRepository.icon = event.icon ?? channelRepository.icon; + channelRepository.name = event.name ?? channelRepository.name; + channelRepository.description = + event.description ?? channelRepository.description; + channelRepository.channelGroup = + event.groupName ?? channelRepository.channelGroup; + channelRepository.type = + event.type ?? channelRepository.type ?? ChannelType.public; + channelRepository.members = + event.participants ?? channelRepository.members ?? []; + channelRepository.def = + event.automaticallyAddNew ?? channelRepository.def ?? true; // print('Updated data: ${repository.toJson()}'); - var newRepo = AddChannelRepository( - icon: repository.icon, - companyId: repository.companyId, - workspaceId: repository.workspaceId, - name: repository.name, - visibility: repository.visibility, - description: repository.description, - channelGroup: repository.channelGroup, - type: repository.type, - members: repository.members, - def: repository.def, + final newRepo = AddChannelRepository( + icon: channelRepository.icon, + companyId: channelRepository.companyId, + workspaceId: channelRepository.workspaceId, + name: channelRepository.name, + visibility: channelRepository.visibility, + description: channelRepository.description, + channelGroup: channelRepository.channelGroup, + type: channelRepository.type, + members: channelRepository.members, + def: channelRepository.def, ); - yield Updated(newRepo); + } else if (event is UpdateDirect) { + directRepository.member = event.member; + // print('Updated data: ${repository.toJson()}'); + final newRepo = AddDirectRepository( + companyId: directRepository.companyId, + workspaceId: directRepository.workspaceId, + member: directRepository.member, + ); + yield DirectUpdated(newRepo); } else if (event is Create) { yield Creation(); - final type = repository.type; - final result = await repository.create(); + final type = channelRepository.type; + final result = await channelRepository.create(); if (result.isNotEmpty) { - repository.clear(); - yield Created(result, type); + channelRepository.clear(); + yield Created(result, channelType: type); } else { yield Error('Channel creation failure!'); } } else if (event is Clear) { - repository.clear(); + channelRepository.clear(); } else if (event is SetFlowType) { yield FlowTypeSet(event.isDirect); + } else if (event is CreateDirect) { + yield Creation(); + final result = await directRepository.create(); + if (result.isNotEmpty) { + yield DirectCreated(result); + } else { + yield Error('Direct creation failure!'); + } } } } diff --git a/lib/blocs/add_channel_bloc/add_channel_event.dart b/lib/blocs/add_channel_bloc/add_channel_event.dart index f71c7100..f164c270 100644 --- a/lib/blocs/add_channel_bloc/add_channel_event.dart +++ b/lib/blocs/add_channel_bloc/add_channel_event.dart @@ -15,6 +15,11 @@ class Create extends AddChannelEvent { List get props => []; } +class CreateDirect extends AddChannelEvent { + @override + List get props => []; +} + class Update extends AddChannelEvent { final String icon; final String name; @@ -46,6 +51,15 @@ class Update extends AddChannelEvent { ]; } +class UpdateDirect extends AddChannelEvent { + final String member; + + UpdateDirect({this.member}); + + @override + List get props => [member]; +} + class SetFlowType extends AddChannelEvent { final bool isDirect; diff --git a/lib/blocs/add_channel_bloc/add_channel_state.dart b/lib/blocs/add_channel_bloc/add_channel_state.dart index 41916813..07419ea8 100644 --- a/lib/blocs/add_channel_bloc/add_channel_state.dart +++ b/lib/blocs/add_channel_bloc/add_channel_state.dart @@ -1,5 +1,6 @@ import 'package:equatable/equatable.dart'; import 'package:twake/repositories/add_channel_repository.dart'; +import 'package:twake/repositories/add_direct_repository.dart'; abstract class AddChannelState extends Equatable { const AddChannelState(); @@ -19,6 +20,15 @@ class Updated extends AddChannelState { List get props => [repository]; } +class DirectUpdated extends AddChannelState { + final AddDirectRepository repository; + + DirectUpdated(this.repository); + + @override + List get props => [repository]; +} + class Creation extends AddChannelState { @override List get props => []; @@ -28,12 +38,21 @@ class Created extends AddChannelState { final String id; final ChannelType channelType; - Created(this.id, this.channelType); + Created(this.id, {this.channelType}); @override List get props => [id, channelType]; } +class DirectCreated extends AddChannelState { + final String id; + + DirectCreated(this.id); + + @override + List get props => [id]; +} + class Error extends AddChannelState { final String message; diff --git a/lib/pages/initial_page.dart b/lib/pages/initial_page.dart index 8c40e3de..fe0b98c6 100644 --- a/lib/pages/initial_page.dart +++ b/lib/pages/initial_page.dart @@ -189,7 +189,10 @@ class _InitialPageState extends State with WidgetsBindingObserver { lazy: false, ), BlocProvider( - create: (_) => AddChannelBloc(state.initData.addChannel), + create: (_) => AddChannelBloc( + state.initData.addChannel, + state.initData.addDirect, + ), lazy: false, ), BlocProvider( diff --git a/lib/repositories/add_direct_repository.dart b/lib/repositories/add_direct_repository.dart index 9211514c..600ec6eb 100644 --- a/lib/repositories/add_direct_repository.dart +++ b/lib/repositories/add_direct_repository.dart @@ -8,7 +8,6 @@ part 'add_direct_repository.g.dart'; @JsonSerializable(explicitToJson: true) class AddDirectRepository { - @JsonKey(required: true, name: 'company_id') String companyId; @JsonKey(required: true) @@ -21,14 +20,17 @@ class AddDirectRepository { @JsonKey(ignore: true) static final _api = Api(); - AddDirectRepository(); + AddDirectRepository({ + this.companyId, + this.workspaceId, + this.member, + }); factory AddDirectRepository.fromJson(Map json) => _$AddDirectRepositoryFromJson(json); Map toJson() => _$AddDirectRepositoryToJson(this); - Future clear() async { companyId = ''; member = ''; @@ -51,4 +53,4 @@ class AddDirectRepository { String directId = resp['id']; return directId; } -} \ No newline at end of file +} diff --git a/lib/services/init.dart b/lib/services/init.dart index 79816c76..d043abcd 100644 --- a/lib/services/init.dart +++ b/lib/services/init.dart @@ -120,6 +120,7 @@ Future initMain() async { threads: threads, sheet: sheet, addChannel: addChannel, + addDirect: addDirect, editChannel: editChannel, channelMembers: channelMembers, addWorkspace: addWorkspace, @@ -140,6 +141,7 @@ class InitData { final MessagesRepository threads; final SheetRepository sheet; final AddChannelRepository addChannel; + final AddDirectRepository addDirect; final EditChannelRepository editChannel; final AddWorkspaceRepository addWorkspace; final DraftRepository draft; @@ -157,6 +159,7 @@ class InitData { this.threads, this.sheet, this.addChannel, + this.addDirect, this.editChannel, this.addWorkspace, this.draft, From e34b4ea11418cb4586c5052c79e3411a0dc6511a Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Thu, 4 Mar 2021 12:15:04 +0300 Subject: [PATCH 15/18] Directs creation has been separated from the channels. --- lib/widgets/sheets/add/participants_list.dart | 21 ++++++++++--------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/lib/widgets/sheets/add/participants_list.dart b/lib/widgets/sheets/add/participants_list.dart index a7a8e3bf..798cb72b 100644 --- a/lib/widgets/sheets/add/participants_list.dart +++ b/lib/widgets/sheets/add/participants_list.dart @@ -108,12 +108,10 @@ class _ParticipantsListState extends State { void _createDirect(List participantsIds) { context.read() - ..add(Update( - name: '', - type: ChannelType.direct, - participants: participantsIds, + ..add(UpdateDirect( + member: participantsIds.first, )) - ..add(Create()); + ..add(CreateDirect()); FocusScope.of(context).requestFocus(FocusNode()); } @@ -127,9 +125,11 @@ class _ParticipantsListState extends State { }, child: BlocConsumer( listener: (context, state) { - if (state is Created) { + if (state is Created || state is DirectCreated) { // Reload channels - context.read().add(ReloadChannels(forceFromApi: true)); + context + .read() + .add(ReloadChannels(forceFromApi: true)); // Reload directs context.read().add(ReloadChannels(forceFromApi: true)); // Close sheet @@ -138,7 +138,7 @@ class _ParticipantsListState extends State { context.read().add(Update(participants: [])); _controller.clear(); // Redirect user to created direct - if (_isDirect) { + if (state is DirectCreated) { String channelId = state.id; openDirect(context, channelId); } @@ -157,6 +157,7 @@ class _ParticipantsListState extends State { }, buildWhen: (_, current) { return (current is Updated || + current is DirectUpdated || current is Creation || current is StageUpdated); }, @@ -205,7 +206,8 @@ class _ParticipantsListState extends State { // print('-------------------------------'); } return BlocBuilder( - buildWhen: (previous, current) => current is Updated, + buildWhen: (previous, current) => + current is Updated, builder: (context, state) { var selectedIds = []; var selectedUsers = []; @@ -222,7 +224,6 @@ class _ParticipantsListState extends State { users.excludeUsers(selectedUsers); } } - // print('Selected UsERS: ${selectedUsers.map((e) => e.username)}'); return ListView.builder( shrinkWrap: true, From ead3e4f78544b871066bd74b0d5dfd4ef02b6593 Mon Sep 17 00:00:00 2001 From: Pavel Zarudnev Date: Thu, 4 Mar 2021 12:34:14 +0300 Subject: [PATCH 16/18] Workspace and Channel names limited by 30 characters. --- lib/pages/edit_channel.dart | 1 + lib/widgets/sheets/add/channel_info_form.dart | 1 + lib/widgets/sheets/add/workspace_info_form.dart | 1 + lib/widgets/sheets/sheet_text_field.dart | 6 ++++++ 4 files changed, 9 insertions(+) diff --git a/lib/pages/edit_channel.dart b/lib/pages/edit_channel.dart index 3d800621..5de6b8f9 100644 --- a/lib/pages/edit_channel.dart +++ b/lib/pages/edit_channel.dart @@ -341,6 +341,7 @@ class _EditChannelState extends State { hint: 'Channel name', controller: _nameController, focusNode: _nameFocusNode, + maxLength: 30, ), Divider( thickness: 0.5, diff --git a/lib/widgets/sheets/add/channel_info_form.dart b/lib/widgets/sheets/add/channel_info_form.dart index 3465dd95..fcf29776 100644 --- a/lib/widgets/sheets/add/channel_info_form.dart +++ b/lib/widgets/sheets/add/channel_info_form.dart @@ -221,6 +221,7 @@ class _ChannelInfoFormState extends State { hint: 'Channel name', controller: _channelNameController, focusNode: _channelNameFocusNode, + maxLength: 30, ), ), ], diff --git a/lib/widgets/sheets/add/workspace_info_form.dart b/lib/widgets/sheets/add/workspace_info_form.dart index a3989ffd..66f788d6 100644 --- a/lib/widgets/sheets/add/workspace_info_form.dart +++ b/lib/widgets/sheets/add/workspace_info_form.dart @@ -152,6 +152,7 @@ class _WorkspaceInfoFormState extends State { hint: 'Workspace name', controller: _workspaceNameController, focusNode: _workspaceNameFocusNode, + maxLength: 30, ), ), ), diff --git a/lib/widgets/sheets/sheet_text_field.dart b/lib/widgets/sheets/sheet_text_field.dart index 5d5325ff..934c545d 100644 --- a/lib/widgets/sheets/sheet_text_field.dart +++ b/lib/widgets/sheets/sheet_text_field.dart @@ -1,5 +1,6 @@ import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; class SheetTextField extends StatefulWidget { final String hint; @@ -7,6 +8,7 @@ class SheetTextField extends StatefulWidget { final Function trailingAction; final TextEditingController controller; final FocusNode focusNode; + final int maxLength; final String Function(String) validator; const SheetTextField({ @@ -16,6 +18,7 @@ class SheetTextField extends StatefulWidget { this.leadingAction, this.trailingAction, this.validator, + this.maxLength, }); @override @@ -29,6 +32,9 @@ class _SheetTextFieldState extends State { padding: const EdgeInsets.only(left: 14.0, right: 7), color: Colors.white, child: TextFormField( + inputFormatters:[ + LengthLimitingTextInputFormatter(widget.maxLength), + ], validator: widget.validator, controller: widget.controller, focusNode: widget.focusNode, From 54a7a30e8075a4b0ec15924ebbe904e556c4cbc1 Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Thu, 4 Mar 2021 13:40:56 +0300 Subject: [PATCH 17/18] Removed unread state updates when sending messages in thread --- lib/blocs/directs_bloc/directs_bloc.dart | 11 ++++++++++- lib/blocs/messages_bloc/messages_bloc.dart | 8 ++++---- lib/blocs/threads_bloc/threads_bloc.dart | 12 +++++++----- lib/repositories/messages_repository.dart | 5 +++-- 4 files changed, 24 insertions(+), 12 deletions(-) diff --git a/lib/blocs/directs_bloc/directs_bloc.dart b/lib/blocs/directs_bloc/directs_bloc.dart index a322f345..13aa2e22 100644 --- a/lib/blocs/directs_bloc/directs_bloc.dart +++ b/lib/blocs/directs_bloc/directs_bloc.dart @@ -4,9 +4,11 @@ import 'package:twake/blocs/base_channel_bloc/base_channel_bloc.dart'; import 'package:twake/blocs/companies_bloc/companies_bloc.dart'; import 'package:twake/blocs/notification_bloc/notification_bloc.dart'; import 'package:twake/blocs/channels_bloc/channel_event.dart'; +import 'package:twake/blocs/profile_bloc/profile_bloc.dart'; import 'package:twake/models/direct.dart'; import 'package:twake/repositories/collection_repository.dart'; import 'package:twake/blocs/channels_bloc/channel_state.dart'; +import 'package:twake/services/service_bundle.dart'; export 'package:twake/blocs/channels_bloc/channel_event.dart'; export 'package:twake/blocs/channels_bloc/channel_state.dart'; @@ -101,7 +103,14 @@ class DirectsBloc extends BaseChannelBloc { await repository.clean(); yield ChannelsEmpty(); } else if (event is ChangeSelectedChannel) { - repository.select(event.channelId); + repository.select(event.channelId, + saveToStore: false, + apiEndpoint: Endpoint.channelsRead, + params: { + "company_id": ProfileBloc.selectedCompany, + "workspace_id": "direct", + "channel_id": event.channelId + }); repository.selected.messagesUnread = 0; repository.selected.hasUnread = 0; repository.saveOne(repository.selected); diff --git a/lib/blocs/messages_bloc/messages_bloc.dart b/lib/blocs/messages_bloc/messages_bloc.dart index 77ddf2eb..21d77f8f 100644 --- a/lib/blocs/messages_bloc/messages_bloc.dart +++ b/lib/blocs/messages_bloc/messages_bloc.dart @@ -198,6 +198,9 @@ class MessagesBloc addToItems: event.channelId == selectedChannel.id, ); _sortItems(); + if (updateParent) { + _updateParentChannel(event.channelId); + } final newState = MessagesLoaded( messages: repository.items, messageCount: repository.itemsCount, @@ -205,9 +208,6 @@ class MessagesBloc parentChannel: selectedChannel, ); yield newState; - if (updateParent) { - _updateParentChannel(event.channelId); - } } else if (event is ModifyResponsesCount) { var thread = await repository.updateResponsesCount(event.threadId); if (thread == null) return; @@ -228,7 +228,7 @@ class MessagesBloc ); // repository.logger.d('YIELDING STATE: ${newState != this.state}'); yield newState; - _updateParentChannel(); + // _updateParentChannel(); } } else if (event is RemoveMessage) { final channelId = event.channelId ?? selectedChannel.id; diff --git a/lib/blocs/threads_bloc/threads_bloc.dart b/lib/blocs/threads_bloc/threads_bloc.dart index 61a76054..e10e2985 100644 --- a/lib/blocs/threads_bloc/threads_bloc.dart +++ b/lib/blocs/threads_bloc/threads_bloc.dart @@ -118,12 +118,14 @@ class ThreadsBloc await Future.delayed(Duration(milliseconds: 100)); attempt -= 1; } - await repository.pullOne(_makeQueryParams(event), + final updateParent = await repository.pullOne(_makeQueryParams(event), addToItems: threadMessage != null ? threadMessage.id == event.threadId : false); + if (updateParent) { + _updateParentChannel(); + } _sortItems(); - _updateParentChannel(); messagesBloc.add(ModifyResponsesCount( channelId: event.channelId, threadId: event.threadId, @@ -149,7 +151,7 @@ class ThreadsBloc _sortItems(); yield messagesLoaded; } - _updateParentChannel(totalModifier: -1); + // _updateParentChannel(totalModifier: -1); } else if (event is SendMessage) { final String dummyId = _DUMMY_ID; final body = _makeQueryParams(event); @@ -186,7 +188,7 @@ class ThreadsBloc channelId: event.channelId, threadId: message.threadId, )); - _updateParentChannel(); + // _updateParentChannel(); }, ); this.repository.items.add(tempItem); @@ -247,7 +249,7 @@ class ThreadsBloc final channelId = messagesBloc.selectedChannel.id; messagesBloc.channelsBloc.add(ModifyMessageCount( channelId: channelId, - workspaceId: ProfileBloc.selectedWorkspace, + workspaceId: T == DirectsBloc ? "direct" : ProfileBloc.selectedWorkspace, companyId: ProfileBloc.selectedCompany, totalModifier: totalModifier, )); diff --git a/lib/repositories/messages_repository.dart b/lib/repositories/messages_repository.dart index 98f84b66..f2502748 100644 --- a/lib/repositories/messages_repository.dart +++ b/lib/repositories/messages_repository.dart @@ -188,13 +188,14 @@ class MessagesRepository { try { resp = (await _api.get(apiEndpoint, params: queryParams)); } on ApiError catch (error) { - logger.d('ERROR while loading more Message from api\n${error.message}'); + logger.e('ERROR while loading more Message from api\n${error.message}'); return false; } if (resp.isEmpty) return false; var item = Message.fromJson(resp[0]); var isNew = true; - if (getItemById(queryParams['message_id']) != null) { + if (await getItemById(queryParams['message_id']) != null) { + logger.e("MESSAGE EXISTS"); isNew = false; } saveOne(item); From 3ce13b63c49f7546d90d0d38f6da088986f44a9e Mon Sep 17 00:00:00 2001 From: Makhmudov Babur Date: Thu, 4 Mar 2021 23:40:26 +0300 Subject: [PATCH 18/18] Build increment 2.2.9+1 --- pubspec.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pubspec.yaml b/pubspec.yaml index 1806b79f..e41acdf9 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -15,7 +15,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev # In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion. # Read more about iOS versioning at # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html -version: 2.2.9 +version: 2.2.9+1 environment: sdk: ">=2.7.0 <3.0.0"