home_screen.dart 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. import 'package:app_flowy/startup/plugin/plugin.dart';
  2. import 'package:app_flowy/workspace/application/home/home_bloc.dart';
  3. import 'package:app_flowy/workspace/application/view/view_ext.dart';
  4. import 'package:app_flowy/workspace/presentation/widgets/edit_panel/panel_animation.dart';
  5. import 'package:app_flowy/workspace/presentation/widgets/float_bubble/question_bubble.dart';
  6. import 'package:app_flowy/startup/startup.dart';
  7. import 'package:flowy_sdk/log.dart';
  8. import 'package:flowy_infra_ui/style_widget/container.dart';
  9. import 'package:flowy_sdk/protobuf/flowy-user/protobuf.dart' show UserProfilePB;
  10. import 'package:flowy_sdk/protobuf/flowy-folder/protobuf.dart';
  11. import 'package:flutter/gestures.dart';
  12. import 'package:flutter/material.dart';
  13. import 'package:flutter_bloc/flutter_bloc.dart';
  14. import 'package:styled_widget/styled_widget.dart';
  15. import '../widgets/edit_panel/edit_panel.dart';
  16. import 'home_layout.dart';
  17. import 'home_stack.dart';
  18. import 'menu/menu.dart';
  19. class HomeScreen extends StatefulWidget {
  20. final UserProfilePB user;
  21. final CurrentWorkspaceSettingPB workspaceSetting;
  22. const HomeScreen(this.user, this.workspaceSetting, {Key? key})
  23. : super(key: key);
  24. @override
  25. State<HomeScreen> createState() => _HomeScreenState();
  26. }
  27. class _HomeScreenState extends State<HomeScreen> {
  28. ViewPB? initialView;
  29. @override
  30. void initState() {
  31. super.initState();
  32. }
  33. @override
  34. void didUpdateWidget(covariant HomeScreen oldWidget) {
  35. initialView = null;
  36. super.didUpdateWidget(oldWidget);
  37. }
  38. @override
  39. Widget build(BuildContext context) {
  40. return MultiBlocProvider(
  41. providers: [
  42. BlocProvider<HomeBloc>(
  43. create: (context) {
  44. return HomeBloc(widget.user, widget.workspaceSetting)
  45. ..add(const HomeEvent.initial());
  46. },
  47. ),
  48. ],
  49. child: Scaffold(
  50. body: BlocListener<HomeBloc, HomeState>(
  51. listenWhen: (p, c) => p.unauthorized != c.unauthorized,
  52. listener: (context, state) {
  53. if (state.unauthorized) {
  54. Log.error("Push to login screen when user token was invalid");
  55. }
  56. },
  57. child: BlocBuilder<HomeBloc, HomeState>(
  58. buildWhen: (previous, current) => previous != current,
  59. builder: (context, state) {
  60. final collapsedNotifier =
  61. getIt<HomeStackManager>().collapsedNotifier;
  62. collapsedNotifier.addPublishListener((isCollapsed) {
  63. context
  64. .read<HomeBloc>()
  65. .add(HomeEvent.forceCollapse(isCollapsed));
  66. });
  67. return FlowyContainer(
  68. Theme.of(context).colorScheme.surface,
  69. // Colors.white,
  70. child: _buildBody(state),
  71. );
  72. },
  73. ),
  74. ),
  75. ),
  76. );
  77. }
  78. Widget _buildBody(HomeState state) {
  79. return LayoutBuilder(
  80. builder: (BuildContext context, BoxConstraints constraints) {
  81. final layout = HomeLayout(context, constraints, state.forceCollapse);
  82. final homeStack = HomeStack(
  83. layout: layout,
  84. );
  85. final menu = _buildHomeMenu(
  86. layout: layout,
  87. context: context,
  88. state: state,
  89. );
  90. final homeMenuResizer = _buildHomeMenuResizer(context: context);
  91. final editPanel = _buildEditPanel(
  92. homeState: state,
  93. layout: layout,
  94. context: context,
  95. );
  96. const bubble = QuestionBubble();
  97. return _layoutWidgets(
  98. layout: layout,
  99. homeStack: homeStack,
  100. homeMenu: menu,
  101. editPanel: editPanel,
  102. bubble: bubble,
  103. homeMenuResizer: homeMenuResizer,
  104. );
  105. },
  106. );
  107. }
  108. Widget _buildHomeMenu(
  109. {required HomeLayout layout,
  110. required BuildContext context,
  111. required HomeState state}) {
  112. final workspaceSetting = state.workspaceSetting;
  113. if (initialView == null && workspaceSetting.hasLatestView()) {
  114. initialView = workspaceSetting.latestView;
  115. final plugin = makePlugin(
  116. pluginType: initialView!.pluginType,
  117. data: initialView,
  118. );
  119. getIt<HomeStackManager>().setPlugin(plugin);
  120. }
  121. HomeMenu homeMenu = HomeMenu(
  122. user: widget.user,
  123. workspaceSetting: workspaceSetting,
  124. collapsedNotifier: getIt<HomeStackManager>().collapsedNotifier,
  125. );
  126. final latestView =
  127. workspaceSetting.hasLatestView() ? workspaceSetting.latestView : null;
  128. if (getIt<MenuSharedState>().latestOpenView == null) {
  129. /// AppFlowy will open the view that the last time the user opened it. The _buildHomeMenu will get called when AppFlowy's screen resizes. So we only set the latestOpenView when it's null.
  130. getIt<MenuSharedState>().latestOpenView = latestView;
  131. }
  132. return FocusTraversalGroup(child: RepaintBoundary(child: homeMenu));
  133. }
  134. Widget _buildEditPanel(
  135. {required HomeState homeState,
  136. required BuildContext context,
  137. required HomeLayout layout}) {
  138. final homeBloc = context.read<HomeBloc>();
  139. return BlocBuilder<HomeBloc, HomeState>(
  140. buildWhen: (previous, current) =>
  141. previous.panelContext != current.panelContext,
  142. builder: (context, state) {
  143. return state.panelContext.fold(
  144. () => const SizedBox(),
  145. (panelContext) => FocusTraversalGroup(
  146. child: RepaintBoundary(
  147. child: EditPanel(
  148. panelContext: panelContext,
  149. onEndEdit: () =>
  150. homeBloc.add(const HomeEvent.dismissEditPanel()),
  151. ),
  152. ),
  153. ),
  154. );
  155. },
  156. );
  157. }
  158. Widget _buildHomeMenuResizer({
  159. required BuildContext context,
  160. }) {
  161. return MouseRegion(
  162. cursor: SystemMouseCursors.resizeLeftRight,
  163. child: GestureDetector(
  164. dragStartBehavior: DragStartBehavior.down,
  165. onPanUpdate: ((details) {
  166. context
  167. .read<HomeBloc>()
  168. .add(HomeEvent.editPanelResized(details.delta.dx));
  169. }),
  170. behavior: HitTestBehavior.translucent,
  171. child: SizedBox(
  172. width: 10,
  173. height: MediaQuery.of(context).size.height,
  174. )),
  175. );
  176. }
  177. Widget _layoutWidgets({
  178. required HomeLayout layout,
  179. required Widget homeMenu,
  180. required Widget homeStack,
  181. required Widget editPanel,
  182. required Widget bubble,
  183. required Widget homeMenuResizer,
  184. }) {
  185. return Stack(
  186. children: [
  187. homeMenu
  188. .animatedPanelX(
  189. closeX: -layout.menuWidth,
  190. isClosed: !layout.showMenu,
  191. )
  192. .positioned(
  193. left: 0,
  194. top: 0,
  195. width: layout.menuWidth,
  196. bottom: 0,
  197. animate: true)
  198. .animate(layout.animDuration, Curves.easeOut),
  199. homeStack
  200. .constrained(minWidth: 500)
  201. .positioned(
  202. left: layout.homePageLOffset,
  203. right: layout.homePageROffset,
  204. bottom: 0,
  205. top: 0,
  206. animate: true)
  207. .animate(layout.animDuration, Curves.easeOut),
  208. homeMenuResizer.positioned(left: layout.homePageLOffset - 5),
  209. bubble
  210. .positioned(
  211. right: 20,
  212. bottom: 16,
  213. animate: true,
  214. )
  215. .animate(layout.animDuration, Curves.easeOut),
  216. editPanel
  217. .animatedPanelX(
  218. duration: layout.animDuration.inMilliseconds * 0.001,
  219. closeX: layout.editPanelWidth,
  220. isClosed: !layout.showEditPanel,
  221. )
  222. .positioned(
  223. right: 0, top: 0, bottom: 0, width: layout.editPanelWidth),
  224. ],
  225. );
  226. }
  227. }