Files
eh_downloader_flutter/lib/globals.dart
2023-11-26 15:12:51 +08:00

356 lines
9.4 KiB
Dart

import 'dart:io';
import 'package:cookie_jar/cookie_jar.dart';
import 'package:dio/dio.dart';
import 'package:dio_cookie_manager/dio_cookie_manager.dart';
import 'package:event_listener/event_listener.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';
import 'package:flutter/services.dart'
show ApplicationSwitcherDescription, SystemChrome;
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:go_router/go_router.dart';
import 'package:logging/logging.dart';
import 'package:path/path.dart' as path;
import 'package:path_provider/path_provider.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:window_manager/window_manager.dart';
import 'api/client.dart';
import 'auth.dart';
import 'config/base.dart';
import 'config/shared_preferences.dart';
import 'config/windows.dart';
import 'main.dart';
import 'platform/clipboard.dart';
import 'platform/display.dart';
import 'platform/path.dart';
import 'platform/set_title.dart';
import 'tags.dart';
import 'utils.dart';
export 'galleries.dart' show GalleriesPageExtra;
export 'gallery.dart' show GalleryPageExtra;
final dio = Dio()
..options.validateStatus = (int? _) {
return true;
}
..options.extra['withCredentials'] = true;
Config? _prefs;
EHApi? _api;
Future<String> _getJarPath() async {
if (isWindows || isLinux) {
try {
final p = await platformPath.getCurrentExe();
if (p != null) {
return path.join(path.dirname(p), "cookies");
}
} catch (e) {
// Do nothing
}
}
final Directory appDocDir = await getApplicationDocumentsDirectory();
final String appDocPath = appDocDir.path;
return '$appDocPath/.eh-cookies/';
}
Future<void> prepareJar() async {
final jar = PersistCookieJar(storage: FileStorage(await _getJarPath()));
dio.interceptors.add(CookieManager(jar));
}
Future<void> preparePrefs() async {
if (isWindows || isLinux) {
try {
var tmp = WindowsConfig();
tmp.reload();
_prefs = tmp;
return;
} catch (e) {
// Do nothing.
}
}
_prefs = SharedPreferencesConfig(await SharedPreferences.getInstance());
}
Config get prefs {
if (_prefs == null) {
throw Exception('SharedPreferences not initialized');
}
return _prefs!;
}
void initApi(String baseUrl) {
_api = EHApi(dio, baseUrl: baseUrl);
}
bool tryInitApi(BuildContext context) {
bool? skipBaseUrl = const bool.fromEnvironment("skipBaseUrl");
if (skipBaseUrl == true) {
if (_api != null) {
return true;
}
initApi("${Uri.base.origin}/api/");
clearAllStates(context);
return true;
}
String? baseUrl = prefs.getString("baseUrl");
if (baseUrl == null) {
SchedulerBinding.instance.addPostFrameCallback((_) {
context.go("/set_server");
});
return false;
}
if (_api != null && _api!.baseUrl == baseUrl) {
return true;
}
initApi(baseUrl);
clearAllStates(context);
return true;
}
bool get apiInited {
return _api != null;
}
EHApi get api {
if (_api == null) {
throw Exception('EHApi not initialized');
}
return _api!;
}
final AuthInfo auth = AuthInfo();
final Clipboard platformClipboard = Clipboard();
final Display platformDisplay = Display();
final Path platformPath = Path();
final TagsInfo tags = TagsInfo();
final GlobalKey<ScaffoldMessengerState> rootScaffoldMessengerKey =
GlobalKey<ScaffoldMessengerState>();
final EventListener listener = EventListener();
enum MoreVertSettings {
setServerUrl,
createRootUser,
settings,
}
void onMoreVertSettingsSelected(BuildContext context, MoreVertSettings value) {
switch (value) {
case MoreVertSettings.setServerUrl:
context.push("/set_server");
break;
case MoreVertSettings.createRootUser:
context.push("/create_root_user");
break;
case MoreVertSettings.settings:
context.push("/settings");
break;
default:
break;
}
}
List<PopupMenuEntry<MoreVertSettings>> buildMoreVertSettings(
BuildContext context) {
var list = <PopupMenuEntry<MoreVertSettings>>[];
var path = GoRouterState.of(context).path;
if (const bool.fromEnvironment("skipBaseUrl") != true &&
path != "/set_server") {
list.add(PopupMenuItem(
value: MoreVertSettings.setServerUrl,
child: Text(AppLocalizations.of(context)!.setServerUrl)));
}
if (auth.status != null &&
auth.status!.noUser &&
prefs.getBool("skipCreateRootUser") == true &&
path != "/create_root_user") {
list.add(PopupMenuItem(
value: MoreVertSettings.createRootUser,
child: Text(AppLocalizations.of(context)!.createRootUser)));
}
if (path != "/settings") {
list.add(PopupMenuItem(
value: MoreVertSettings.settings,
child: Text(AppLocalizations.of(context)!.settings)));
}
var showNsfw = prefs.getBool("showNsfw") ?? false;
list.add(PopupMenuItem(
child: StatefulBuilder(
builder: (context, setState) => CheckboxListTile(
controlAffinity: ListTileControlAffinity.leading,
value: showNsfw,
onChanged: (value) {
if (value != null) {
prefs.setBool("showNsfw", value);
listener.emit("showNsfwChanged", null);
setState(() {
showNsfw = value;
});
}
},
title: Text(AppLocalizations.of(context)!.showNsfw),
),
)));
var displayAd = prefs.getBool("displayAd") ?? false;
list.add(PopupMenuItem(
child: StatefulBuilder(
builder: (context, setState) => CheckboxListTile(
controlAffinity: ListTileControlAffinity.leading,
value: displayAd,
onChanged: (value) {
if (value != null) {
prefs.setBool("displayAd", value);
listener.emit("displayAdChanged", null);
setState(() {
displayAd = value;
});
}
},
title: Text(AppLocalizations.of(context)!.displayAd),
),
)));
return list;
}
Widget buildMoreVertSettingsButon(BuildContext context) {
return PopupMenuButton(
icon: const Icon(Icons.more_vert),
onSelected: (MoreVertSettings value) {
onMoreVertSettingsSelected(context, value);
},
itemBuilder: buildMoreVertSettings,
);
}
ThemeMode themeModeNext(ThemeMode mode) {
if (mode == ThemeMode.system) return ThemeMode.light;
if (mode == ThemeMode.dark) return ThemeMode.system;
return ThemeMode.dark;
}
mixin ThemeModeWidget<T extends StatefulWidget> on State<T> {
@protected
Widget buildThemeModeIcon(BuildContext context) {
final mode = MainApp.of(context).themeMode;
return IconButton(
onPressed: () {
final n = themeModeNext(mode);
MainApp.of(context).changeThemeMode(n);
setState(() {});
},
icon: Icon(mode == ThemeMode.system
? Icons.brightness_auto
: mode == ThemeMode.dark
? Icons.dark_mode
: Icons.light_mode));
}
}
mixin IsTopWidget on Widget {
@protected
bool isTop(BuildContext context) {
final last = GoRouter.of(context)
.routerDelegate
.currentConfiguration
.matches
.last
.pageKey;
return last == key;
}
}
mixin IsTopWidget2<T extends StatefulWidget> on State<T> {
@protected
bool isTop(BuildContext context) {
final last = GoRouter.of(context)
.routerDelegate
.currentConfiguration
.matches
.last
.pageKey;
return last == widget.key;
}
}
enum Lang {
system("System"),
english("English"),
simplifiedChinese("简体中文");
const Lang(String lang) : langName = lang;
final String langName;
Locale toLocale() {
switch (this) {
case Lang.english:
return const Locale("en");
case Lang.simplifiedChinese:
return const Locale("zh");
default:
return PlatformDispatcher.instance.locale;
}
}
}
final _authLog = Logger("AuthLog");
void clearAllStates(BuildContext context) {
auth.clear();
tags.clear();
checkAuth(context);
}
void checkAuth(BuildContext context) {
if (!auth.isAuthed && !auth.checked && !auth.isChecking) {
auth.checkAuth().then((re) {
if (!re) {
if (auth.status!.noUser && prefs.getBool("skipCreateRootUser") == true)
return;
context.push(auth.status!.noUser ? "/create_root_user" : "/login");
}
}).catchError((err) {
_authLog.log(Level.SEVERE, "Failed to check auth info:", err);
});
}
}
String? _currentTitle;
String? _prefix;
final _titleLog = Logger("Title");
void setCurrentTitle(String title, int primaryColor,
{bool isPrefix = false,
bool includePrefix = true,
bool usePrefix = false}) {
if (!isPrefix && includePrefix && _prefix != null) {
title = "$_prefix - $title";
}
if (usePrefix) {
if (_prefix == null) return;
title = _prefix!;
}
if (_currentTitle != null && title == _currentTitle) return;
if (isDesktop) {
windowManager.setTitle(title).then((_) {
_currentTitle = title;
if (isPrefix) _prefix = title;
}).catchError((err) {
_titleLog.warning("Failed to set title:", err);
});
} else if (kIsWeb) {
setTitleWeb(title);
_currentTitle = title;
if (isPrefix) _prefix = title;
} else {
SystemChrome.setApplicationSwitcherDescription(
ApplicationSwitcherDescription(
label: title, primaryColor: primaryColor))
.then((_) {
_currentTitle = title;
if (isPrefix) _prefix = title;
}).catchError((err) {
_titleLog.warning("Failed to set title:", err);
});
}
}