menu.dart 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. export './app/header/header.dart';
  2. export './app/menu_app.dart';
  3. import 'dart:io' show Platform;
  4. import 'package:app_flowy/workspace/presentation/home/home_sizes.dart';
  5. import 'package:app_flowy/workspace/presentation/home/home_stack.dart';
  6. import 'package:app_flowy/workspace/presentation/plugins/trash/menu.dart';
  7. import 'package:flowy_infra/notifier.dart';
  8. import 'package:flowy_infra/size.dart';
  9. import 'package:flowy_infra/theme.dart';
  10. import 'package:flowy_infra_ui/style_widget/scrolling/styled_list.dart';
  11. import 'package:flowy_infra_ui/widget/spacing.dart';
  12. import 'package:flowy_sdk/protobuf/flowy-user/protobuf.dart' show UserProfilePB;
  13. import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
  14. import 'package:flowy_sdk/protobuf/flowy-folder/workspace.pb.dart';
  15. import 'package:flutter/material.dart';
  16. import 'package:flutter_bloc/flutter_bloc.dart';
  17. import 'package:styled_widget/styled_widget.dart';
  18. import 'package:expandable/expandable.dart';
  19. import 'package:flowy_infra/time/duration.dart';
  20. import 'package:app_flowy/startup/startup.dart';
  21. import 'package:app_flowy/workspace/application/menu/menu_bloc.dart';
  22. import 'package:app_flowy/workspace/application/home/home_bloc.dart';
  23. import 'package:app_flowy/core/frameless_window.dart';
  24. // import 'package:app_flowy/workspace/presentation/home/home_sizes.dart';
  25. import 'package:flowy_infra/image.dart';
  26. import 'package:flowy_infra_ui/style_widget/icon_button.dart';
  27. import 'app/menu_app.dart';
  28. import 'app/create_button.dart';
  29. import 'menu_user.dart';
  30. class HomeMenu extends StatelessWidget {
  31. final PublishNotifier<bool> _collapsedNotifier;
  32. final UserProfilePB user;
  33. final CurrentWorkspaceSettingPB workspaceSetting;
  34. const HomeMenu({
  35. Key? key,
  36. required this.user,
  37. required this.workspaceSetting,
  38. required PublishNotifier<bool> collapsedNotifier,
  39. }) : _collapsedNotifier = collapsedNotifier,
  40. super(key: key);
  41. @override
  42. Widget build(BuildContext context) {
  43. return MultiBlocProvider(
  44. providers: [
  45. BlocProvider<MenuBloc>(
  46. create: (context) {
  47. final menuBloc = getIt<MenuBloc>(
  48. param1: user, param2: workspaceSetting.workspace.id);
  49. menuBloc.add(const MenuEvent.initial());
  50. return menuBloc;
  51. },
  52. ),
  53. ],
  54. child: MultiBlocListener(
  55. listeners: [
  56. BlocListener<MenuBloc, MenuState>(
  57. listenWhen: (p, c) => p.plugin.id != c.plugin.id,
  58. listener: (context, state) {
  59. getIt<HomeStackManager>().setPlugin(state.plugin);
  60. },
  61. ),
  62. BlocListener<HomeBloc, HomeState>(
  63. listenWhen: (p, c) => p.isMenuCollapsed != c.isMenuCollapsed,
  64. listener: (context, state) {
  65. _collapsedNotifier.value = state.isMenuCollapsed;
  66. },
  67. )
  68. ],
  69. child: BlocBuilder<MenuBloc, MenuState>(
  70. builder: (context, state) => _renderBody(context),
  71. ),
  72. ),
  73. );
  74. }
  75. Widget _renderBody(BuildContext context) {
  76. // nested column: https://siddharthmolleti.com/flutter-box-constraints-nested-column-s-row-s-3dfacada7361
  77. final theme = context.watch<AppTheme>();
  78. return Container(
  79. color: theme.bg1,
  80. child: Column(
  81. mainAxisAlignment: MainAxisAlignment.start,
  82. children: [
  83. Expanded(
  84. child: Column(
  85. mainAxisAlignment: MainAxisAlignment.start,
  86. children: [
  87. const MenuTopBar(),
  88. const VSpace(10),
  89. _renderApps(context),
  90. ],
  91. ).padding(horizontal: Insets.l),
  92. ),
  93. const VSpace(20),
  94. const MenuTrash().padding(horizontal: Insets.l),
  95. const VSpace(20),
  96. _renderNewAppButton(context),
  97. ],
  98. ),
  99. );
  100. }
  101. Widget _renderApps(BuildContext context) {
  102. return ExpandableTheme(
  103. data: ExpandableThemeData(
  104. useInkWell: true, animationDuration: Durations.medium),
  105. child: Expanded(
  106. child: ScrollConfiguration(
  107. behavior: const ScrollBehavior().copyWith(scrollbars: false),
  108. child: BlocSelector<MenuBloc, MenuState, List<Widget>>(
  109. selector: (state) => state.apps
  110. .map((app) => MenuApp(app, key: ValueKey(app.id)))
  111. .toList(),
  112. builder: (context, menuItems) {
  113. return ReorderableListView.builder(
  114. itemCount: menuItems.length,
  115. buildDefaultDragHandles: false,
  116. header: Padding(
  117. padding:
  118. EdgeInsets.only(bottom: 20.0 - MenuAppSizes.appVPadding),
  119. child: MenuUser(user),
  120. ),
  121. onReorder: (oldIndex, newIndex) {
  122. // Moving item1 from index 0 to index 1
  123. // expect: oldIndex: 0, newIndex: 1
  124. // receive: oldIndex: 0, newIndex: 2
  125. // Workaround: if newIndex > oldIndex, we just minus one
  126. int index = newIndex > oldIndex ? newIndex - 1 : newIndex;
  127. context
  128. .read<MenuBloc>()
  129. .add(MenuEvent.moveApp(oldIndex, index));
  130. },
  131. physics: StyledScrollPhysics(),
  132. itemBuilder: (BuildContext context, int index) {
  133. return ReorderableDragStartListener(
  134. key: ValueKey(menuItems[index].key),
  135. index: index,
  136. child: Padding(
  137. padding: EdgeInsets.symmetric(
  138. vertical: MenuAppSizes.appVPadding / 2),
  139. child: menuItems[index],
  140. ),
  141. );
  142. },
  143. );
  144. },
  145. ),
  146. ),
  147. ),
  148. );
  149. }
  150. Widget _renderNewAppButton(BuildContext context) {
  151. return NewAppButton(
  152. press: (appName) =>
  153. context.read<MenuBloc>().add(MenuEvent.createApp(appName, desc: "")),
  154. );
  155. }
  156. }
  157. class MenuSharedState {
  158. final ValueNotifier<ViewPB?> _latestOpenView = ValueNotifier<ViewPB?>(null);
  159. MenuSharedState({ViewPB? view}) {
  160. _latestOpenView.value = view;
  161. }
  162. ViewPB? get latestOpenView => _latestOpenView.value;
  163. set latestOpenView(ViewPB? view) {
  164. if (_latestOpenView.value != view) {
  165. _latestOpenView.value = view;
  166. }
  167. }
  168. VoidCallback addLatestViewListener(void Function(ViewPB?) callback) {
  169. listener() {
  170. callback(_latestOpenView.value);
  171. }
  172. _latestOpenView.addListener(listener);
  173. return listener;
  174. }
  175. void removeLatestViewListener(VoidCallback listener) {
  176. _latestOpenView.removeListener(listener);
  177. }
  178. }
  179. class MenuTopBar extends StatelessWidget {
  180. const MenuTopBar({Key? key}) : super(key: key);
  181. Widget renderIcon(BuildContext context) {
  182. if (Platform.isMacOS) {
  183. return Container();
  184. }
  185. final theme = context.watch<AppTheme>();
  186. return (theme.isDark
  187. ? svgWithSize("flowy_logo_dark_mode", const Size(92, 17))
  188. : svgWithSize("flowy_logo_with_text", const Size(92, 17)));
  189. }
  190. @override
  191. Widget build(BuildContext context) {
  192. final theme = context.watch<AppTheme>();
  193. return BlocBuilder<MenuBloc, MenuState>(
  194. builder: (context, state) {
  195. return SizedBox(
  196. height: HomeSizes.topBarHeight,
  197. child: MoveWindowDetector(
  198. child: Row(
  199. children: [
  200. renderIcon(context),
  201. const Spacer(),
  202. Tooltip(
  203. richMessage: TextSpan(children: [
  204. const TextSpan(text: "Close sidebar\n"),
  205. TextSpan(
  206. text: Platform.isMacOS ? "⌘+\\" : "Ctrl+\\",
  207. style: const TextStyle(color: Colors.white60),
  208. ),
  209. ]),
  210. child: FlowyIconButton(
  211. width: 28,
  212. onPressed: () => context
  213. .read<HomeBloc>()
  214. .add(const HomeEvent.collapseMenu()),
  215. iconPadding: const EdgeInsets.fromLTRB(4, 4, 4, 4),
  216. icon: svgWidget("home/hide_menu", color: theme.iconColor),
  217. ))
  218. ],
  219. )),
  220. );
  221. },
  222. );
  223. }
  224. }