| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 | import 'package:appflowy/core/frameless_window.dart';import 'package:appflowy/startup/entry_point.dart';import 'package:appflowy/startup/launch_configuration.dart';import 'package:appflowy/startup/startup.dart';import 'package:appflowy/user/application/auth/auth_service.dart';import 'package:appflowy/workspace/application/appearance.dart';import 'package:appflowy/workspace/presentation/settings/widgets/settings_language_view.dart';import 'package:appflowy_popover/appflowy_popover.dart';import 'package:dartz/dartz.dart' as dartz;import 'package:easy_localization/easy_localization.dart';import 'package:flowy_infra/image.dart';import 'package:flowy_infra/language.dart';import 'package:flowy_infra/size.dart';import 'package:flowy_infra_ui/flowy_infra_ui.dart';import 'package:appflowy_backend/dispatch/dispatch.dart';import 'package:appflowy_backend/log.dart';import 'package:appflowy_backend/protobuf/flowy-error/errors.pb.dart';import 'package:appflowy_backend/protobuf/flowy-folder2/protobuf.dart';import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart';import 'package:flutter/material.dart';import 'package:flutter_bloc/flutter_bloc.dart';import 'package:google_fonts/google_fonts.dart';import 'package:url_launcher/url_launcher.dart';import '../../generated/locale_keys.g.dart';import 'folder/folder_widget.dart';import 'router.dart';import 'widgets/background.dart';class SkipLogInScreen extends StatefulWidget {  final AuthRouter router;  final AuthService authService;  const SkipLogInScreen({    Key? key,    required this.router,    required this.authService,  }) : super(key: key);  @override  State<SkipLogInScreen> createState() => _SkipLogInScreenState();}class _SkipLogInScreenState extends State<SkipLogInScreen> {  var _didCustomizeFolder = false;  @override  Widget build(BuildContext context) {    return Scaffold(      appBar: const _SkipLoginMoveWindow(),      body: Center(        child: _renderBody(context),      ),    );  }  Widget _renderBody(BuildContext context) {    final size = MediaQuery.of(context).size;    return Column(      mainAxisAlignment: MainAxisAlignment.center,      crossAxisAlignment: CrossAxisAlignment.center,      children: [        const Spacer(),        FlowyLogoTitle(          title: LocaleKeys.welcomeText.tr(),          logoSize: const Size.square(40),        ),        const VSpace(32),        GoButton(          onPressed: () {            if (_didCustomizeFolder) {              _relaunchAppAndAutoRegister();            } else {              _autoRegister(context);            }          },        ),        const VSpace(32),        SizedBox(          width: size.width * 0.5,          child: FolderWidget(            createFolderCallback: () async {              _didCustomizeFolder = true;            },          ),        ),        const Spacer(),        const SkipLoginPageFooter(),        const VSpace(20),      ],    );  }  Future<void> _autoRegister(BuildContext context) async {    final result = await widget.authService.signUpAsGuest();    result.fold(      (error) {        Log.error(error);      },      (user) {        FolderEventGetCurrentWorkspace().send().then((result) {          _openCurrentWorkspace(context, user, result);        });      },    );  }  Future<void> _relaunchAppAndAutoRegister() async {    await FlowyRunner.run(      FlowyApp(),      integrationEnv(),      config: const LaunchConfiguration(        autoRegistrationSupported: true,      ),    );  }  void _openCurrentWorkspace(    BuildContext context,    UserProfilePB user,    dartz.Either<WorkspaceSettingPB, FlowyError> workspacesOrError,  ) {    workspacesOrError.fold(      (workspaceSetting) {        widget.router            .pushHomeScreenWithWorkSpace(context, user, workspaceSetting);      },      (error) {        Log.error(error);      },    );  }}class SkipLoginPageFooter extends StatelessWidget {  const SkipLoginPageFooter({    super.key,  });  @override  Widget build(BuildContext context) {    // The placeholderWidth should be greater than the longest width of the LanguageSelectorOnWelcomePage    const double placeholderWidth = 180;    return const Padding(      padding: EdgeInsets.symmetric(horizontal: 16),      child: Row(        mainAxisAlignment: MainAxisAlignment.center,        children: [          HSpace(placeholderWidth),          Expanded(child: SubscribeButtons()),          SizedBox(            width: placeholderWidth,            height: 28,            child: Row(              children: [                Spacer(),                LanguageSelectorOnWelcomePage(),              ],            ),          ),        ],      ),    );  }}class SubscribeButtons extends StatelessWidget {  const SubscribeButtons({    super.key,  });  @override  Widget build(BuildContext context) {    return Wrap(      alignment: WrapAlignment.center,      children: [        Row(          mainAxisAlignment: MainAxisAlignment.center,          mainAxisSize: MainAxisSize.min,          children: [            FlowyText.regular(              LocaleKeys.youCanAlso.tr(),              fontSize: FontSizes.s12,            ),            FlowyTextButton(              LocaleKeys.githubStarText.tr(),              padding: const EdgeInsets.symmetric(horizontal: 4),              fontWeight: FontWeight.w500,              fontColor: Theme.of(context).colorScheme.primary,              hoverColor: Colors.transparent,              fillColor: Colors.transparent,              onPressed: () => _launchURL(                'https://github.com/AppFlowy-IO/appflowy',              ),            ),          ],        ),        Row(          mainAxisAlignment: MainAxisAlignment.center,          mainAxisSize: MainAxisSize.min,          children: [            FlowyText.regular(              LocaleKeys.and.tr(),              fontSize: FontSizes.s12,            ),            FlowyTextButton(              LocaleKeys.subscribeNewsletterText.tr(),              padding: const EdgeInsets.symmetric(horizontal: 4.0),              fontWeight: FontWeight.w500,              fontColor: Theme.of(context).colorScheme.primary,              hoverColor: Colors.transparent,              fillColor: Colors.transparent,              onPressed: () => _launchURL('https://www.appflowy.io/blog'),            ),          ],        ),      ],    );  }  Future<void> _launchURL(String url) async {    final uri = Uri.parse(url);    if (await canLaunchUrl(uri)) {      await launchUrl(uri);    } else {      throw 'Could not launch $url';    }  }}class LanguageSelectorOnWelcomePage extends StatelessWidget {  const LanguageSelectorOnWelcomePage({    super.key,  });  @override  Widget build(BuildContext context) {    return AppFlowyPopover(      offset: const Offset(0, -450),      direction: PopoverDirection.bottomWithRightAligned,      child: FlowyButton(        useIntrinsicWidth: true,        text: Row(          mainAxisSize: MainAxisSize.min,          mainAxisAlignment: MainAxisAlignment.end,          children: [            const FlowySvg(              name: 'login/language',              size: Size.square(20),            ),            const HSpace(4),            Builder(              builder: (context) {                final currentLocale =                    context.watch<AppearanceSettingsCubit>().state.locale;                return FlowyText(                  languageFromLocale(currentLocale),                );              },            ),            const FlowySvg(              name: 'home/drop_down_hide',              size: Size.square(20),            ),          ],        ),      ),      popupBuilder: (BuildContext context) {        final easyLocalization = EasyLocalization.of(context);        if (easyLocalization == null) {          return const SizedBox.shrink();        }        final allLocales = easyLocalization.supportedLocales;        return LanguageItemsListView(          allLocales: allLocales,        );      },    );  }}class GoButton extends StatelessWidget {  final VoidCallback onPressed;  const GoButton({    super.key,    required this.onPressed,  });  @override  Widget build(BuildContext context) {    return FlowyTextButton(      LocaleKeys.letsGoButtonText.tr(),      constraints: const BoxConstraints(        maxWidth: 340,        maxHeight: 48,      ),      radius: BorderRadius.circular(12),      mainAxisAlignment: MainAxisAlignment.center,      fontSize: FontSizes.s14,      fontFamily: GoogleFonts.poppins(fontWeight: FontWeight.w500).fontFamily,      padding: const EdgeInsets.symmetric(vertical: 14.0),      onPressed: onPressed,      fillColor: Theme.of(context).colorScheme.primary,      fontColor: Theme.of(context).colorScheme.onPrimary,      hoverColor: Theme.of(context).colorScheme.primaryContainer,    );  }}class _SkipLoginMoveWindow extends StatelessWidget    implements PreferredSizeWidget {  const _SkipLoginMoveWindow();  @override  Widget build(BuildContext context) {    return const Row(      children: [        Expanded(          child: MoveWindowDetector(),        ),      ],    );  }  @override  Size get preferredSize => const Size.fromHeight(55.0);}
 |