| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316 | 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 VSpace(48),        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 Row(      mainAxisAlignment: MainAxisAlignment.center,      mainAxisSize: MainAxisSize.min,      children: [        FlowyText.regular(          LocaleKeys.youCanAlso.tr(),          fontSize: FontSizes.s12,        ),        FlowyTextButton(          LocaleKeys.githubStarText.tr(),          fontWeight: FontWeight.w500,          fontColor: Theme.of(context).colorScheme.primary,          hoverColor: Colors.transparent,          fillColor: Colors.transparent,          onPressed: () => _launchURL(            'https://github.com/AppFlowy-IO/appflowy',          ),        ),        FlowyText.regular(          LocaleKeys.and.tr(),          fontSize: FontSizes.s12,        ),        FlowyTextButton(          LocaleKeys.subscribeNewsletterText.tr(),          overflow: TextOverflow.ellipsis,          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 BlocBuilder<AppearanceSettingsCubit, AppearanceSettingsState>(      builder: (context, state) {        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),                FlowyText(                  languageFromLocale(state.locale),                ),                // const HSpace(4),                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);}
 |