/* * This file is part of Stack Wallet. * * Copyright (c) 2023 Cypher Stack * All Rights Reserved. * The code is distributed under GPLv3 license, see LICENSE file for details. * Generated by Cypher Stack on 2023-05-26 * */ import 'dart:io'; import 'package:flutter/gestures.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/svg.dart'; import 'package:url_launcher/url_launcher.dart'; import '../app_config.dart'; import '../pages_desktop_specific/password/create_password_view.dart'; import '../themes/stack_colors.dart'; import '../themes/theme_providers.dart'; import '../utilities/prefs.dart'; import '../utilities/text_styles.dart'; import '../utilities/util.dart'; import '../widgets/app_icon.dart'; import '../widgets/background.dart'; import '../widgets/desktop/secondary_button.dart'; import 'stack_privacy_calls.dart'; class IntroView extends ConsumerStatefulWidget { const IntroView({super.key}); static const String routeName = "/introView"; @override ConsumerState createState() => _IntroViewState(); } class _IntroViewState extends ConsumerState { late final bool isDesktop; @override void initState() { isDesktop = Util.isDesktop; super.initState(); } @override Widget build(BuildContext context) { debugPrint("BUILD: $runtimeType "); return Background( child: Scaffold( backgroundColor: Theme.of(context).extension()!.background, body: Center( child: !isDesktop ? Column( crossAxisAlignment: CrossAxisAlignment.center, children: [ const Spacer( flex: 2, ), Padding( padding: const EdgeInsets.symmetric( horizontal: 16, ), child: ConstrainedBox( constraints: const BoxConstraints( maxWidth: 300, ), child: SvgPicture.file( File( ref.watch( themeProvider.select( (value) => value.assets.stack, ), ), ), width: isDesktop ? 324 : 266, height: isDesktop ? 324 : 266, ), ), ), const Spacer( flex: 1, ), AppNameText( isDesktop: isDesktop, ), const SizedBox( height: 8, ), Padding( padding: const EdgeInsets.symmetric( horizontal: 48, ), child: IntroAboutText( isDesktop: isDesktop, ), ), const Spacer( flex: 4, ), Padding( padding: const EdgeInsets.symmetric( horizontal: 32, ), child: PrivacyAndTOSText( isDesktop: isDesktop, ), ), Padding( padding: const EdgeInsets.symmetric( horizontal: 16, vertical: 16, ), child: Row( children: [ Expanded( child: GetStartedButton( isDesktop: isDesktop, ), ), ], ), ), ], ) : SizedBox( width: 350, height: 540, child: Column( children: [ const Spacer( flex: 2, ), const SizedBox( width: 130, height: 130, child: AppIcon(), ), const Spacer( flex: 42, ), AppNameText( isDesktop: isDesktop, ), const Spacer( flex: 24, ), IntroAboutText( isDesktop: isDesktop, ), const Spacer( flex: 42, ), GetStartedButton( isDesktop: isDesktop, ), if (isDesktop) const SizedBox( height: 20, ), if (isDesktop) SecondaryButton( label: "Restore from Stack backup", onPressed: () { Navigator.of(context).pushNamed( CreatePasswordView.routeName, arguments: true, ); }, ), const Spacer( flex: 65, ), PrivacyAndTOSText( isDesktop: isDesktop, ), ], ), ), ), ), ); } } class AppNameText extends StatelessWidget { const AppNameText({Key? key, required this.isDesktop}) : super(key: key); final bool isDesktop; @override Widget build(BuildContext context) { return Text( AppConfig.appName, // TODO [prio=high]: appName is now static due to `'appName' can't be // accessed using static access.`", check that that doesn't cause issues // elsewhere. textAlign: TextAlign.center, style: !isDesktop ? STextStyles.pageTitleH1(context) : STextStyles.pageTitleH1(context).copyWith( fontSize: 40, ), ); } } class IntroAboutText extends StatelessWidget { const IntroAboutText({Key? key, required this.isDesktop}) : super(key: key); final bool isDesktop; @override Widget build(BuildContext context) { return Text( "An open-source, multicoin wallet for everyone", textAlign: TextAlign.center, style: !isDesktop ? STextStyles.subtitle(context) : STextStyles.subtitle(context).copyWith( fontSize: 24, ), ); } } class PrivacyAndTOSText extends StatelessWidget { const PrivacyAndTOSText({Key? key, required this.isDesktop}) : super(key: key); final bool isDesktop; @override Widget build(BuildContext context) { final fontSize = isDesktop ? 18.0 : 12.0; return RichText( textAlign: TextAlign.center, text: TextSpan( style: STextStyles.label(context).copyWith(fontSize: fontSize), children: [ TextSpan(text: "By using ${AppConfig.appName}, you agree to the "), TextSpan( text: "Terms of service", style: STextStyles.richLink(context).copyWith(fontSize: fontSize), recognizer: TapGestureRecognizer() ..onTap = () { launchUrl( Uri.parse("https://stackwallet.com/terms-of-service.html"), mode: LaunchMode.externalApplication, ); }, ), const TextSpan(text: " and "), TextSpan( text: "Privacy policy", style: STextStyles.richLink(context).copyWith(fontSize: fontSize), recognizer: TapGestureRecognizer() ..onTap = () { launchUrl( Uri.parse("https://stackwallet.com/privacy-policy.html"), mode: LaunchMode.externalApplication, ); }, ), ], ), ); } } class GetStartedButton extends StatelessWidget { const GetStartedButton({Key? key, required this.isDesktop}) : super(key: key); final bool isDesktop; @override Widget build(BuildContext context) { return !isDesktop ? TextButton( style: Theme.of(context) .extension()! .getPrimaryEnabledButtonStyle(context), onPressed: () { Prefs.instance.externalCalls = true; Navigator.of(context).pushNamed( StackPrivacyCalls.routeName, arguments: false, ); }, child: Text( "Get started", style: STextStyles.button(context), ), ) : SizedBox( width: double.infinity, height: 70, child: TextButton( style: Theme.of(context) .extension()! .getPrimaryEnabledButtonStyle(context), onPressed: () { Navigator.of(context).pushNamed( StackPrivacyCalls.routeName, arguments: false, ); }, child: Text( "Create new Stack", style: STextStyles.button(context).copyWith(fontSize: 20), ), ), ); } }