home_screen.dart 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. import 'package:appflowy/plugins/blank/blank.dart';
  2. import 'package:appflowy/startup/plugin/plugin.dart';
  3. import 'package:appflowy/startup/startup.dart';
  4. import 'package:appflowy/workspace/application/appearance.dart';
  5. import 'package:appflowy/workspace/application/home/home_bloc.dart';
  6. import 'package:appflowy/workspace/application/home/home_service.dart';
  7. import 'package:appflowy/workspace/application/home/home_setting_bloc.dart';
  8. import 'package:appflowy/workspace/application/tabs/tabs_bloc.dart';
  9. import 'package:appflowy/workspace/application/view/view_ext.dart';
  10. import 'package:appflowy/workspace/presentation/home/hotkeys.dart';
  11. import 'package:appflowy/workspace/presentation/home/menu/sidebar/sidebar.dart';
  12. import 'package:appflowy/workspace/presentation/widgets/edit_panel/panel_animation.dart';
  13. import 'package:appflowy/workspace/presentation/widgets/float_bubble/question_bubble.dart';
  14. import 'package:appflowy_backend/log.dart';
  15. import 'package:appflowy_backend/protobuf/flowy-folder2/protobuf.dart';
  16. import 'package:appflowy_backend/protobuf/flowy-user/protobuf.dart'
  17. show UserProfilePB;
  18. import 'package:flowy_infra_ui/style_widget/container.dart';
  19. import 'package:flutter/gestures.dart';
  20. import 'package:flutter/material.dart';
  21. import 'package:flutter_bloc/flutter_bloc.dart';
  22. import 'package:styled_widget/styled_widget.dart';
  23. import '../widgets/edit_panel/edit_panel.dart';
  24. import 'home_layout.dart';
  25. import 'home_stack.dart';
  26. class HomeScreen extends StatefulWidget {
  27. final UserProfilePB user;
  28. final WorkspaceSettingPB workspaceSetting;
  29. const HomeScreen(this.user, this.workspaceSetting, {Key? key})
  30. : super(key: key);
  31. @override
  32. State<HomeScreen> createState() => _HomeScreenState();
  33. }
  34. class _HomeScreenState extends State<HomeScreen> {
  35. @override
  36. Widget build(BuildContext context) {
  37. return MultiBlocProvider(
  38. providers: [
  39. BlocProvider<TabsBloc>.value(value: getIt<TabsBloc>()),
  40. BlocProvider<HomeBloc>(
  41. create: (context) {
  42. return HomeBloc(widget.user, widget.workspaceSetting)
  43. ..add(const HomeEvent.initial());
  44. },
  45. ),
  46. BlocProvider<HomeSettingBloc>(
  47. create: (context) {
  48. return HomeSettingBloc(
  49. widget.user,
  50. widget.workspaceSetting,
  51. context.read<AppearanceSettingsCubit>(),
  52. )..add(const HomeSettingEvent.initial());
  53. },
  54. ),
  55. ],
  56. child: HomeHotKeys(
  57. child: Scaffold(
  58. body: MultiBlocListener(
  59. listeners: [
  60. BlocListener<HomeBloc, HomeState>(
  61. listenWhen: (p, c) => p.latestView != c.latestView,
  62. listener: (context, state) {
  63. final view = state.latestView;
  64. if (view != null) {
  65. // Only open the last opened view if the [TabsState.currentPageManager] current opened plugin is blank and the last opened view is not null.
  66. // All opened widgets that display on the home screen are in the form of plugins. There is a list of built-in plugins defined in the [PluginType] enum, including board, grid and trash.
  67. final currentPageManager =
  68. context.read<TabsBloc>().state.currentPageManager;
  69. if (currentPageManager.plugin.pluginType ==
  70. PluginType.blank) {
  71. getIt<TabsBloc>().add(
  72. TabsEvent.openPlugin(
  73. plugin: view.plugin(listenOnViewChanged: true),
  74. ),
  75. );
  76. }
  77. }
  78. },
  79. ),
  80. ],
  81. child: BlocBuilder<HomeSettingBloc, HomeSettingState>(
  82. buildWhen: (previous, current) => previous != current,
  83. builder: (context, state) {
  84. return FlowyContainer(
  85. Theme.of(context).colorScheme.surface,
  86. child: _buildBody(context),
  87. );
  88. },
  89. ),
  90. ),
  91. ),
  92. ),
  93. );
  94. }
  95. Widget _buildBody(BuildContext context) {
  96. return LayoutBuilder(
  97. builder: (BuildContext context, BoxConstraints constraints) {
  98. final layout = HomeLayout(context, constraints);
  99. final homeStack = HomeStack(
  100. layout: layout,
  101. delegate: HomeScreenStackAdaptor(
  102. buildContext: context,
  103. ),
  104. );
  105. final menu = _buildHomeSidebar(
  106. layout: layout,
  107. context: context,
  108. );
  109. final homeMenuResizer = _buildHomeMenuResizer(context: context);
  110. final editPanel = _buildEditPanel(
  111. layout: layout,
  112. context: context,
  113. );
  114. const bubble = QuestionBubble();
  115. return _layoutWidgets(
  116. layout: layout,
  117. homeStack: homeStack,
  118. homeMenu: menu,
  119. editPanel: editPanel,
  120. bubble: bubble,
  121. homeMenuResizer: homeMenuResizer,
  122. );
  123. },
  124. );
  125. }
  126. Widget _buildHomeSidebar({
  127. required HomeLayout layout,
  128. required BuildContext context,
  129. }) {
  130. final workspaceSetting = widget.workspaceSetting;
  131. final homeMenu = HomeSideBar(
  132. user: widget.user,
  133. workspaceSetting: workspaceSetting,
  134. );
  135. return FocusTraversalGroup(child: RepaintBoundary(child: homeMenu));
  136. }
  137. Widget _buildEditPanel({
  138. required BuildContext context,
  139. required HomeLayout layout,
  140. }) {
  141. final homeBloc = context.read<HomeSettingBloc>();
  142. return BlocBuilder<HomeSettingBloc, HomeSettingState>(
  143. buildWhen: (previous, current) =>
  144. previous.panelContext != current.panelContext,
  145. builder: (context, state) {
  146. return state.panelContext.fold(
  147. () => const SizedBox(),
  148. (panelContext) => FocusTraversalGroup(
  149. child: RepaintBoundary(
  150. child: EditPanel(
  151. panelContext: panelContext,
  152. onEndEdit: () =>
  153. homeBloc.add(const HomeSettingEvent.dismissEditPanel()),
  154. ),
  155. ),
  156. ),
  157. );
  158. },
  159. );
  160. }
  161. Widget _buildHomeMenuResizer({
  162. required BuildContext context,
  163. }) {
  164. return MouseRegion(
  165. cursor: SystemMouseCursors.resizeLeftRight,
  166. child: GestureDetector(
  167. dragStartBehavior: DragStartBehavior.down,
  168. onHorizontalDragStart: (details) => context
  169. .read<HomeSettingBloc>()
  170. .add(const HomeSettingEvent.editPanelResizeStart()),
  171. onHorizontalDragUpdate: (details) => context
  172. .read<HomeSettingBloc>()
  173. .add(HomeSettingEvent.editPanelResized(details.localPosition.dx)),
  174. onHorizontalDragEnd: (details) => context
  175. .read<HomeSettingBloc>()
  176. .add(const HomeSettingEvent.editPanelResizeEnd()),
  177. onHorizontalDragCancel: () => context
  178. .read<HomeSettingBloc>()
  179. .add(const HomeSettingEvent.editPanelResizeEnd()),
  180. behavior: HitTestBehavior.translucent,
  181. child: SizedBox(
  182. width: 10,
  183. height: MediaQuery.of(context).size.height,
  184. ),
  185. ),
  186. );
  187. }
  188. Widget _layoutWidgets({
  189. required HomeLayout layout,
  190. required Widget homeMenu,
  191. required Widget homeStack,
  192. required Widget editPanel,
  193. required Widget bubble,
  194. required Widget homeMenuResizer,
  195. }) {
  196. return Stack(
  197. children: [
  198. homeStack
  199. .constrained(minWidth: 500)
  200. .positioned(
  201. left: layout.homePageLOffset,
  202. right: layout.homePageROffset,
  203. bottom: 0,
  204. top: 0,
  205. animate: true,
  206. )
  207. .animate(layout.animDuration, Curves.easeOut),
  208. bubble
  209. .positioned(
  210. right: 20,
  211. bottom: 16,
  212. animate: true,
  213. )
  214. .animate(layout.animDuration, Curves.easeOut),
  215. editPanel
  216. .animatedPanelX(
  217. duration: layout.animDuration.inMilliseconds * 0.001,
  218. closeX: layout.editPanelWidth,
  219. isClosed: !layout.showEditPanel,
  220. )
  221. .positioned(
  222. right: 0,
  223. top: 0,
  224. bottom: 0,
  225. width: layout.editPanelWidth,
  226. ),
  227. homeMenu
  228. .animatedPanelX(
  229. closeX: -layout.menuWidth,
  230. isClosed: !layout.showMenu,
  231. )
  232. .positioned(
  233. left: 0,
  234. top: 0,
  235. width: layout.menuWidth,
  236. bottom: 0,
  237. animate: true,
  238. )
  239. .animate(layout.animDuration, Curves.easeOut),
  240. homeMenuResizer
  241. .positioned(left: layout.menuWidth - 5)
  242. .animate(layout.animDuration, Curves.easeOut),
  243. ],
  244. );
  245. }
  246. }
  247. class HomeScreenStackAdaptor extends HomeStackDelegate {
  248. final BuildContext buildContext;
  249. HomeScreenStackAdaptor({
  250. required this.buildContext,
  251. });
  252. @override
  253. void didDeleteStackWidget(ViewPB view, int? index) {
  254. final homeService = HomeService();
  255. homeService.readApp(appId: view.parentViewId).then((result) {
  256. result.fold(
  257. (parentView) {
  258. final List<ViewPB> views = parentView.childViews;
  259. if (views.isNotEmpty) {
  260. ViewPB lastView = views.last;
  261. if (index != null && index != 0 && views.length > index - 1) {
  262. lastView = views[index - 1];
  263. }
  264. getIt<TabsBloc>().add(
  265. TabsEvent.openPlugin(
  266. plugin: lastView.plugin(listenOnViewChanged: true),
  267. ),
  268. );
  269. } else {
  270. getIt<TabsBloc>().add(
  271. TabsEvent.openPlugin(
  272. plugin: BlankPagePlugin(),
  273. ),
  274. );
  275. }
  276. },
  277. (err) => Log.error(err),
  278. );
  279. });
  280. }
  281. }