import 'dart:convert'; import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:obtainium/components/custom_app_bar.dart'; import 'package:obtainium/components/generated_form.dart'; import 'package:obtainium/components/generated_form_modal.dart'; import 'package:obtainium/custom_errors.dart'; import 'package:obtainium/providers/apps_provider.dart'; import 'package:obtainium/providers/settings_provider.dart'; import 'package:obtainium/providers/source_provider.dart'; import 'package:provider/provider.dart'; import 'package:file_picker/file_picker.dart'; import 'package:url_launcher/url_launcher_string.dart'; class ImportExportPage extends StatefulWidget { const ImportExportPage({super.key}); @override State createState() => _ImportExportPageState(); } class _ImportExportPageState extends State { bool importInProgress = false; @override Widget build(BuildContext context) { SourceProvider sourceProvider = SourceProvider(); var settingsProvider = context.read(); var appsProvider = context.read(); var outlineButtonStyle = ButtonStyle( shape: MaterialStateProperty.all( StadiumBorder( side: BorderSide( width: 1, color: Theme.of(context).colorScheme.primary, ), ), ), ); Future>> addApps(List urls) async { await settingsProvider.getInstallPermission(); List results = await sourceProvider.getApps(urls, ignoreUrls: appsProvider.apps.values.map((e) => e.app.url).toList()); List apps = results[0]; Map errorsMap = results[1]; for (var app in apps) { if (appsProvider.apps.containsKey(app.id)) { errorsMap.addAll({app.id: 'App already added'}); } else { await appsProvider.saveApps([app]); } } List> errors = errorsMap.keys.map((e) => [e, errorsMap[e].toString()]).toList(); return errors; } return Scaffold( backgroundColor: Theme.of(context).colorScheme.surface, body: CustomScrollView(slivers: [ const CustomAppBar(title: 'Import/Export'), SliverFillRemaining( child: Padding( padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 16), child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Row( children: [ Expanded( child: TextButton( style: outlineButtonStyle, onPressed: appsProvider.apps.isEmpty || importInProgress ? null : () { HapticFeedback.selectionClick(); appsProvider .exportApps() .then((String path) { showError( 'Exported to $path', context); }); }, child: const Text('Obtainium Export'))), const SizedBox( width: 16, ), Expanded( child: TextButton( style: outlineButtonStyle, onPressed: importInProgress ? null : () { HapticFeedback.selectionClick(); FilePicker.platform .pickFiles() .then((result) { setState(() { importInProgress = true; }); if (result != null) { String data = File( result.files.single.path!) .readAsStringSync(); try { jsonDecode(data); } catch (e) { throw ObtainiumError( 'Invalid input'); } appsProvider .importApps(data) .then((value) { showError( '$value App${value == 1 ? '' : 's'} Imported', context); }); } else { // User canceled the picker } }).catchError((e) { showError(e, context); }).whenComplete(() { setState(() { importInProgress = false; }); }); }, child: const Text('Obtainium Import'))) ], ), if (importInProgress) Column( children: const [ SizedBox( height: 14, ), LinearProgressIndicator(), SizedBox( height: 14, ), ], ) else const Divider( height: 32, ), TextButton( onPressed: importInProgress ? null : () { showDialog( context: context, builder: (BuildContext ctx) { return GeneratedFormModal( title: 'Import from URL List', items: [ [ GeneratedFormItem( label: 'App URL List', max: 7, additionalValidators: [ (String? value) { if (value != null && value.isNotEmpty) { var lines = value .trim() .split('\n'); for (int i = 0; i < lines.length; i++) { try { sourceProvider .getSource( lines[i]); } catch (e) { return 'Line ${i + 1}: $e'; } } } return null; } ]) ] ], defaultValues: const [], ); }).then((values) { if (values != null) { var urls = (values[0] as String).split('\n'); setState(() { importInProgress = true; }); addApps(urls).then((errors) { if (errors.isEmpty) { showError( 'Imported ${urls.length} Apps', context); } else { showDialog( context: context, builder: (BuildContext ctx) { return ImportErrorDialog( urlsLength: urls.length, errors: errors); }); } }).catchError((e) { showError(e, context); }).whenComplete(() { setState(() { importInProgress = false; }); }); } }); }, child: const Text( 'Import from URL List', )), ...sourceProvider.sources .where((element) => element.canSearch) .map((source) => Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ const SizedBox(height: 8), TextButton( onPressed: importInProgress ? null : () { () async { var values = await showDialog< List>( context: context, builder: (BuildContext ctx) { return GeneratedFormModal( title: 'Search ${source.runtimeType}', items: [ [ GeneratedFormItem( label: '${source.runtimeType} Search Query') ] ], defaultValues: const [], ); }); if (values != null && values[0].isNotEmpty) { setState(() { importInProgress = true; }); var urlsWithDescriptions = await source .search(values[0]); if (urlsWithDescriptions .isNotEmpty) { var selectedUrls = await showDialog< List< String>?>( context: context, builder: (BuildContext ctx) { return UrlSelectionModal( urlsWithDescriptions: urlsWithDescriptions, defaultSelected: false, ); }); if (selectedUrls != null && selectedUrls .isNotEmpty) { var errors = await addApps( selectedUrls); if (errors.isEmpty) { // ignore: use_build_context_synchronously showError( 'Imported ${selectedUrls.length} Apps', context); } else { showDialog( context: context, builder: (BuildContext ctx) { return ImportErrorDialog( urlsLength: selectedUrls .length, errors: errors); }); } } } else { throw ObtainiumError( 'No results found'); } } }() .catchError((e) { showError(e, context); }).whenComplete(() { setState(() { importInProgress = false; }); }); }, child: Text( 'Search ${source.runtimeType}')) ])) .toList(), ...sourceProvider.massUrlSources .map((source) => Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ const SizedBox(height: 8), TextButton( onPressed: importInProgress ? null : () { () async { var values = await showDialog( context: context, builder: (BuildContext ctx) { return GeneratedFormModal( title: 'Import ${source.name}', items: source .requiredArgs .map( (e) => [ GeneratedFormItem(label: e) ]) .toList(), defaultValues: const [], ); }); if (values != null) { setState(() { importInProgress = true; }); var urls = await source .getUrls(values); var selectedUrls = await showDialog< List?>( context: context, builder: (BuildContext ctx) { return UrlSelectionModal( urlsWithDescriptions: Map.fromIterable( urls, value: (item) => '')); }); if (selectedUrls != null) { var errors = await addApps( selectedUrls); if (errors.isEmpty) { // ignore: use_build_context_synchronously showError( 'Imported ${selectedUrls.length} Apps', context); } else { showDialog( context: context, builder: (BuildContext ctx) { return ImportErrorDialog( urlsLength: selectedUrls .length, errors: errors); }); } } } }() .catchError((e) { showError(e, context); }).whenComplete(() { setState(() { importInProgress = false; }); }); }, child: Text('Import ${source.name}')) ])) .toList(), const Spacer(), const Divider( height: 32, ), const Text( 'Imported Apps may incorrectly show as "Not Installed".\nTo fix this, re-install them through Obtainium.\nThis should not affect App data.\n\nOnly affects URL and third-party import methods.', textAlign: TextAlign.center, style: TextStyle( fontStyle: FontStyle.italic, fontSize: 12)), const SizedBox( height: 8, ) ], ))) ])); } } class ImportErrorDialog extends StatefulWidget { const ImportErrorDialog( {super.key, required this.urlsLength, required this.errors}); final int urlsLength; final List> errors; @override State createState() => _ImportErrorDialogState(); } class _ImportErrorDialogState extends State { @override Widget build(BuildContext context) { return AlertDialog( scrollable: true, title: const Text('Import Errors'), content: Column(crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Text( '${widget.urlsLength - widget.errors.length} of ${widget.urlsLength} Apps imported.', style: Theme.of(context).textTheme.bodyLarge, ), const SizedBox(height: 16), Text( 'The following URLs had errors:', style: Theme.of(context).textTheme.bodyLarge, ), ...widget.errors.map((e) { return Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ const SizedBox( height: 16, ), Text(e[0]), Text( e[1], style: const TextStyle(fontStyle: FontStyle.italic), ) ]); }).toList() ]), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(null); }, child: const Text('Okay')) ], ); } } // ignore: must_be_immutable class UrlSelectionModal extends StatefulWidget { UrlSelectionModal( {super.key, required this.urlsWithDescriptions, this.defaultSelected = true}); Map urlsWithDescriptions; bool defaultSelected; @override State createState() => _UrlSelectionModalState(); } class _UrlSelectionModalState extends State { Map, bool> urlWithDescriptionSelections = {}; @override void initState() { super.initState(); for (var url in widget.urlsWithDescriptions.entries) { urlWithDescriptionSelections.putIfAbsent( url, () => widget.defaultSelected); } } @override Widget build(BuildContext context) { return AlertDialog( scrollable: true, title: const Text('Select URLs to Import'), content: Column(children: [ ...urlWithDescriptionSelections.keys.map((urlWithD) { return Row(children: [ Checkbox( value: urlWithDescriptionSelections[urlWithD], onChanged: (value) { setState(() { urlWithDescriptionSelections[urlWithD] = value ?? false; }); }), const SizedBox( width: 8, ), Expanded( child: Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisAlignment: MainAxisAlignment.center, children: [ const SizedBox( height: 8, ), GestureDetector( onTap: () { launchUrlString(urlWithD.key, mode: LaunchMode.externalApplication); }, child: Text( Uri.parse(urlWithD.key).path.substring(1), style: const TextStyle(decoration: TextDecoration.underline), textAlign: TextAlign.start, )), Text( urlWithD.value.length > 128 ? '${urlWithD.value.substring(0, 128)}...' : urlWithD.value, style: const TextStyle( fontStyle: FontStyle.italic, fontSize: 12), ), const SizedBox( height: 8, ) ], )) ]); }) ]), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(); }, child: const Text('Cancel')), TextButton( onPressed: () { Navigator.of(context).pop(urlWithDescriptionSelections.entries .where((entry) => entry.value) .map((e) => e.key.key) .toList()); }, child: Text( 'Import ${urlWithDescriptionSelections.values.where((b) => b).length} URLs')) ], ); } }