mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2024-08-30 18:12:39 +00:00
fix: minor ui issues (#4102)
* fix: minor ui issues * feat: support using emoji as icon * chore: update langauges * fix: missing reminder bloc in detail page * fix: integration test
This commit is contained in:
parent
99b2b2712b
commit
fe5ce75ea8
@ -3,6 +3,8 @@ import 'package:appflowy/workspace/presentation/settings/widgets/settings_user_v
|
|||||||
import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
|
import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
import 'package:flutter_test/flutter_test.dart';
|
||||||
import 'package:integration_test/integration_test.dart';
|
import 'package:integration_test/integration_test.dart';
|
||||||
|
|
||||||
|
import '../util/emoji.dart';
|
||||||
import '../util/util.dart';
|
import '../util/util.dart';
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
@ -28,27 +30,12 @@ void main() {
|
|||||||
await tester.pumpAndSettle();
|
await tester.pumpAndSettle();
|
||||||
|
|
||||||
// Select first option that isn't default
|
// Select first option that isn't default
|
||||||
await tester.tap(find.byType(IconOption).first);
|
await tester.tapEmoji('😁');
|
||||||
await tester.pumpAndSettle();
|
await tester.pumpAndSettle();
|
||||||
|
|
||||||
UserAvatar userAvatar = tester.widget(userAvatarFinder) as UserAvatar;
|
final UserAvatar userAvatar =
|
||||||
expect(userAvatar.iconUrl, isNotEmpty);
|
tester.widget(userAvatarFinder) as UserAvatar;
|
||||||
|
expect(userAvatar.iconUrl, '😁');
|
||||||
// Open icon picker dialog again
|
|
||||||
await tester.tap(userAvatarFinder);
|
|
||||||
await tester.pumpAndSettle();
|
|
||||||
|
|
||||||
// Tap the default option
|
|
||||||
await tester.tap(
|
|
||||||
find.descendant(
|
|
||||||
of: find.byType(IconGallery),
|
|
||||||
matching: find.byType(UserAvatar),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
await tester.pumpAndSettle();
|
|
||||||
|
|
||||||
userAvatar = tester.widget(userAvatarFinder) as UserAvatar;
|
|
||||||
expect(userAvatar.iconUrl, isEmpty);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
||||||
|
import 'package:appflowy/generated/locale_keys.g.dart';
|
||||||
import 'package:appflowy/mobile/presentation/home/mobile_home_setting_page.dart';
|
import 'package:appflowy/mobile/presentation/home/mobile_home_setting_page.dart';
|
||||||
import 'package:appflowy/plugins/base/emoji/emoji_picker_screen.dart';
|
import 'package:appflowy/plugins/base/emoji/emoji_picker_screen.dart';
|
||||||
import 'package:appflowy/plugins/base/icon/icon_picker.dart';
|
import 'package:appflowy/plugins/base/icon/icon_picker.dart';
|
||||||
import 'package:appflowy/startup/startup.dart';
|
import 'package:appflowy/startup/startup.dart';
|
||||||
import 'package:appflowy/workspace/application/user/settings_user_bloc.dart';
|
import 'package:appflowy/workspace/application/user/settings_user_bloc.dart';
|
||||||
|
import 'package:appflowy/workspace/presentation/settings/widgets/settings_user_view.dart';
|
||||||
import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart';
|
import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart';
|
||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
|
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
@ -31,33 +34,7 @@ class MobileHomePageHeader extends StatelessWidget {
|
|||||||
constraints: const BoxConstraints(minHeight: 48),
|
constraints: const BoxConstraints(minHeight: 48),
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
FlowyButton(
|
_UserIcon(userIcon: userIcon),
|
||||||
useIntrinsicWidth: true,
|
|
||||||
text: FlowyText(
|
|
||||||
// replace with user icon
|
|
||||||
userIcon.isNotEmpty ? userIcon : '🐻',
|
|
||||||
fontSize: 26,
|
|
||||||
),
|
|
||||||
onTap: () async {
|
|
||||||
final icon = await context.push<EmojiPickerResult>(
|
|
||||||
Uri(
|
|
||||||
path: MobileEmojiPickerScreen.routeName,
|
|
||||||
queryParameters: {
|
|
||||||
MobileEmojiPickerScreen.pageTitle: 'User icon',
|
|
||||||
},
|
|
||||||
).toString(),
|
|
||||||
);
|
|
||||||
if (icon != null) {
|
|
||||||
if (context.mounted) {
|
|
||||||
context.read<SettingsUserViewBloc>().add(
|
|
||||||
SettingsUserEvent.updateUserIcon(
|
|
||||||
iconUrl: icon.emoji,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
),
|
|
||||||
const HSpace(12),
|
const HSpace(12),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Column(
|
child: Column(
|
||||||
@ -96,3 +73,49 @@ class MobileHomePageHeader extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _UserIcon extends StatelessWidget {
|
||||||
|
const _UserIcon({
|
||||||
|
required this.userIcon,
|
||||||
|
});
|
||||||
|
|
||||||
|
final String userIcon;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return FlowyButton(
|
||||||
|
useIntrinsicWidth: true,
|
||||||
|
text: builtInSVGIcons.contains(userIcon)
|
||||||
|
// to be compatible with old user icon
|
||||||
|
? FlowySvg(
|
||||||
|
FlowySvgData('emoji/$userIcon'),
|
||||||
|
size: const Size.square(32),
|
||||||
|
blendMode: null,
|
||||||
|
)
|
||||||
|
: FlowyText(
|
||||||
|
userIcon.isNotEmpty ? userIcon : '🐻',
|
||||||
|
fontSize: 26,
|
||||||
|
),
|
||||||
|
onTap: () async {
|
||||||
|
final icon = await context.push<EmojiPickerResult>(
|
||||||
|
Uri(
|
||||||
|
path: MobileEmojiPickerScreen.routeName,
|
||||||
|
queryParameters: {
|
||||||
|
MobileEmojiPickerScreen.pageTitle:
|
||||||
|
LocaleKeys.titleBar_userIcon.tr(),
|
||||||
|
},
|
||||||
|
).toString(),
|
||||||
|
);
|
||||||
|
if (icon != null) {
|
||||||
|
if (context.mounted) {
|
||||||
|
context.read<SettingsUserViewBloc>().add(
|
||||||
|
SettingsUserEvent.updateUserIcon(
|
||||||
|
iconUrl: icon.emoji,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -13,7 +13,7 @@ class AppFlowyCloudPage extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(LocaleKeys.settings_menu_cloudSetting.tr()),
|
title: Text(LocaleKeys.settings_menu_cloudSettings.tr()),
|
||||||
),
|
),
|
||||||
body: Padding(
|
body: Padding(
|
||||||
padding: const EdgeInsets.all(20.0),
|
padding: const EdgeInsets.all(20.0),
|
||||||
|
@ -17,7 +17,7 @@ class CloudSettingGroup extends StatelessWidget {
|
|||||||
return FutureBuilder(
|
return FutureBuilder(
|
||||||
future: PackageInfo.fromPlatform(),
|
future: PackageInfo.fromPlatform(),
|
||||||
builder: (context, snapshot) => MobileSettingGroup(
|
builder: (context, snapshot) => MobileSettingGroup(
|
||||||
groupTitle: LocaleKeys.settings_menu_cloudSetting.tr(),
|
groupTitle: LocaleKeys.settings_menu_cloudSettings.tr(),
|
||||||
settingItemList: [
|
settingItemList: [
|
||||||
MobileSettingItem(
|
MobileSettingItem(
|
||||||
name: LocaleKeys.settings_menu_cloudAppFlowy.tr(),
|
name: LocaleKeys.settings_menu_cloudAppFlowy.tr(),
|
||||||
|
@ -2,8 +2,9 @@ import 'package:appflowy/plugins/database_view/application/field/field_controlle
|
|||||||
import 'package:appflowy/plugins/database_view/application/row/row_controller.dart';
|
import 'package:appflowy/plugins/database_view/application/row/row_controller.dart';
|
||||||
import 'package:appflowy/plugins/database_view/grid/application/row/row_detail_bloc.dart';
|
import 'package:appflowy/plugins/database_view/grid/application/row/row_detail_bloc.dart';
|
||||||
import 'package:appflowy/plugins/database_view/widgets/row/row_document.dart';
|
import 'package:appflowy/plugins/database_view/widgets/row/row_document.dart';
|
||||||
|
import 'package:appflowy/startup/startup.dart';
|
||||||
|
import 'package:appflowy/user/application/reminder/reminder_bloc.dart';
|
||||||
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
|
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
|
||||||
@ -43,9 +44,17 @@ class _RowDetailPageState extends State<RowDetailPage> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return FlowyDialog(
|
return FlowyDialog(
|
||||||
child: BlocProvider(
|
child: MultiBlocProvider(
|
||||||
create: (context) => RowDetailBloc(rowController: widget.rowController)
|
providers: [
|
||||||
..add(const RowDetailEvent.initial()),
|
BlocProvider(
|
||||||
|
create: (context) =>
|
||||||
|
RowDetailBloc(rowController: widget.rowController)
|
||||||
|
..add(const RowDetailEvent.initial()),
|
||||||
|
),
|
||||||
|
BlocProvider.value(
|
||||||
|
value: getIt<ReminderBloc>(),
|
||||||
|
),
|
||||||
|
],
|
||||||
child: ListView(
|
child: ListView(
|
||||||
controller: scrollController,
|
controller: scrollController,
|
||||||
children: [
|
children: [
|
||||||
|
@ -189,35 +189,30 @@ class _MobileSignInButton extends StatelessWidget {
|
|||||||
width: 0.5,
|
width: 0.5,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
child: Center(
|
alignment: Alignment.center,
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
SizedBox(
|
SizedBox(
|
||||||
// The icon could be in different height as original aspect ratio, we use a fixed sizebox to wrap it to make sure they all occupy the same space.
|
// The icon could be in different height as original aspect ratio, we use a fixed sizebox to wrap it to make sure they all occupy the same space.
|
||||||
width: 30,
|
width: 30,
|
||||||
height: 30,
|
height: 30,
|
||||||
child: Center(
|
child: Center(
|
||||||
child: SizedBox(
|
child: SizedBox(
|
||||||
width: 24,
|
width: 24,
|
||||||
child: FlowySvg(
|
child: FlowySvg(
|
||||||
icon,
|
icon,
|
||||||
blendMode: null,
|
blendMode: null,
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const HSpace(8),
|
),
|
||||||
SizedBox(
|
const HSpace(8),
|
||||||
// To fit the longest label 'Log in with Discord'
|
Text(
|
||||||
width: 135,
|
labelText,
|
||||||
child: Text(
|
style: Theme.of(context).textTheme.titleSmall,
|
||||||
labelText,
|
),
|
||||||
style: Theme.of(context).textTheme.titleSmall,
|
],
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
@ -1,21 +1,21 @@
|
|||||||
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
||||||
|
import 'package:appflowy/generated/locale_keys.g.dart';
|
||||||
import 'package:appflowy/plugins/document/presentation/more/cubit/document_appearance_cubit.dart';
|
import 'package:appflowy/plugins/document/presentation/more/cubit/document_appearance_cubit.dart';
|
||||||
import 'package:appflowy/startup/startup.dart';
|
import 'package:appflowy/startup/startup.dart';
|
||||||
import 'package:appflowy/workspace/application/menu/menu_user_bloc.dart';
|
import 'package:appflowy/workspace/application/menu/menu_user_bloc.dart';
|
||||||
import 'package:appflowy/workspace/presentation/notifications/widgets/notification_button.dart';
|
import 'package:appflowy/workspace/presentation/notifications/widgets/notification_button.dart';
|
||||||
import 'package:appflowy/workspace/presentation/settings/settings_dialog.dart';
|
import 'package:appflowy/workspace/presentation/settings/settings_dialog.dart';
|
||||||
|
import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
|
||||||
import 'package:appflowy_backend/log.dart';
|
import 'package:appflowy_backend/log.dart';
|
||||||
import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
|
import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
|
||||||
import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
|
import 'package:appflowy_backend/protobuf/flowy-user/protobuf.dart'
|
||||||
|
show UserProfilePB;
|
||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
import 'package:flowy_infra_ui/style_widget/text.dart';
|
import 'package:flowy_infra_ui/style_widget/text.dart';
|
||||||
import 'package:flowy_infra_ui/widget/flowy_tooltip.dart';
|
import 'package:flowy_infra_ui/widget/flowy_tooltip.dart';
|
||||||
import 'package:flowy_infra_ui/widget/spacing.dart';
|
import 'package:flowy_infra_ui/widget/spacing.dart';
|
||||||
import 'package:appflowy_backend/protobuf/flowy-user/protobuf.dart'
|
|
||||||
show UserProfilePB;
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:appflowy/generated/locale_keys.g.dart';
|
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
|
||||||
|
|
||||||
class SidebarUser extends StatelessWidget {
|
class SidebarUser extends StatelessWidget {
|
||||||
const SidebarUser({
|
const SidebarUser({
|
||||||
@ -42,7 +42,7 @@ class SidebarUser extends StatelessWidget {
|
|||||||
iconUrl: state.userProfile.iconUrl,
|
iconUrl: state.userProfile.iconUrl,
|
||||||
name: state.userProfile.name,
|
name: state.userProfile.name,
|
||||||
),
|
),
|
||||||
const HSpace(10),
|
const HSpace(4),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: _buildUserName(context, state),
|
child: _buildUserName(context, state),
|
||||||
),
|
),
|
||||||
|
@ -61,7 +61,7 @@ class SettingsMenu extends StatelessWidget {
|
|||||||
SettingsMenuElement(
|
SettingsMenuElement(
|
||||||
page: SettingsPage.cloud,
|
page: SettingsPage.cloud,
|
||||||
selectedPage: currentPage,
|
selectedPage: currentPage,
|
||||||
label: LocaleKeys.settings_menu_cloudSetting.tr(),
|
label: LocaleKeys.settings_menu_cloudSettings.tr(),
|
||||||
icon: Icons.sync,
|
icon: Icons.sync,
|
||||||
changeSelectedPage: changeSelectedPage,
|
changeSelectedPage: changeSelectedPage,
|
||||||
),
|
),
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'dart:convert';
|
|
||||||
|
|
||||||
import 'package:appflowy/env/cloud_env.dart';
|
import 'package:appflowy/env/cloud_env.dart';
|
||||||
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
||||||
import 'package:appflowy/generated/locale_keys.g.dart';
|
import 'package:appflowy/generated/locale_keys.g.dart';
|
||||||
|
import 'package:appflowy/plugins/base/emoji/emoji_picker.dart';
|
||||||
import 'package:appflowy/startup/startup.dart';
|
import 'package:appflowy/startup/startup.dart';
|
||||||
import 'package:appflowy/user/application/auth/auth_service.dart';
|
import 'package:appflowy/user/application/auth/auth_service.dart';
|
||||||
import 'package:appflowy/util/debounce.dart';
|
import 'package:appflowy/util/debounce.dart';
|
||||||
@ -114,21 +114,16 @@ class SettingsUserView extends StatelessWidget {
|
|||||||
fontSize: FontSizes.s16,
|
fontSize: FontSizes.s16,
|
||||||
),
|
),
|
||||||
children: [
|
children: [
|
||||||
SizedBox(
|
Container(
|
||||||
height: 300,
|
height: 380,
|
||||||
width: 300,
|
width: 360,
|
||||||
child: IconGallery(
|
margin: const EdgeInsets.symmetric(horizontal: 12),
|
||||||
defaultOption: _defaultIconOption(context),
|
child: FlowyEmojiPicker(
|
||||||
selectedIcon: user.iconUrl,
|
onEmojiSelected: (_, emoji) {
|
||||||
onSelectIcon: (iconUrl, isSelected) {
|
|
||||||
if (isSelected) {
|
|
||||||
return Navigator.of(context).pop();
|
|
||||||
}
|
|
||||||
|
|
||||||
context
|
context
|
||||||
.read<SettingsUserViewBloc>()
|
.read<SettingsUserViewBloc>()
|
||||||
.add(SettingsUserEvent.updateUserIcon(iconUrl: iconUrl));
|
.add(SettingsUserEvent.updateUserIcon(iconUrl: emoji));
|
||||||
Navigator.of(context).pop();
|
Navigator.of(dialogContext).pop();
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -137,50 +132,6 @@ class SettingsUserView extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns a Widget that is the Default Option for the
|
|
||||||
// Icon Gallery, enabling users to choose the auto-generated
|
|
||||||
// icon again.
|
|
||||||
Widget _defaultIconOption(BuildContext context) {
|
|
||||||
return GestureDetector(
|
|
||||||
behavior: HitTestBehavior.opaque,
|
|
||||||
onTap: () {
|
|
||||||
context
|
|
||||||
.read<SettingsUserViewBloc>()
|
|
||||||
.add(const SettingsUserEvent.removeUserIcon());
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: DecoratedBox(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(6),
|
|
||||||
color: user.iconUrl.isEmpty
|
|
||||||
? Theme.of(context).colorScheme.primary
|
|
||||||
: Colors.transparent,
|
|
||||||
),
|
|
||||||
child: FlowyHover(
|
|
||||||
style: HoverStyle(
|
|
||||||
hoverColor: user.iconUrl.isEmpty
|
|
||||||
? Colors.transparent
|
|
||||||
: Theme.of(context).colorScheme.tertiaryContainer,
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(4.0),
|
|
||||||
child: DecoratedBox(
|
|
||||||
decoration: const BoxDecoration(
|
|
||||||
color: Colors.white,
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
),
|
|
||||||
child: UserAvatar(
|
|
||||||
iconUrl: "",
|
|
||||||
name: user.name,
|
|
||||||
isLarge: true,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Renders either a login or logout button based on the user's authentication status, or nothing if Supabase is not enabled.
|
/// Renders either a login or logout button based on the user's authentication status, or nothing if Supabase is not enabled.
|
||||||
///
|
///
|
||||||
/// This function checks the current user's authentication type and Supabase
|
/// This function checks the current user's authentication type and Supabase
|
||||||
@ -480,6 +431,21 @@ class _AIAccessKeyInputState extends State<_AIAccessKeyInput> {
|
|||||||
|
|
||||||
typedef SelectIconCallback = void Function(String iconUrl, bool isSelected);
|
typedef SelectIconCallback = void Function(String iconUrl, bool isSelected);
|
||||||
|
|
||||||
|
final builtInSVGIcons = [
|
||||||
|
'1F9CC',
|
||||||
|
'1F9DB',
|
||||||
|
'1F9DD-200D-2642-FE0F',
|
||||||
|
'1F9DE-200D-2642-FE0F',
|
||||||
|
'1F9DF',
|
||||||
|
'1F42F',
|
||||||
|
'1F43A',
|
||||||
|
'1F431',
|
||||||
|
'1F435',
|
||||||
|
'1F600',
|
||||||
|
'1F984',
|
||||||
|
];
|
||||||
|
|
||||||
|
// REMOVE this widget in next version 0.3.10
|
||||||
class IconGallery extends StatelessWidget {
|
class IconGallery extends StatelessWidget {
|
||||||
final String selectedIcon;
|
final String selectedIcon;
|
||||||
final SelectIconCallback onSelectIcon;
|
final SelectIconCallback onSelectIcon;
|
||||||
@ -492,52 +458,26 @@ class IconGallery extends StatelessWidget {
|
|||||||
this.defaultOption,
|
this.defaultOption,
|
||||||
});
|
});
|
||||||
|
|
||||||
Future<List<String>> _getIcons(BuildContext context) async {
|
|
||||||
final manifestContent =
|
|
||||||
await DefaultAssetBundle.of(context).loadString('AssetManifest.json');
|
|
||||||
|
|
||||||
final Map<String, dynamic> manifestMap = json.decode(manifestContent);
|
|
||||||
|
|
||||||
final iconUrls = manifestMap.keys
|
|
||||||
.where(
|
|
||||||
(String key) =>
|
|
||||||
key.startsWith('assets/images/emoji/') && key.endsWith('.svg'),
|
|
||||||
)
|
|
||||||
.map((String key) => key.split('/').last.split('.').first)
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
return iconUrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return FutureBuilder<List<String>>(
|
return GridView.count(
|
||||||
future: _getIcons(context),
|
padding: const EdgeInsets.all(20),
|
||||||
builder: (BuildContext context, AsyncSnapshot<List<String>> snapshot) {
|
crossAxisCount: 5,
|
||||||
if (snapshot.hasData && snapshot.data!.isNotEmpty) {
|
mainAxisSpacing: 4,
|
||||||
return GridView.count(
|
crossAxisSpacing: 4,
|
||||||
padding: const EdgeInsets.all(20),
|
children: [
|
||||||
crossAxisCount: 5,
|
if (defaultOption != null) defaultOption!,
|
||||||
mainAxisSpacing: 4,
|
...builtInSVGIcons
|
||||||
crossAxisSpacing: 4,
|
.mapIndexed(
|
||||||
children: [
|
(int index, String iconUrl) => IconOption(
|
||||||
if (defaultOption != null) defaultOption!,
|
emoji: FlowySvgData('emoji/$iconUrl'),
|
||||||
...snapshot.data!
|
iconUrl: iconUrl,
|
||||||
.mapIndexed(
|
onSelectIcon: onSelectIcon,
|
||||||
(int index, String iconUrl) => IconOption(
|
isSelected: iconUrl == selectedIcon,
|
||||||
emoji: FlowySvgData('emoji/$iconUrl'),
|
),
|
||||||
iconUrl: iconUrl,
|
)
|
||||||
onSelectIcon: onSelectIcon,
|
.toList(),
|
||||||
isSelected: iconUrl == selectedIcon,
|
],
|
||||||
),
|
|
||||||
)
|
|
||||||
.toList(),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return const Center(child: CircularProgressIndicator());
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
||||||
import 'package:appflowy/generated/locale_keys.g.dart';
|
import 'package:appflowy/generated/locale_keys.g.dart';
|
||||||
|
import 'package:appflowy/plugins/base/emoji/emoji_text.dart';
|
||||||
import 'package:appflowy/util/color_generator/color_generator.dart';
|
import 'package:appflowy/util/color_generator/color_generator.dart';
|
||||||
|
import 'package:appflowy/workspace/presentation/settings/widgets/settings_user_view.dart';
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
import 'package:flowy_infra/size.dart';
|
import 'package:flowy_infra/size.dart';
|
||||||
import 'package:flowy_infra_ui/style_widget/text.dart';
|
import 'package:flowy_infra_ui/style_widget/text.dart';
|
||||||
@ -66,10 +68,12 @@ class UserAvatar extends StatelessWidget {
|
|||||||
borderRadius: Corners.s5Border,
|
borderRadius: Corners.s5Border,
|
||||||
child: CircleAvatar(
|
child: CircleAvatar(
|
||||||
backgroundColor: Colors.transparent,
|
backgroundColor: Colors.transparent,
|
||||||
child: FlowySvg(
|
child: builtInSVGIcons.contains(iconUrl)
|
||||||
FlowySvgData('emoji/$iconUrl'),
|
? FlowySvg(
|
||||||
blendMode: null,
|
FlowySvgData('emoji/$iconUrl'),
|
||||||
),
|
blendMode: null,
|
||||||
|
)
|
||||||
|
: EmojiText(emoji: iconUrl, fontSize: isLarge ? 36 : 18),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
@ -268,7 +268,7 @@
|
|||||||
"logoutPrompt": "Are you sure to logout?",
|
"logoutPrompt": "Are you sure to logout?",
|
||||||
"selfEncryptionLogoutPrompt": "Are you sure you want to log out? Please ensure you have copied the encryption secret",
|
"selfEncryptionLogoutPrompt": "Are you sure you want to log out? Please ensure you have copied the encryption secret",
|
||||||
"syncSetting": "Sync Setting",
|
"syncSetting": "Sync Setting",
|
||||||
"cloudSetting": "Cloud Setting",
|
"cloudSettings": "Cloud Settings",
|
||||||
"enableSync": "Enable sync",
|
"enableSync": "Enable sync",
|
||||||
"enableEncrypt": "Encrypt data",
|
"enableEncrypt": "Encrypt data",
|
||||||
"cloudURL": "Base URL",
|
"cloudURL": "Base URL",
|
||||||
@ -294,7 +294,7 @@
|
|||||||
"inputEncryptPrompt": "Please enter your encryption secret for",
|
"inputEncryptPrompt": "Please enter your encryption secret for",
|
||||||
"clickToCopySecret": "Click to copy secret",
|
"clickToCopySecret": "Click to copy secret",
|
||||||
"configServerSetting": "Configurate your server settings",
|
"configServerSetting": "Configurate your server settings",
|
||||||
"configServerGuide": "After selecting `Quick Start`, navigate to `Settings` and then \"Cloud Setting\" to configure your self-hosted server.",
|
"configServerGuide": "After selecting `Quick Start`, navigate to `Settings` and then \"Cloud Settings\" to configure your self-hosted server.",
|
||||||
"inputTextFieldHint": "Your secret",
|
"inputTextFieldHint": "Your secret",
|
||||||
"historicalUserList": "User login history",
|
"historicalUserList": "User login history",
|
||||||
"historicalUserListTooltip": "This list displays your anonymous accounts. You can click on an account to view its details. Anonymous accounts are created by clicking the 'Get Started' button",
|
"historicalUserListTooltip": "This list displays your anonymous accounts. You can click on an account to view its details. Anonymous accounts are created by clicking the 'Get Started' button",
|
||||||
@ -1148,6 +1148,7 @@
|
|||||||
"font": "Font",
|
"font": "Font",
|
||||||
"actions": "Actions",
|
"actions": "Actions",
|
||||||
"date": "Date",
|
"date": "Date",
|
||||||
"addField": "Add field"
|
"addField": "Add field",
|
||||||
|
"userIcon": "User icon"
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user