418 lines
21 KiB
Dart
418 lines
21 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_localizations/flutter_localizations.dart';
|
|
import 'package:flutter_redux/flutter_redux.dart';
|
|
import 'package:flutter_styled_toast/flutter_styled_toast.dart';
|
|
import 'package:intl/intl.dart';
|
|
import 'package:invoiceninja_flutter/constants.dart';
|
|
import 'package:invoiceninja_flutter/redux/app/app_state.dart';
|
|
import 'package:invoiceninja_flutter/redux/company/company_selectors.dart';
|
|
import 'package:invoiceninja_flutter/redux/ui/pref_state.dart';
|
|
import 'package:invoiceninja_flutter/ui/app/app_builder.dart';
|
|
import 'package:invoiceninja_flutter/ui/app/change_layout_banner.dart';
|
|
import 'package:invoiceninja_flutter/ui/app/main_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/app/screen_imports.dart';
|
|
import 'package:invoiceninja_flutter/ui/auth/init_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/auth/lock_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/auth/login_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/company_gateway/company_gateway_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/company_gateway/company_gateway_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/company_gateway/edit/company_gateway_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/company_gateway/view/company_gateway_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/credit/credit_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/credit/credit_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/credit/edit/credit_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/credit/view/credit_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/design/design_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/design/design_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/design/edit/design_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/design/view/design_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/payment/refund/payment_refund_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/payment_term/edit/payment_term_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/payment_term/payment_term_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/payment_term/payment_term_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/payment_term/view/payment_term_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/reports/reports_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/reports/reports_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/account_management_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/expense_settings_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/online_payments_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/settings_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/task_settings_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/settings/tax_settings_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/tax_rate/edit/tax_rate_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/tax_rate/tax_rate_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/tax_rate/view/tax_rate_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/user/edit/user_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/user/user_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/user/view/user_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/utils/localization.dart';
|
|
import 'package:invoiceninja_flutter/utils/platforms.dart';
|
|
import 'package:local_auth/local_auth.dart';
|
|
import 'package:redux/redux.dart';
|
|
|
|
// STARTER: import - do not remove comment
|
|
import 'package:invoiceninja_flutter/ui/task_status/task_status_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/task_status/edit/task_status_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/task_status/view/task_status_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/task_status/task_status_screen_vm.dart';
|
|
|
|
import 'package:invoiceninja_flutter/ui/expense_category/expense_category_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/expense_category/edit/expense_category_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/expense_category/view/expense_category_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/expense_category/expense_category_screen_vm.dart';
|
|
|
|
import 'package:invoiceninja_flutter/ui/recurring_invoice/recurring_invoice_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/recurring_invoice/edit/recurring_invoice_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/recurring_invoice/view/recurring_invoice_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/recurring_invoice/recurring_invoice_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/webhook/webhook_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/webhook/edit/webhook_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/webhook/view/webhook_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/webhook/webhook_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/token/token_screen.dart';
|
|
import 'package:invoiceninja_flutter/ui/token/edit/token_edit_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/token/view/token_view_vm.dart';
|
|
import 'package:invoiceninja_flutter/ui/token/token_screen_vm.dart';
|
|
import 'package:invoiceninja_flutter/utils/web_stub.dart'
|
|
if (dart.library.html) 'package:invoiceninja_flutter/utils/web.dart';
|
|
|
|
class InvoiceNinjaApp extends StatefulWidget {
|
|
const InvoiceNinjaApp({Key key, this.store}) : super(key: key);
|
|
final Store<AppState> store;
|
|
|
|
@override
|
|
InvoiceNinjaAppState createState() => InvoiceNinjaAppState();
|
|
}
|
|
|
|
class InvoiceNinjaAppState extends State<InvoiceNinjaApp> {
|
|
bool _authenticated = false;
|
|
|
|
Future<Null> _authenticate() async {
|
|
bool authenticated = false;
|
|
|
|
try {
|
|
authenticated = await LocalAuthentication().authenticateWithBiometrics(
|
|
localizedReason: 'Please authenticate to access the app',
|
|
useErrorDialogs: true,
|
|
stickyAuth: false);
|
|
} catch (e) {
|
|
print(e);
|
|
}
|
|
|
|
if (authenticated) {
|
|
setState(() => _authenticated = true);
|
|
}
|
|
}
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
|
|
WebUtils.warnChanges(widget.store);
|
|
}
|
|
|
|
/*
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
|
|
const QuickActions quickActions = QuickActions();
|
|
quickActions.initialize((String shortcutType) {
|
|
if (shortcutType == 'action_new_client') {
|
|
widget.store
|
|
.dispatch(EditClient(context: context, client: ClientEntity()));
|
|
}
|
|
});
|
|
|
|
quickActions.setShortcutItems(<ShortcutItem>[
|
|
const ShortcutItem(
|
|
type: 'action_new_client',
|
|
localizedTitle: 'New Client',
|
|
icon: 'AppIcon'),
|
|
]);
|
|
}
|
|
*/
|
|
|
|
@override
|
|
void didChangeDependencies() {
|
|
final state = widget.store.state;
|
|
if (state.prefState.requireAuthentication && !_authenticated) {
|
|
_authenticate();
|
|
}
|
|
super.didChangeDependencies();
|
|
}
|
|
|
|
Route<dynamic> generateRoute(RouteSettings settings) {
|
|
/*
|
|
print('## generateRoute: ${settings.name}, isInitial: ${settings.isInitialRoute}');
|
|
print('## pathname: ${html5.window.location.pathname} hash: ${html5.window.location.hash}, href: ${html5.window.location.href}');
|
|
html5.window.history.replaceState(null, settings.name, '/#${settings.name}');
|
|
widget.store.dispatch(UpdateCurrentRoute(settings.name));
|
|
*/
|
|
switch (settings.name) {
|
|
case '/login':
|
|
return MaterialPageRoute<dynamic>(builder: (_) => LoginScreen());
|
|
default:
|
|
return MaterialPageRoute<dynamic>(builder: (_) => MainScreen());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return StoreProvider<AppState>(
|
|
store: widget.store,
|
|
child: AppBuilder(builder: (context) {
|
|
final store = widget.store;
|
|
final state = store.state;
|
|
final hasAccentColor = state.hasAccentColor;
|
|
final accentColor = state.accentColor;
|
|
const fontFamily = kIsWeb ? 'Roboto' : null;
|
|
final pageTransitionsTheme = PageTransitionsTheme(builders: {
|
|
TargetPlatform.android: ZoomPageTransitionsBuilder(),
|
|
});
|
|
Intl.defaultLocale = localeSelector(state);
|
|
final locale = AppLocalization.createLocale(localeSelector(state));
|
|
|
|
return StyledToast(
|
|
locale: locale,
|
|
duration: Duration(seconds: 3),
|
|
backgroundColor:
|
|
state.prefState.enableDarkMode ? Colors.white : Colors.black,
|
|
textStyle: TextStyle(
|
|
color:
|
|
state.prefState.enableDarkMode ? Colors.black87 : Colors.white,
|
|
),
|
|
child: MaterialApp(
|
|
supportedLocales: kLanguages
|
|
.map((String locale) => AppLocalization.createLocale(locale))
|
|
.toList(),
|
|
//debugShowCheckedModeBanner: false,
|
|
//showPerformanceOverlay: true,
|
|
localizationsDelegates: [
|
|
const AppLocalizationsDelegate(),
|
|
GlobalCupertinoLocalizations.delegate,
|
|
GlobalWidgetsLocalizations.delegate,
|
|
GlobalMaterialLocalizations.delegate
|
|
],
|
|
home: state.prefState.requireAuthentication && !_authenticated
|
|
? LockScreen(onAuthenticatePressed: _authenticate)
|
|
: InitScreen(),
|
|
locale: locale,
|
|
theme: state.prefState.enableDarkMode
|
|
? ThemeData(
|
|
pageTransitionsTheme: pageTransitionsTheme,
|
|
brightness: Brightness.dark,
|
|
accentColor: accentColor,
|
|
indicatorColor: accentColor,
|
|
textSelectionHandleColor: accentColor,
|
|
fontFamily: fontFamily,
|
|
backgroundColor: Colors.black,
|
|
canvasColor: Colors.black,
|
|
cardColor: const Color(0xFF1B1C1E),
|
|
bottomAppBarColor: const Color(0xFF1B1C1E),
|
|
primaryColorDark: Colors.black,
|
|
buttonColor: accentColor,
|
|
)
|
|
: ThemeData(
|
|
pageTransitionsTheme: pageTransitionsTheme,
|
|
primaryColor: accentColor,
|
|
accentColor: accentColor,
|
|
indicatorColor: accentColor,
|
|
textSelectionColor: accentColor,
|
|
fontFamily: fontFamily,
|
|
backgroundColor: Colors.white,
|
|
canvasColor: Colors.white,
|
|
cardColor: Colors.white,
|
|
bottomAppBarColor: Colors.white,
|
|
primaryColorDark:
|
|
hasAccentColor ? accentColor : const Color(0xFF0D5D91),
|
|
primaryColorLight:
|
|
hasAccentColor ? accentColor : const Color(0xFF5dabf4),
|
|
buttonColor:
|
|
hasAccentColor ? accentColor : const Color(0xFF0D5D91),
|
|
scaffoldBackgroundColor: const Color(0xFFE7EBEE),
|
|
tabBarTheme: TabBarTheme(
|
|
labelColor: hasAccentColor ? Colors.white : Colors.black,
|
|
unselectedLabelColor: hasAccentColor
|
|
? Colors.white.withOpacity(.65)
|
|
: Colors.black.withOpacity(.65),
|
|
),
|
|
/*
|
|
buttonTheme: ButtonThemeData(
|
|
textTheme: ButtonTextTheme.primary,
|
|
colorScheme: ColorScheme.light(
|
|
//primary: hasAccentColor ? Colors.white : Colors.black,
|
|
),
|
|
),
|
|
*/
|
|
iconTheme: IconThemeData(
|
|
color: hasAccentColor ? null : accentColor,
|
|
),
|
|
appBarTheme: AppBarTheme(
|
|
brightness: Brightness.light,
|
|
color: hasAccentColor ? accentColor : Colors.white,
|
|
iconTheme: IconThemeData(
|
|
color: hasAccentColor ? Colors.white : accentColor,
|
|
),
|
|
textTheme: TextTheme(
|
|
headline6: Theme.of(context)
|
|
.textTheme
|
|
.headline6
|
|
.copyWith(
|
|
color:
|
|
hasAccentColor ? Colors.white : Colors.black,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
title: kAppName,
|
|
onGenerateRoute: isMobile(context) ? null : generateRoute,
|
|
routes: isMobile(context)
|
|
? {
|
|
LoginScreen.route: (context) => LoginScreen(),
|
|
MainScreen.route: (context) => MainScreen(),
|
|
DashboardScreenBuilder.route: (context) =>
|
|
ChangeLayoutBanner(
|
|
suggestedLayout: AppLayout.mobile,
|
|
appLayout: state.prefState.appLayout,
|
|
child: DashboardScreenBuilder(),
|
|
),
|
|
ProductScreen.route: (context) => ProductScreenBuilder(),
|
|
ProductViewScreen.route: (context) => ProductViewScreen(),
|
|
ProductEditScreen.route: (context) => ProductEditScreen(),
|
|
ClientScreen.route: (context) => ClientScreenBuilder(),
|
|
ClientViewScreen.route: (context) => ClientViewScreen(),
|
|
ClientEditScreen.route: (context) => ClientEditScreen(),
|
|
InvoiceScreen.route: (context) => InvoiceScreenBuilder(),
|
|
InvoiceViewScreen.route: (context) => InvoiceViewScreen(),
|
|
InvoiceEditScreen.route: (context) => InvoiceEditScreen(),
|
|
InvoiceEmailScreen.route: (context) => InvoiceEmailScreen(),
|
|
DocumentScreen.route: (context) => DocumentScreenBuilder(),
|
|
DocumentViewScreen.route: (context) => DocumentViewScreen(),
|
|
DocumentEditScreen.route: (context) => DocumentEditScreen(),
|
|
ExpenseScreen.route: (context) => ExpenseScreenBuilder(),
|
|
ExpenseViewScreen.route: (context) => ExpenseViewScreen(),
|
|
ExpenseEditScreen.route: (context) => ExpenseEditScreen(),
|
|
VendorScreen.route: (context) => VendorScreenBuilder(),
|
|
VendorViewScreen.route: (context) => VendorViewScreen(),
|
|
VendorEditScreen.route: (context) => VendorEditScreen(),
|
|
TaskScreen.route: (context) => TaskScreenBuilder(),
|
|
TaskViewScreen.route: (context) => TaskViewScreen(),
|
|
TaskEditScreen.route: (context) => TaskEditScreen(),
|
|
ProjectScreen.route: (context) => ProjectScreenBuilder(),
|
|
ProjectViewScreen.route: (context) => ProjectViewScreen(),
|
|
ProjectEditScreen.route: (context) => ProjectEditScreen(),
|
|
PaymentScreen.route: (context) => PaymentScreenBuilder(),
|
|
PaymentViewScreen.route: (context) => PaymentViewScreen(),
|
|
PaymentEditScreen.route: (context) => PaymentEditScreen(),
|
|
PaymentRefundScreen.route: (context) =>
|
|
PaymentRefundScreen(),
|
|
QuoteScreen.route: (context) => QuoteScreenBuilder(),
|
|
QuoteViewScreen.route: (context) => QuoteViewScreen(),
|
|
QuoteEditScreen.route: (context) => QuoteEditScreen(),
|
|
QuoteEmailScreen.route: (context) => QuoteEmailScreen(),
|
|
// STARTER: routes - do not remove comment
|
|
TaskStatusScreen.route: (context) =>
|
|
TaskStatusScreenBuilder(),
|
|
TaskStatusViewScreen.route: (context) =>
|
|
TaskStatusViewScreen(),
|
|
TaskStatusEditScreen.route: (context) =>
|
|
TaskStatusEditScreen(),
|
|
ExpenseCategoryScreen.route: (context) =>
|
|
ExpenseCategoryScreenBuilder(),
|
|
ExpenseCategoryViewScreen.route: (context) =>
|
|
ExpenseCategoryViewScreen(),
|
|
ExpenseCategoryEditScreen.route: (context) =>
|
|
ExpenseCategoryEditScreen(),
|
|
RecurringInvoiceScreen.route: (context) =>
|
|
RecurringInvoiceScreenBuilder(),
|
|
RecurringInvoiceViewScreen.route: (context) =>
|
|
RecurringInvoiceViewScreen(),
|
|
RecurringInvoiceEditScreen.route: (context) =>
|
|
RecurringInvoiceEditScreen(),
|
|
WebhookScreen.route: (context) => WebhookScreenBuilder(),
|
|
WebhookViewScreen.route: (context) => WebhookViewScreen(),
|
|
WebhookEditScreen.route: (context) => WebhookEditScreen(),
|
|
TokenScreen.route: (context) => TokenScreenBuilder(),
|
|
TokenViewScreen.route: (context) => TokenViewScreen(),
|
|
TokenEditScreen.route: (context) => TokenEditScreen(),
|
|
PaymentTermScreen.route: (context) =>
|
|
PaymentTermScreenBuilder(),
|
|
PaymentTermEditScreen.route: (context) =>
|
|
PaymentTermEditScreen(),
|
|
PaymentTermViewScreen.route: (context) =>
|
|
PaymentTermViewScreen(),
|
|
DesignScreen.route: (context) => DesignScreenBuilder(),
|
|
DesignViewScreen.route: (context) => DesignViewScreen(),
|
|
DesignEditScreen.route: (context) => DesignEditScreen(),
|
|
CreditScreen.route: (context) => CreditScreenBuilder(),
|
|
CreditViewScreen.route: (context) => CreditViewScreen(),
|
|
CreditEditScreen.route: (context) => CreditEditScreen(),
|
|
UserScreen.route: (context) => UserScreenBuilder(),
|
|
UserViewScreen.route: (context) => UserViewScreen(),
|
|
UserEditScreen.route: (context) => UserEditScreen(),
|
|
GroupSettingsScreen.route: (context) =>
|
|
GroupScreenBuilder(),
|
|
GroupViewScreen.route: (context) => GroupViewScreen(),
|
|
GroupEditScreen.route: (context) => GroupEditScreen(),
|
|
SettingsScreen.route: (context) => SettingsScreenBuilder(),
|
|
ReportsScreen.route: (context) => ReportsScreenBuilder(),
|
|
CompanyDetailsScreen.route: (context) =>
|
|
CompanyDetailsScreen(),
|
|
UserDetailsScreen.route: (context) => UserDetailsScreen(),
|
|
LocalizationScreen.route: (context) => LocalizationScreen(),
|
|
OnlinePaymentsScreen.route: (context) =>
|
|
OnlinePaymentsScreen(),
|
|
CompanyGatewayScreen.route: (context) =>
|
|
CompanyGatewayScreenBuilder(),
|
|
CompanyGatewayViewScreen.route: (context) =>
|
|
CompanyGatewayViewScreen(),
|
|
CompanyGatewayEditScreen.route: (context) =>
|
|
CompanyGatewayEditScreen(),
|
|
TaxSettingsScreen.route: (context) => TaxSettingsScreen(),
|
|
TaxRateSettingsScreen.route: (context) =>
|
|
TaxRateScreenBuilder(),
|
|
TaxRateViewScreen.route: (context) => TaxRateViewScreen(),
|
|
TaxRateEditScreen.route: (context) => TaxRateEditScreen(),
|
|
ProductSettingsScreen.route: (context) =>
|
|
ProductSettingsScreen(),
|
|
ExpenseSettingsScreen.route: (context) =>
|
|
ExpenseSettingsScreen(),
|
|
TaskSettingsScreen.route: (context) => TaskSettingsScreen(),
|
|
IntegrationSettingsScreen.route: (context) =>
|
|
IntegrationSettingsScreen(),
|
|
ImportExportScreen.route: (context) => ImportExportScreen(),
|
|
DeviceSettingsScreen.route: (context) =>
|
|
DeviceSettingsScreen(),
|
|
AccountManagementScreen.route: (context) =>
|
|
AccountManagementScreen(),
|
|
CustomFieldsScreen.route: (context) => CustomFieldsScreen(),
|
|
GeneratedNumbersScreen.route: (context) =>
|
|
GeneratedNumbersScreen(),
|
|
WorkflowSettingsScreen.route: (context) =>
|
|
WorkflowSettingsScreen(),
|
|
InvoiceDesignScreen.route: (context) =>
|
|
InvoiceDesignScreen(),
|
|
ClientPortalScreen.route: (context) => ClientPortalScreen(),
|
|
BuyNowButtonsScreen.route: (context) =>
|
|
BuyNowButtonsScreen(),
|
|
EmailSettingsScreen.route: (context) =>
|
|
EmailSettingsScreen(),
|
|
TemplatesAndRemindersScreen.route: (context) =>
|
|
TemplatesAndRemindersScreen(),
|
|
CreditCardsAndBanksScreen.route: (context) =>
|
|
CreditCardsAndBanksScreen(),
|
|
DataVisualizationsScreen.route: (context) =>
|
|
DataVisualizationsScreen(),
|
|
}
|
|
: {},
|
|
),
|
|
);
|
|
}),
|
|
);
|
|
}
|
|
}
|