mirror of
https://mirror.ghproxy.com/https://github.com/StarCitizenToolBox/app.git
synced 2025-01-11 05:43:41 +08:00
204 lines
6.3 KiB
Dart
204 lines
6.3 KiB
Dart
import 'dart:async';
|
|
import 'dart:convert';
|
|
|
|
import 'package:dart_rss/domain/rss_item.dart';
|
|
import 'package:fluent_ui/fluent_ui.dart';
|
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
|
import 'package:hive/hive.dart';
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
import 'package:starcitizen_doctor/api/api.dart';
|
|
import 'package:starcitizen_doctor/api/rss.dart';
|
|
import 'package:starcitizen_doctor/common/conf/url_conf.dart';
|
|
import 'package:starcitizen_doctor/common/io/rs_http.dart';
|
|
import 'package:starcitizen_doctor/common/utils/async.dart';
|
|
import 'package:starcitizen_doctor/common/utils/base_utils.dart';
|
|
import 'package:starcitizen_doctor/common/utils/log.dart';
|
|
import 'package:starcitizen_doctor/common/utils/provider.dart';
|
|
import 'package:starcitizen_doctor/data/app_placard_data.dart';
|
|
import 'package:starcitizen_doctor/data/app_web_localization_versions_data.dart';
|
|
import 'package:starcitizen_doctor/data/countdown_festival_item_data.dart';
|
|
import 'package:html/parser.dart' as html;
|
|
import 'package:html/dom.dart' as html_dom;
|
|
import 'package:url_launcher/url_launcher_string.dart';
|
|
|
|
import 'localization/localization_ui_model.dart';
|
|
|
|
part 'home_ui_model.freezed.dart';
|
|
|
|
part 'home_ui_model.g.dart';
|
|
|
|
@freezed
|
|
class HomeUIModelState with _$HomeUIModelState {
|
|
factory HomeUIModelState({
|
|
AppPlacardData? appPlacardData,
|
|
@Default(false) bool isFixing,
|
|
@Default("") String isFixingString,
|
|
String? scInstalledPath,
|
|
@Default([]) List<String> scInstallPaths,
|
|
AppWebLocalizationVersionsData? webLocalizationVersionsData,
|
|
@Default("") String lastScreenInfo,
|
|
List<RssItem>? rssVideoItems,
|
|
List<RssItem>? rssTextItems,
|
|
MapEntry<String, bool>? localizationUpdateInfo,
|
|
List? scServerStatus,
|
|
List<CountdownFestivalItemData>? countdownFestivalListData,
|
|
@Default({}) Map<String, bool> isGameRunning,
|
|
}) = _HomeUIModelState;
|
|
}
|
|
|
|
extension HomeUIModelStateEx on HomeUIModelState {
|
|
bool get isCurGameRunning => isGameRunning[scInstalledPath] ?? false;
|
|
}
|
|
|
|
@riverpod
|
|
class HomeUIModel extends _$HomeUIModel {
|
|
@override
|
|
HomeUIModelState build() {
|
|
state = HomeUIModelState();
|
|
_init();
|
|
_loadData();
|
|
return state;
|
|
}
|
|
|
|
closePlacard() async {
|
|
final box = await Hive.openBox("app_conf");
|
|
await box.put("close_placard", state.appPlacardData?.version);
|
|
state = state.copyWith(appPlacardData: null);
|
|
}
|
|
|
|
Future<void> reScanPath() async {
|
|
state = state.copyWith(scInstalledPath: "not_install", lastScreenInfo: "");
|
|
}
|
|
|
|
String getRssImage(RssItem item) {
|
|
final h = html.parse(item.description ?? "");
|
|
if (h.body == null) return "";
|
|
for (var node in h.body!.nodes) {
|
|
if (node is html_dom.Element) {
|
|
if (node.localName == "img") {
|
|
var image = node.attributes["src"]?.trim() ?? "";
|
|
var updatedImage = image.replaceAllMapped(
|
|
RegExp(r'http(s)?://i(\d+)\.hdslb\.com/bfs/'),
|
|
(match) => 'https://web-proxy.scbox.xkeyc.cn/bfs${match[2]}/bfs/'
|
|
);
|
|
return updatedImage;
|
|
}
|
|
}
|
|
}
|
|
return "";
|
|
}
|
|
|
|
String handleTitle(String? title) {
|
|
if (title == null) return "";
|
|
title = title.replaceAll("【", "[ ");
|
|
title = title.replaceAll("】", " ] ");
|
|
return title;
|
|
}
|
|
|
|
// ignore: avoid_build_context_in_providers
|
|
Future<void> goWebView(BuildContext context, String title, String url,
|
|
{bool useLocalization = false, bool loginMode = false}) async {
|
|
launchUrlString(url);
|
|
}
|
|
|
|
bool isRSIServerStatusOK(Map map) {
|
|
return (map["status"] == "ok" || map["status"] == "operational");
|
|
}
|
|
|
|
Timer? _serverUpdateTimer;
|
|
Timer? _appUpdateTimer;
|
|
|
|
void _init() {
|
|
reScanPath();
|
|
_serverUpdateTimer = Timer.periodic(
|
|
const Duration(minutes: 10),
|
|
(timer) {
|
|
_updateSCServerStatus();
|
|
},
|
|
);
|
|
|
|
_appUpdateTimer = Timer.periodic(const Duration(minutes: 30), (timer) {
|
|
checkLocalizationUpdate();
|
|
});
|
|
|
|
ref.onDispose(() {
|
|
_serverUpdateTimer?.cancel();
|
|
_serverUpdateTimer = null;
|
|
_appUpdateTimer?.cancel();
|
|
_appUpdateTimer = null;
|
|
});
|
|
}
|
|
|
|
void _loadData() async {
|
|
if (appGlobalState.networkVersionData == null) return;
|
|
try {
|
|
final r = await Api.getAppPlacard();
|
|
final box = await Hive.openBox("app_conf");
|
|
final version = box.get("close_placard", defaultValue: "");
|
|
if (r.enable == true) {
|
|
if (r.alwaysShow != true && version == r.version) {
|
|
} else {
|
|
state = state.copyWith(appPlacardData: r);
|
|
}
|
|
}
|
|
|
|
final appWebLocalizationVersionsData =
|
|
AppWebLocalizationVersionsData.fromJson(json.decode(
|
|
(await RSHttp.getText(
|
|
"${URLConf.webTranslateHomeUrl}/versions.json"))));
|
|
final countdownFestivalListData = await Api.getFestivalCountdownList();
|
|
state = state.copyWith(
|
|
webLocalizationVersionsData: appWebLocalizationVersionsData,
|
|
countdownFestivalListData: countdownFestivalListData);
|
|
_updateSCServerStatus();
|
|
_loadRRS();
|
|
} catch (e) {
|
|
dPrint(e);
|
|
}
|
|
// check Localization update
|
|
checkLocalizationUpdate();
|
|
}
|
|
|
|
Future<void> _updateSCServerStatus() async {
|
|
try {
|
|
final s = await Api.getScServerStatus();
|
|
dPrint("updateSCServerStatus===$s");
|
|
state = state.copyWith(scServerStatus: s);
|
|
} catch (e) {
|
|
dPrint(e);
|
|
}
|
|
}
|
|
|
|
Future _loadRRS() async {
|
|
try {
|
|
final rssVideoItems = await RSSApi.getRssVideo();
|
|
state = state.copyWith(rssVideoItems: rssVideoItems);
|
|
final rssTextItems = await RSSApi.getRssText();
|
|
state = state.copyWith(rssTextItems: rssTextItems);
|
|
dPrint("RSS update Success !");
|
|
} catch (e) {
|
|
dPrint("_loadRRS Error:$e");
|
|
}
|
|
}
|
|
|
|
Future<void> checkLocalizationUpdate({bool skipReload = false}) async {
|
|
dPrint("_checkLocalizationUpdate");
|
|
await (ref.read(localizationUIModelProvider.notifier))
|
|
.checkLangUpdate(skipReload: skipReload)
|
|
.unwrap<List<String>>();
|
|
}
|
|
|
|
// ignore: avoid_build_context_in_providers
|
|
launchRSI(BuildContext context) async {
|
|
if (state.scInstalledPath == "not_install") {
|
|
showToast(context, S.current.home_info_valid_installation_required);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void onChangeInstallPath(String? value) {
|
|
if (value == null) return;
|
|
state = state.copyWith(scInstalledPath: value);
|
|
}
|
|
}
|