| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184 | import 'package:app_flowy/plugin/plugin.dart';import 'package:app_flowy/workspace/application/home/home_bloc.dart';import 'package:app_flowy/workspace/application/home/home_listen_bloc.dart';import 'package:app_flowy/workspace/presentation/widgets/edit_pannel/pannel_animation.dart';import 'package:app_flowy/workspace/presentation/widgets/float_bubble/question_bubble.dart';import 'package:app_flowy/startup/startup.dart';import 'package:flowy_sdk/log.dart';import 'package:flowy_infra_ui/style_widget/container.dart';import 'package:flowy_sdk/protobuf/flowy-user-data-model/protobuf.dart' show UserProfile;import 'package:flowy_sdk/protobuf/flowy-folder-data-model/protobuf.dart';import 'package:flutter/material.dart';import 'package:flutter_bloc/flutter_bloc.dart';import 'package:styled_widget/styled_widget.dart';import '../widgets/edit_pannel/edit_pannel.dart';import 'home_layout.dart';import 'home_stack.dart';import 'menu/menu.dart';class HomeScreen extends StatefulWidget {  static GlobalKey<ScaffoldState> scaffoldKey = GlobalKey();  final UserProfile user;  final CurrentWorkspaceSetting workspaceSetting;  const HomeScreen(this.user, this.workspaceSetting, {Key? key}) : super(key: key);  @override  State<HomeScreen> createState() => _HomeScreenState();}class _HomeScreenState extends State<HomeScreen> {  View? initialView;  @override  void initState() {    super.initState();  }  @override  void didUpdateWidget(covariant HomeScreen oldWidget) {    initialView = null;    super.didUpdateWidget(oldWidget);  }  @override  Widget build(BuildContext context) {    return MultiBlocProvider(      providers: [        BlocProvider<HomeListenBloc>(          create: (context) => getIt<HomeListenBloc>(param1: widget.user)..add(const HomeListenEvent.started()),        ),        BlocProvider<HomeBloc>(create: (context) => getIt<HomeBloc>()),      ],      child: Scaffold(        key: HomeScreen.scaffoldKey,        body: BlocListener<HomeListenBloc, HomeListenState>(          listener: (context, state) {            state.map(              loading: (_) {},              unauthorized: (unauthorized) {                // TODO: push to login screen when user token was invalid                Log.error("Push to login screen when user token was invalid");              },            );          },          child: BlocBuilder<HomeBloc, HomeState>(            buildWhen: (previous, current) => previous != current,            builder: (context, state) {              final collapasedNotifier = getIt<HomeStackManager>().collapsedNotifier;              collapasedNotifier.addPublishListener((isCollapsed) {                context.read<HomeBloc>().add(HomeEvent.forceCollapse(isCollapsed));              });              return FlowyContainer(                Theme.of(context).colorScheme.surface,                // Colors.white,                child: _buildBody(state, context.read<HomeBloc>().state.forceCollapse),              );            },          ),        ),      ),    );  }  Widget _buildBody(HomeState state, bool forceCollapse) {    return LayoutBuilder(      builder: (BuildContext context, BoxConstraints constraints) {        final layout = HomeLayout(context, constraints, forceCollapse);        const homeStack = HomeStack();        final menu = _buildHomeMenu(          layout: layout,          context: context,        );        final editPannel = _buildEditPannel(          homeState: state,          layout: layout,          context: context,        );        const bubble = QuestionBubble();        return _layoutWidgets(          layout: layout,          homeStack: homeStack,          homeMenu: menu,          editPannel: editPannel,          bubble: bubble,        );      },    );  }  Widget _buildHomeMenu({required HomeLayout layout, required BuildContext context}) {    if (initialView == null && widget.workspaceSetting.hasLatestView()) {      initialView = widget.workspaceSetting.latestView;      final plugin = makePlugin(pluginType: initialView!.pluginType, data: initialView);      getIt<HomeStackManager>().setPlugin(plugin);    }    HomeMenu homeMenu = HomeMenu(      user: widget.user,      workspaceSetting: widget.workspaceSetting,      collapsedNotifier: getIt<HomeStackManager>().collapsedNotifier,    );    return FocusTraversalGroup(child: RepaintBoundary(child: homeMenu));  }  Widget _buildEditPannel({required HomeState homeState, required BuildContext context, required HomeLayout layout}) {    final homeBloc = context.read<HomeBloc>();    return BlocBuilder<HomeBloc, HomeState>(      buildWhen: (previous, current) => previous.pannelContext != current.pannelContext,      builder: (context, state) {        return state.pannelContext.fold(          () => const SizedBox(),          (pannelContext) => FocusTraversalGroup(            child: RepaintBoundary(              child: EditPannel(                pannelContext: pannelContext,                onEndEdit: () => homeBloc.add(const HomeEvent.dismissEditPannel()),              ),            ),          ),        );      },    );  }  Widget _layoutWidgets({    required HomeLayout layout,    required Widget homeMenu,    required Widget homeStack,    required Widget editPannel,    required Widget bubble,  }) {    return Stack(      children: [        homeMenu            .animatedPanelX(              closeX: -layout.menuWidth,              isClosed: !layout.showMenu,            )            .positioned(left: 0, top: 0, width: layout.menuWidth, bottom: 0, animate: true)            .animate(layout.animDuration, Curves.easeOut),        homeStack            .constrained(minWidth: 500)            .positioned(left: layout.homePageLOffset, right: layout.homePageROffset, bottom: 0, top: 0, animate: true)            .animate(layout.animDuration, Curves.easeOut),        bubble            .positioned(              right: 20,              bottom: 16,              animate: true,            )            .animate(layout.animDuration, Curves.easeOut),        editPannel            .animatedPanelX(              duration: layout.animDuration.inMilliseconds * 0.001,              closeX: layout.editPannelWidth,              isClosed: !layout.showEditPannel,            )            .positioned(right: 0, top: 0, bottom: 0, width: layout.editPannelWidth),      ],    );  }}
 |