2024-10-06 19:29:47 +08:00
|
|
|
import 'dart:convert';
|
|
|
|
import 'dart:io';
|
|
|
|
|
|
|
|
import 'package:dropdown_button2/dropdown_button2.dart';
|
2024-07-31 02:44:49 +08:00
|
|
|
import 'package:flutter/material.dart';
|
2024-10-06 19:29:47 +08:00
|
|
|
import 'package:gap/gap.dart';
|
2024-07-31 02:44:49 +08:00
|
|
|
import 'package:get/get.dart';
|
2024-10-06 19:29:47 +08:00
|
|
|
import 'package:image_picker/image_picker.dart';
|
2024-09-24 22:10:45 +08:00
|
|
|
import 'package:in_app_review/in_app_review.dart';
|
2024-10-06 19:29:47 +08:00
|
|
|
import 'package:path_provider/path_provider.dart';
|
2024-07-31 22:48:22 +08:00
|
|
|
import 'package:provider/provider.dart';
|
2024-07-31 02:44:49 +08:00
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
2024-09-21 22:44:08 +08:00
|
|
|
import 'package:solian/exceptions/request.dart';
|
2024-07-31 02:44:49 +08:00
|
|
|
import 'package:solian/exts.dart';
|
2024-10-06 19:29:47 +08:00
|
|
|
import 'package:solian/models/theme.dart';
|
2024-09-15 15:55:14 +08:00
|
|
|
import 'package:solian/platform.dart';
|
2024-09-21 22:44:08 +08:00
|
|
|
import 'package:solian/providers/auth.dart';
|
2024-09-15 12:25:50 +08:00
|
|
|
import 'package:solian/providers/database/database.dart';
|
2024-07-31 22:48:22 +08:00
|
|
|
import 'package:solian/providers/theme_switcher.dart';
|
2024-08-02 00:29:51 +08:00
|
|
|
import 'package:solian/router.dart';
|
2024-09-21 22:44:08 +08:00
|
|
|
import 'package:solian/widgets/reports/abuse_report.dart';
|
2024-07-31 02:44:49 +08:00
|
|
|
|
|
|
|
class SettingScreen extends StatefulWidget {
|
|
|
|
const SettingScreen({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<SettingScreen> createState() => _SettingScreenState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SettingScreenState extends State<SettingScreen> {
|
2024-09-15 15:55:14 +08:00
|
|
|
SharedPreferences? _prefs;
|
2024-10-06 19:29:47 +08:00
|
|
|
String _docBasepath = '/';
|
2024-07-31 02:44:49 +08:00
|
|
|
|
|
|
|
Widget _buildCaptionHeader(String title) {
|
|
|
|
return Container(
|
|
|
|
width: MediaQuery.of(context).size.width,
|
|
|
|
color: Theme.of(context).colorScheme.surfaceContainer,
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
|
|
|
|
child: Text(title),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-10-06 19:29:47 +08:00
|
|
|
static final List<SolianThemeData> _presentTheme = [
|
|
|
|
SolianThemeData(
|
|
|
|
id: 'themeColorRed',
|
|
|
|
seedColor: const Color.fromRGBO(154, 98, 91, 1),
|
|
|
|
),
|
|
|
|
SolianThemeData(
|
|
|
|
id: 'themeColorBlue',
|
|
|
|
seedColor: const Color.fromRGBO(103, 96, 193, 1),
|
|
|
|
),
|
|
|
|
SolianThemeData(
|
|
|
|
id: 'themeColorMiku',
|
|
|
|
seedColor: const Color.fromRGBO(56, 120, 126, 1),
|
|
|
|
),
|
|
|
|
SolianThemeData(
|
|
|
|
id: 'themeColorKagamine',
|
|
|
|
seedColor: const Color.fromRGBO(244, 183, 63, 1),
|
|
|
|
),
|
|
|
|
SolianThemeData(
|
|
|
|
id: 'themeColorLuka',
|
|
|
|
seedColor: const Color.fromRGBO(243, 174, 218, 1),
|
|
|
|
),
|
2024-07-31 13:29:26 +08:00
|
|
|
];
|
|
|
|
|
2024-07-31 02:44:49 +08:00
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
2024-10-06 19:29:47 +08:00
|
|
|
getApplicationDocumentsDirectory().then((dir) {
|
|
|
|
_docBasepath = dir.path;
|
|
|
|
if (mounted) {
|
|
|
|
setState(() {});
|
|
|
|
}
|
|
|
|
});
|
2024-07-31 02:44:49 +08:00
|
|
|
SharedPreferences.getInstance().then((inst) {
|
|
|
|
_prefs = inst;
|
2024-09-15 15:55:14 +08:00
|
|
|
if (mounted) {
|
|
|
|
setState(() {});
|
|
|
|
}
|
2024-07-31 02:44:49 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-10-06 23:06:33 +08:00
|
|
|
return ListView(
|
|
|
|
children: [
|
|
|
|
_buildCaptionHeader('theme'.tr),
|
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.palette),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('globalTheme'.tr),
|
|
|
|
trailing: DropdownButtonHideUnderline(
|
|
|
|
child: DropdownButton2<SolianThemeData>(
|
|
|
|
isExpanded: true,
|
|
|
|
hint: Text(
|
|
|
|
'theme'.tr,
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
color: Theme.of(context).hintColor,
|
2024-10-06 19:29:47 +08:00
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
),
|
|
|
|
items: _presentTheme
|
|
|
|
.map((SolianThemeData item) =>
|
|
|
|
DropdownMenuItem<SolianThemeData>(
|
|
|
|
value: item,
|
|
|
|
child: Row(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
|
|
children: [
|
|
|
|
Icon(Icons.circle, color: item.seedColor),
|
|
|
|
const Gap(8),
|
|
|
|
Expanded(
|
|
|
|
child: Text(
|
2024-10-06 19:29:47 +08:00
|
|
|
item.id.tr,
|
2024-10-06 23:06:33 +08:00
|
|
|
maxLines: 1,
|
|
|
|
overflow: TextOverflow.ellipsis,
|
2024-10-06 19:29:47 +08:00
|
|
|
style: const TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
),
|
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
))
|
|
|
|
.toList(),
|
|
|
|
value: (_prefs?.containsKey('global_theme') ?? false)
|
|
|
|
? SolianThemeData.fromJson(
|
|
|
|
jsonDecode(_prefs!.getString('global_theme')!),
|
|
|
|
)
|
|
|
|
: null,
|
|
|
|
onChanged: (SolianThemeData? value) {
|
|
|
|
context.read<ThemeSwitcher>().setThemeData(value);
|
|
|
|
setState(() {});
|
|
|
|
},
|
|
|
|
buttonStyleData: const ButtonStyleData(
|
|
|
|
padding: EdgeInsets.symmetric(horizontal: 8),
|
|
|
|
height: 40,
|
|
|
|
width: 140,
|
|
|
|
),
|
|
|
|
menuItemStyleData: const MenuItemStyleData(
|
|
|
|
height: 40,
|
2024-10-06 19:29:47 +08:00
|
|
|
),
|
|
|
|
),
|
2024-07-31 02:44:49 +08:00
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
),
|
|
|
|
CheckboxListTile(
|
|
|
|
secondary: const Icon(Icons.military_tech),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('agedTheme'.tr),
|
|
|
|
subtitle: Text('agedThemeDesc'.tr),
|
|
|
|
value: _prefs?.getBool('aged_theme') ?? false,
|
|
|
|
onChanged: (value) {
|
|
|
|
if (value != null) {
|
|
|
|
context.read<ThemeSwitcher>().setAgedTheme(value);
|
|
|
|
}
|
|
|
|
setState(() {});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
if (!PlatformInfo.isWeb)
|
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.wallpaper),
|
|
|
|
contentPadding: const EdgeInsets.only(left: 22, right: 31),
|
|
|
|
title: Text('appBackgroundImage'.tr),
|
|
|
|
subtitle: Text('appBackgroundImageDesc'.tr),
|
|
|
|
trailing: File('$_docBasepath/app_background_image').existsSync()
|
|
|
|
? const Icon(Icons.check_box)
|
|
|
|
: const Icon(Icons.check_box_outline_blank),
|
|
|
|
onTap: () async {
|
|
|
|
if (File('$_docBasepath/app_background_image').existsSync()) {
|
|
|
|
File('$_docBasepath/app_background_image').deleteSync();
|
|
|
|
} else {
|
|
|
|
final image = await ImagePicker().pickImage(
|
|
|
|
source: ImageSource.gallery,
|
|
|
|
);
|
|
|
|
if (image == null) return;
|
|
|
|
|
|
|
|
await File(image.path)
|
|
|
|
.copy('$_docBasepath/app_background_image');
|
2024-10-06 19:29:47 +08:00
|
|
|
}
|
2024-10-06 23:06:33 +08:00
|
|
|
|
2024-10-06 19:29:47 +08:00
|
|
|
setState(() {});
|
|
|
|
},
|
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
_buildCaptionHeader('notification'.tr),
|
|
|
|
Tooltip(
|
|
|
|
message: 'settingsNotificationBgServiceDesc'.tr,
|
|
|
|
child: CheckboxListTile(
|
2024-09-17 21:37:20 +08:00
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
2024-10-06 23:06:33 +08:00
|
|
|
secondary: const Icon(Icons.system_security_update_warning),
|
|
|
|
enabled: PlatformInfo.isAndroid,
|
|
|
|
title: Text('settingsNotificationBgService'.tr),
|
|
|
|
subtitle: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
2024-09-21 22:44:08 +08:00
|
|
|
children: [
|
2024-10-06 23:06:33 +08:00
|
|
|
Text('holdToSeeDetail'.tr),
|
|
|
|
Text(
|
|
|
|
'needRestartToApply'.tr,
|
|
|
|
style: const TextStyle(fontWeight: FontWeight.bold),
|
|
|
|
)
|
2024-09-21 22:44:08 +08:00
|
|
|
],
|
2024-10-06 23:06:33 +08:00
|
|
|
),
|
|
|
|
value: _prefs?.getBool('service_background_notification') ?? false,
|
2024-10-06 17:31:44 +08:00
|
|
|
onChanged: (value) {
|
|
|
|
_prefs
|
2024-10-06 23:06:33 +08:00
|
|
|
?.setBool('service_background_notification', value ?? false)
|
2024-10-06 17:31:44 +08:00
|
|
|
.then((_) {
|
|
|
|
setState(() {});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
),
|
|
|
|
_buildCaptionHeader('update'.tr),
|
|
|
|
CheckboxListTile(
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
secondary: const Icon(Icons.sync_alt),
|
|
|
|
title: Text('updateCheckStrictly'.tr),
|
|
|
|
subtitle: Text('updateCheckStrictlyDesc'.tr),
|
|
|
|
value: _prefs?.getBool('check_update_strictly') ?? false,
|
|
|
|
onChanged: (value) {
|
|
|
|
_prefs?.setBool('check_update_strictly', value ?? false).then((_) {
|
|
|
|
setState(() {});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
Obx(() {
|
|
|
|
final AuthProvider auth = Get.find<AuthProvider>();
|
|
|
|
if (!auth.isAuthorized.value) return const SizedBox.shrink();
|
|
|
|
return Column(
|
|
|
|
children: [
|
|
|
|
_buildCaptionHeader('account'.tr),
|
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.flag),
|
|
|
|
trailing: const Icon(Icons.chevron_right),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('reportAbuse'.tr),
|
|
|
|
subtitle: Text('reportAbuseDesc'.tr),
|
|
|
|
onTap: () {
|
|
|
|
showDialog(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => const AbuseReportDialog(),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.person_remove),
|
|
|
|
trailing: const Icon(Icons.chevron_right),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('accountDeletion'.tr),
|
|
|
|
subtitle: Text('accountDeletionDesc'.tr),
|
|
|
|
onTap: () {
|
|
|
|
context
|
|
|
|
.showSlideToConfirmDialog(
|
|
|
|
'accountDeletionConfirm'.tr,
|
|
|
|
'accountDeletionConfirmDesc'.trParams({
|
|
|
|
'account': '@${auth.userProfile.value!['name']}',
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
.then((value) async {
|
|
|
|
if (value != true) return;
|
|
|
|
final client = await auth.configureClient('id');
|
|
|
|
final resp = await client.post('/users/me/deletion', {});
|
|
|
|
if (resp.statusCode != 200) {
|
|
|
|
context.showErrorDialog(RequestException(resp));
|
|
|
|
} else {
|
|
|
|
context.showSnackbar('accountDeletionRequested'.tr);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
_buildCaptionHeader('performance'.tr),
|
|
|
|
CheckboxListTile(
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
secondary: const Icon(Icons.message),
|
|
|
|
title: Text('animatedMessageList'.tr),
|
|
|
|
subtitle: Text('animatedMessageListDesc'.tr),
|
|
|
|
value: _prefs?.getBool('non_animated_message_list') ?? false,
|
|
|
|
onChanged: (value) {
|
|
|
|
_prefs
|
|
|
|
?.setBool('non_animated_message_list', value ?? false)
|
|
|
|
.then((_) {
|
|
|
|
setState(() {});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
_buildCaptionHeader('more'.tr),
|
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.delete_sweep),
|
|
|
|
trailing: const Icon(Icons.chevron_right),
|
|
|
|
subtitle: FutureBuilder(
|
|
|
|
future: AppDatabase.getDatabaseSize(),
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
if (!snapshot.hasData) {
|
2024-09-15 12:25:50 +08:00
|
|
|
return Text('localDatabaseSize'.trParams(
|
2024-10-06 23:06:33 +08:00
|
|
|
{'size': 'unknown'.tr},
|
2024-09-15 12:25:50 +08:00
|
|
|
));
|
2024-10-06 23:06:33 +08:00
|
|
|
}
|
|
|
|
return Text('localDatabaseSize'.trParams(
|
|
|
|
{'size': snapshot.data!.formatBytes()},
|
|
|
|
));
|
2024-09-13 23:19:33 +08:00
|
|
|
},
|
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('localDatabaseWipe'.tr),
|
|
|
|
onTap: () {
|
|
|
|
AppDatabase.removeDatabase().then((_) {
|
|
|
|
setState(() {});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
if (PlatformInfo.canRateTheApp)
|
2024-09-13 23:19:33 +08:00
|
|
|
ListTile(
|
2024-10-06 23:06:33 +08:00
|
|
|
leading: const Icon(Icons.star),
|
2024-09-13 23:19:33 +08:00
|
|
|
trailing: const Icon(Icons.chevron_right),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
2024-10-06 23:06:33 +08:00
|
|
|
title: Text('rateTheApp'.tr),
|
|
|
|
subtitle: Text('rateTheAppDesc'.tr),
|
2024-09-13 23:19:33 +08:00
|
|
|
onTap: () {
|
2024-10-06 23:06:33 +08:00
|
|
|
final inAppReview = InAppReview.instance;
|
|
|
|
|
|
|
|
inAppReview.openStoreListing(
|
|
|
|
appStoreId: '6499032345',
|
|
|
|
);
|
2024-09-13 23:19:33 +08:00
|
|
|
},
|
|
|
|
),
|
2024-10-06 23:06:33 +08:00
|
|
|
ListTile(
|
|
|
|
leading: const Icon(Icons.info_outline),
|
|
|
|
trailing: const Icon(Icons.chevron_right),
|
|
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 22),
|
|
|
|
title: Text('about'.tr),
|
|
|
|
onTap: () {
|
|
|
|
AppRouter.instance.pushNamed('about');
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
2024-07-31 02:44:49 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|