Files
flutter_server_box/lib/view/page/backup.dart
2023-09-25 17:43:45 +08:00

181 lines
4.9 KiB
Dart

import 'dart:async';
import 'dart:io';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:toolbox/core/extension/context/common.dart';
import 'package:toolbox/core/extension/context/dialog.dart';
import 'package:toolbox/core/extension/context/locale.dart';
import 'package:toolbox/core/extension/context/snackbar.dart';
import 'package:toolbox/core/utils/platform/base.dart';
import 'package:toolbox/core/utils/rebuild.dart';
import 'package:toolbox/data/model/app/backup.dart';
import 'package:toolbox/data/res/logger.dart';
import 'package:toolbox/data/res/path.dart';
import 'package:toolbox/data/res/provider.dart';
import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/view/widget/round_rect_card.dart';
import '../../core/utils/misc.dart';
import '../../data/res/ui.dart';
import '../widget/custom_appbar.dart';
import '../widget/store_switch.dart';
class BackupPage extends StatelessWidget {
const BackupPage({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: CustomAppBar(
title: Text(l10n.backupAndRestore, style: UIs.textSize18),
),
body: _buildBody(context),
);
}
Widget _buildBody(BuildContext context) {
final tip = () {
if (isMacOS || isIOS) {
return '${l10n.syncTip}\n${l10n.backupTip}';
}
return l10n.backupTip;
}();
return Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
if (isMacOS || isIOS) _buildIcloudSync(context),
UIs.height13,
Padding(
padding: const EdgeInsets.all(37),
child: Text(
tip,
textAlign: TextAlign.center,
),
),
UIs.height77,
_buildCard(
l10n.restore,
Icons.download,
() => _onRestore(context),
),
UIs.height13,
const SizedBox(
width: 37,
child: Divider(),
),
UIs.height13,
_buildCard(
l10n.backup,
Icons.save,
() async {
await Backup.backup();
await shareFiles([await Paths.bak]);
},
)
],
);
}
Widget _buildCard(
String text,
IconData icon,
FutureOr Function() onTap,
) {
return RoundRectCard(
InkWell(
onTap: onTap,
child: Padding(
padding: const EdgeInsets.symmetric(vertical: 7, horizontal: 17),
child: Row(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(icon, size: 20),
UIs.width7,
Text(text),
],
),
),
),
);
}
Widget _buildIcloudSync(BuildContext context) {
return Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Text(
'iCloud',
textAlign: TextAlign.center,
),
UIs.width13,
// Hive db only save data into local file after app exit,
// so this button is useless
// IconButton(
// onPressed: () async {
// showLoadingDialog(context);
// await ICloud.syncDb();
// context.pop();
// showRestartSnackbar(context, btn: s.restart, msg: s.icloudSynced);
// },
// icon: const Icon(Icons.sync)),
// width13,
StoreSwitch(prop: Stores.setting.icloudSync)
],
);
}
Future<void> _onRestore(BuildContext context) async {
final path = await pickOneFile();
if (path == null) return;
final file = File(path);
if (!await file.exists()) {
context.showSnackBar(l10n.fileNotExist(path));
return;
}
final text = await file.readAsString();
if (text.isEmpty) {
context.showSnackBar(l10n.fieldMustNotEmpty);
return;
}
try {
context.showLoadingDialog();
final backup = await compute(Backup.fromJsonString, text.trim());
if (backupFormatVersion != backup.version) {
context.showSnackBar(l10n.backupVersionNotMatch);
return;
}
await context.showRoundDialog(
title: Text(l10n.restore),
child: Text(l10n.restoreSureWithDate(backup.date)),
actions: [
TextButton(
onPressed: () => context.pop(),
child: Text(l10n.cancel),
),
TextButton(
onPressed: () async {
backup.restore();
context.pop();
RebuildNodes.app.rebuild();
Providers.reload();
},
child: Text(l10n.ok),
),
],
);
} catch (e, trace) {
Loggers.app.warning('Import backup failed', e, trace);
context.showSnackBar(e.toString());
} finally {
context.pop();
}
}
}