Add update checking
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"general": {
|
"general": {
|
||||||
"appTitle": "hiddify",
|
"appTitle": "Hiddify Next",
|
||||||
"reset": "reset",
|
"reset": "reset",
|
||||||
"toggle": {
|
"toggle": {
|
||||||
"enabled": "enabled",
|
"enabled": "enabled",
|
||||||
@@ -125,6 +125,15 @@
|
|||||||
"telegramChannel": "telegram channel",
|
"telegramChannel": "telegram channel",
|
||||||
"checkForUpdate": "check for update"
|
"checkForUpdate": "check for update"
|
||||||
},
|
},
|
||||||
|
"appUpdate": {
|
||||||
|
"dialogTitle": "update available",
|
||||||
|
"updateMsg": "A new version of @:general.appTitle is available! Would you like to update now?",
|
||||||
|
"currentVersionLbl": "current version",
|
||||||
|
"newVersionLbl": "new version",
|
||||||
|
"updateNowBtnTxt": "update now",
|
||||||
|
"laterBtnTxt": "later",
|
||||||
|
"ignoreBtnTxt": "ignore"
|
||||||
|
},
|
||||||
"tray": {
|
"tray": {
|
||||||
"dashboard": "dashboard",
|
"dashboard": "dashboard",
|
||||||
"quit": "quit",
|
"quit": "quit",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"general": {
|
"general": {
|
||||||
"appTitle": "هیدیفای",
|
"appTitle": "هیدیفای نکست",
|
||||||
"reset": "ریست",
|
"reset": "ریست",
|
||||||
"toggle": {
|
"toggle": {
|
||||||
"enabled": "فعال",
|
"enabled": "فعال",
|
||||||
@@ -125,6 +125,15 @@
|
|||||||
"telegramChannel": "کانال تلگرام",
|
"telegramChannel": "کانال تلگرام",
|
||||||
"checkForUpdate": "بررسی آپدیت جدید"
|
"checkForUpdate": "بررسی آپدیت جدید"
|
||||||
},
|
},
|
||||||
|
"appUpdate": {
|
||||||
|
"dialogTitle": "نسخه جدید",
|
||||||
|
"updateMsg": "نسخه جدیدی از @:general.appTitle موجود است! الان بروزرسانی شود؟",
|
||||||
|
"currentVersionLbl": "نسخه فعلی",
|
||||||
|
"newVersionLbl": "نسخه جدید",
|
||||||
|
"updateNowBtnTxt": "بروزرسانی",
|
||||||
|
"laterBtnTxt": "بعدا",
|
||||||
|
"ignoreBtnTxt": "نادیدهگرفتن"
|
||||||
|
},
|
||||||
"tray": {
|
"tray": {
|
||||||
"dashboard": "داشبورد",
|
"dashboard": "داشبورد",
|
||||||
"quit": "خروج",
|
"quit": "خروج",
|
||||||
|
|||||||
@@ -2,6 +2,8 @@ import 'package:dio/dio.dart';
|
|||||||
import 'package:hiddify/data/local/dao/dao.dart';
|
import 'package:hiddify/data/local/dao/dao.dart';
|
||||||
import 'package:hiddify/data/local/database.dart';
|
import 'package:hiddify/data/local/database.dart';
|
||||||
import 'package:hiddify/data/repository/repository.dart';
|
import 'package:hiddify/data/repository/repository.dart';
|
||||||
|
import 'package:hiddify/data/repository/update_repository_impl.dart';
|
||||||
|
import 'package:hiddify/domain/app/app.dart';
|
||||||
import 'package:hiddify/domain/clash/clash.dart';
|
import 'package:hiddify/domain/clash/clash.dart';
|
||||||
import 'package:hiddify/domain/profiles/profiles.dart';
|
import 'package:hiddify/domain/profiles/profiles.dart';
|
||||||
import 'package:hiddify/services/service_providers.dart';
|
import 'package:hiddify/services/service_providers.dart';
|
||||||
@@ -40,3 +42,7 @@ ProfilesRepository profilesRepository(ProfilesRepositoryRef ref) =>
|
|||||||
clashFacade: ref.watch(clashFacadeProvider),
|
clashFacade: ref.watch(clashFacadeProvider),
|
||||||
dio: ref.watch(dioProvider),
|
dio: ref.watch(dioProvider),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@Riverpod(keepAlive: true)
|
||||||
|
UpdateRepository updateRepository(UpdateRepositoryRef ref) =>
|
||||||
|
UpdateRepositoryImpl(ref.watch(dioProvider));
|
||||||
|
|||||||
59
lib/data/repository/update_repository_impl.dart
Normal file
59
lib/data/repository/update_repository_impl.dart
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
import 'package:dio/dio.dart';
|
||||||
|
import 'package:fpdart/fpdart.dart';
|
||||||
|
import 'package:hiddify/data/repository/exception_handlers.dart';
|
||||||
|
import 'package:hiddify/domain/app/app.dart';
|
||||||
|
import 'package:hiddify/domain/constants.dart';
|
||||||
|
import 'package:hiddify/utils/custom_loggers.dart';
|
||||||
|
import 'package:package_info_plus/package_info_plus.dart';
|
||||||
|
|
||||||
|
class UpdateRepositoryImpl
|
||||||
|
with ExceptionHandler, InfraLogger
|
||||||
|
implements UpdateRepository {
|
||||||
|
UpdateRepositoryImpl(this.dio);
|
||||||
|
|
||||||
|
final Dio dio;
|
||||||
|
|
||||||
|
@override
|
||||||
|
TaskEither<UpdateFailure, InstalledVersionInfo> getCurrentVersion() {
|
||||||
|
return exceptionHandler(
|
||||||
|
() async {
|
||||||
|
final packageInfo = await PackageInfo.fromPlatform();
|
||||||
|
return right(
|
||||||
|
InstalledVersionInfo(
|
||||||
|
version: packageInfo.version,
|
||||||
|
buildNumber: packageInfo.buildNumber,
|
||||||
|
installerMedia: packageInfo.installerStore,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
UpdateFailure.unexpected,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
TaskEither<UpdateFailure, RemoteVersionInfo> getLatestVersion({
|
||||||
|
bool includePreReleases = false,
|
||||||
|
}) {
|
||||||
|
return exceptionHandler(
|
||||||
|
() async {
|
||||||
|
final response = await dio.get<List>(Constants.githubReleasesApiUrl);
|
||||||
|
|
||||||
|
if (response.statusCode != 200 || response.data == null) {
|
||||||
|
loggy.warning("failed to fetch latest version info");
|
||||||
|
return left(const UpdateFailure.unexpected());
|
||||||
|
}
|
||||||
|
|
||||||
|
final releases = response.data!
|
||||||
|
.map((e) => RemoteVersionInfo.fromJson(e as Map<String, dynamic>));
|
||||||
|
late RemoteVersionInfo latest;
|
||||||
|
if (includePreReleases) {
|
||||||
|
latest = releases.first;
|
||||||
|
} else {
|
||||||
|
latest = releases.firstWhere((e) => e.preRelease == false);
|
||||||
|
}
|
||||||
|
return right(latest);
|
||||||
|
},
|
||||||
|
UpdateFailure.unexpected,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
3
lib/domain/app/app.dart
Normal file
3
lib/domain/app/app.dart
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export 'update_failure.dart';
|
||||||
|
export 'update_repository.dart';
|
||||||
|
export 'version_info.dart';
|
||||||
22
lib/domain/app/update_failure.dart
Normal file
22
lib/domain/app/update_failure.dart
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
|
import 'package:hiddify/core/locale/locale.dart';
|
||||||
|
import 'package:hiddify/domain/failures.dart';
|
||||||
|
|
||||||
|
part 'update_failure.freezed.dart';
|
||||||
|
|
||||||
|
@freezed
|
||||||
|
sealed class UpdateFailure with _$UpdateFailure, Failure {
|
||||||
|
const UpdateFailure._();
|
||||||
|
|
||||||
|
const factory UpdateFailure.unexpected([
|
||||||
|
Object? error,
|
||||||
|
StackTrace? stackTrace,
|
||||||
|
]) = UpdateUnexpectedFailure;
|
||||||
|
|
||||||
|
@override
|
||||||
|
String present(TranslationsEn t) {
|
||||||
|
return switch (this) {
|
||||||
|
UpdateUnexpectedFailure() => t.failure.unexpected,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
11
lib/domain/app/update_repository.dart
Normal file
11
lib/domain/app/update_repository.dart
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import 'package:fpdart/fpdart.dart';
|
||||||
|
import 'package:hiddify/domain/app/update_failure.dart';
|
||||||
|
import 'package:hiddify/domain/app/version_info.dart';
|
||||||
|
|
||||||
|
abstract interface class UpdateRepository {
|
||||||
|
TaskEither<UpdateFailure, InstalledVersionInfo> getCurrentVersion();
|
||||||
|
|
||||||
|
TaskEither<UpdateFailure, RemoteVersionInfo> getLatestVersion({
|
||||||
|
bool includePreReleases = false,
|
||||||
|
});
|
||||||
|
}
|
||||||
56
lib/domain/app/version_info.dart
Normal file
56
lib/domain/app/version_info.dart
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
import 'package:dartx/dartx.dart';
|
||||||
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
|
|
||||||
|
part 'version_info.freezed.dart';
|
||||||
|
part 'version_info.g.dart';
|
||||||
|
|
||||||
|
@freezed
|
||||||
|
class InstalledVersionInfo with _$InstalledVersionInfo {
|
||||||
|
const InstalledVersionInfo._();
|
||||||
|
|
||||||
|
const factory InstalledVersionInfo({
|
||||||
|
required String version,
|
||||||
|
required String buildNumber,
|
||||||
|
String? installerMedia,
|
||||||
|
}) = _InstalledVersionInfo;
|
||||||
|
|
||||||
|
String get fullVersion =>
|
||||||
|
buildNumber.isBlank ? version : "$version+$buildNumber";
|
||||||
|
|
||||||
|
factory InstalledVersionInfo.fromJson(Map<String, dynamic> json) =>
|
||||||
|
_$InstalledVersionInfoFromJson(json);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO ignore drafts
|
||||||
|
@Freezed()
|
||||||
|
class RemoteVersionInfo with _$RemoteVersionInfo {
|
||||||
|
const RemoteVersionInfo._();
|
||||||
|
|
||||||
|
const factory RemoteVersionInfo({
|
||||||
|
required String version,
|
||||||
|
required String buildNumber,
|
||||||
|
required String releaseTag,
|
||||||
|
required bool preRelease,
|
||||||
|
required DateTime publishedAt,
|
||||||
|
}) = _RemoteVersionInfo;
|
||||||
|
|
||||||
|
String get fullVersion =>
|
||||||
|
buildNumber.isBlank ? version : "$version+$buildNumber";
|
||||||
|
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static RemoteVersionInfo fromJson(Map<String, dynamic> json) {
|
||||||
|
final fullTag = json['tag_name'] as String;
|
||||||
|
final fullVersion = fullTag.removePrefix("v").split("-").first.split("+");
|
||||||
|
final version = fullVersion.first;
|
||||||
|
final buildNumber = fullVersion.elementAtOrElse(1, (index) => "");
|
||||||
|
final preRelease = json["prerelease"] as bool;
|
||||||
|
final publishedAt = DateTime.parse(json["published_at"] as String);
|
||||||
|
return RemoteVersionInfo(
|
||||||
|
version: version,
|
||||||
|
buildNumber: buildNumber,
|
||||||
|
releaseTag: fullTag,
|
||||||
|
preRelease: preRelease,
|
||||||
|
publishedAt: publishedAt,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,5 +5,9 @@ abstract class Constants {
|
|||||||
static const configFileName = "config";
|
static const configFileName = "config";
|
||||||
static const countryMMDBFileName = "Country";
|
static const countryMMDBFileName = "Country";
|
||||||
static const githubUrl = "https://github.com/hiddify/hiddify-next";
|
static const githubUrl = "https://github.com/hiddify/hiddify-next";
|
||||||
|
static const githubReleasesApiUrl =
|
||||||
|
"https://api.github.com/repos/hiddify/hiddify-next/releases";
|
||||||
|
static const githubLatestReleaseUrl =
|
||||||
|
"https://github.com/hiddify/hiddify-next/releases/latest";
|
||||||
static const telegramChannelUrl = "https://t.me/hiddify";
|
static const telegramChannelUrl = "https://t.me/hiddify";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,8 +2,11 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:gap/gap.dart';
|
import 'package:gap/gap.dart';
|
||||||
import 'package:hiddify/core/core_providers.dart';
|
import 'package:hiddify/core/core_providers.dart';
|
||||||
import 'package:hiddify/domain/constants.dart';
|
import 'package:hiddify/domain/constants.dart';
|
||||||
|
import 'package:hiddify/domain/failures.dart';
|
||||||
|
import 'package:hiddify/features/common/new_version_dialog.dart';
|
||||||
import 'package:hiddify/features/common/runtime_details.dart';
|
import 'package:hiddify/features/common/runtime_details.dart';
|
||||||
import 'package:hiddify/gen/assets.gen.dart';
|
import 'package:hiddify/gen/assets.gen.dart';
|
||||||
|
import 'package:hiddify/utils/alerts.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:recase/recase.dart';
|
import 'package:recase/recase.dart';
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
@@ -14,7 +17,36 @@ class AboutPage extends HookConsumerWidget {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
final t = ref.watch(translationsProvider);
|
final t = ref.watch(translationsProvider);
|
||||||
final details = ref.watch(runtimeDetailsNotifierProvider);
|
final appVersion = ref.watch(appVersionProvider);
|
||||||
|
|
||||||
|
final isCheckingForUpdate = ref.watch(
|
||||||
|
runtimeDetailsNotifierProvider.select(
|
||||||
|
(value) => value.maybeWhen(
|
||||||
|
data: (data) => data.latestVersion.isLoading,
|
||||||
|
orElse: () => false,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
ref.listen(
|
||||||
|
runtimeDetailsNotifierProvider,
|
||||||
|
(_, next) async {
|
||||||
|
if (next case AsyncData(:final value)) {
|
||||||
|
switch (value.latestVersion) {
|
||||||
|
case AsyncError(:final error):
|
||||||
|
CustomToast.error(t.presentError(error)).show(context);
|
||||||
|
default:
|
||||||
|
if (value.newVersionAvailable) {
|
||||||
|
await NewVersionDialog(
|
||||||
|
value.appVersion,
|
||||||
|
value.latestVersion.value!,
|
||||||
|
canIgnore: false,
|
||||||
|
).show(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
body: CustomScrollView(
|
body: CustomScrollView(
|
||||||
@@ -22,7 +54,7 @@ class AboutPage extends HookConsumerWidget {
|
|||||||
SliverAppBar(
|
SliverAppBar(
|
||||||
title: Text(t.about.pageTitle.titleCase),
|
title: Text(t.about.pageTitle.titleCase),
|
||||||
),
|
),
|
||||||
...switch (details) {
|
...switch (appVersion) {
|
||||||
AsyncData(:final value) => [
|
AsyncData(:final value) => [
|
||||||
SliverToBoxAdapter(
|
SliverToBoxAdapter(
|
||||||
child: Padding(
|
child: Padding(
|
||||||
@@ -77,6 +109,18 @@ class AboutPage extends HookConsumerWidget {
|
|||||||
),
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text(t.about.checkForUpdate.sentenceCase),
|
title: Text(t.about.checkForUpdate.sentenceCase),
|
||||||
|
trailing: isCheckingForUpdate
|
||||||
|
? const SizedBox(
|
||||||
|
width: 24,
|
||||||
|
height: 24,
|
||||||
|
child: CircularProgressIndicator(),
|
||||||
|
)
|
||||||
|
: const Icon(Icons.update),
|
||||||
|
onTap: () async {
|
||||||
|
await ref
|
||||||
|
.read(runtimeDetailsNotifierProvider.notifier)
|
||||||
|
.checkForUpdates();
|
||||||
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
99
lib/features/common/new_version_dialog.dart
Normal file
99
lib/features/common/new_version_dialog.dart
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:gap/gap.dart';
|
||||||
|
import 'package:go_router/go_router.dart';
|
||||||
|
import 'package:hiddify/core/core_providers.dart';
|
||||||
|
import 'package:hiddify/domain/app/app.dart';
|
||||||
|
import 'package:hiddify/domain/constants.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
import 'package:recase/recase.dart';
|
||||||
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
|
||||||
|
// TODO add release notes
|
||||||
|
class NewVersionDialog extends HookConsumerWidget {
|
||||||
|
const NewVersionDialog(
|
||||||
|
this.currentVersion,
|
||||||
|
this.newVersion, {
|
||||||
|
super.key,
|
||||||
|
this.canIgnore = true,
|
||||||
|
});
|
||||||
|
|
||||||
|
final InstalledVersionInfo currentVersion;
|
||||||
|
final RemoteVersionInfo newVersion;
|
||||||
|
final bool canIgnore;
|
||||||
|
|
||||||
|
Future<void> show(BuildContext context) {
|
||||||
|
return showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => this,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
|
final t = ref.watch(translationsProvider);
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
|
||||||
|
return AlertDialog(
|
||||||
|
title: Text(t.appUpdate.dialogTitle.titleCase),
|
||||||
|
content: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(t.appUpdate.updateMsg),
|
||||||
|
const Gap(8),
|
||||||
|
Text.rich(
|
||||||
|
TextSpan(
|
||||||
|
children: [
|
||||||
|
TextSpan(
|
||||||
|
text: "${t.appUpdate.currentVersionLbl}: ",
|
||||||
|
style: theme.textTheme.bodySmall,
|
||||||
|
),
|
||||||
|
TextSpan(
|
||||||
|
text: currentVersion.fullVersion,
|
||||||
|
style: theme.textTheme.labelMedium,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Text.rich(
|
||||||
|
TextSpan(
|
||||||
|
children: [
|
||||||
|
TextSpan(
|
||||||
|
text: "${t.appUpdate.newVersionLbl}: ",
|
||||||
|
style: theme.textTheme.bodySmall,
|
||||||
|
),
|
||||||
|
TextSpan(
|
||||||
|
text: newVersion.fullVersion,
|
||||||
|
style: theme.textTheme.labelMedium,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
if (canIgnore)
|
||||||
|
TextButton(
|
||||||
|
onPressed: () {
|
||||||
|
// TODO add prefs for ignoring version
|
||||||
|
context.pop();
|
||||||
|
},
|
||||||
|
child: Text(t.appUpdate.ignoreBtnTxt.titleCase),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: context.pop,
|
||||||
|
child: Text(t.appUpdate.laterBtnTxt.titleCase),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
await launchUrl(
|
||||||
|
Uri.parse(Constants.githubLatestReleaseUrl),
|
||||||
|
mode: LaunchMode.externalApplication,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
child: Text(t.appUpdate.updateNowBtnTxt.titleCase),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,39 +1,78 @@
|
|||||||
import 'package:freezed_annotation/freezed_annotation.dart';
|
import 'package:freezed_annotation/freezed_annotation.dart';
|
||||||
|
import 'package:hiddify/data/data_providers.dart';
|
||||||
|
import 'package:hiddify/domain/app/app.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:package_info_plus/package_info_plus.dart';
|
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
|
|
||||||
part 'runtime_details.freezed.dart';
|
part 'runtime_details.freezed.dart';
|
||||||
part 'runtime_details.g.dart';
|
part 'runtime_details.g.dart';
|
||||||
|
|
||||||
// TODO implement clash version
|
// TODO add clash version
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
class RuntimeDetailsNotifier extends _$RuntimeDetailsNotifier with AppLogger {
|
class RuntimeDetailsNotifier extends _$RuntimeDetailsNotifier with AppLogger {
|
||||||
@override
|
@override
|
||||||
Future<RuntimeDetails> build() async {
|
Future<RuntimeDetails> build() async {
|
||||||
final packageInfo = await PackageInfo.fromPlatform();
|
final appVersion = await ref
|
||||||
return RuntimeDetails(
|
.watch(updateRepositoryProvider)
|
||||||
version: packageInfo.version,
|
.getCurrentVersion()
|
||||||
buildNumber: packageInfo.buildNumber,
|
.getOrElse((l) => throw l)
|
||||||
installerStore: packageInfo.installerStore,
|
.run();
|
||||||
clashVersion: "",
|
return RuntimeDetails(appVersion: appVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> checkForUpdates() async {
|
||||||
|
if (state case AsyncData(:final value)) {
|
||||||
|
switch (value.latestVersion) {
|
||||||
|
case AsyncLoading():
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
loggy.debug("checking for updates");
|
||||||
|
state =
|
||||||
|
AsyncData(value.copyWith(latestVersion: const AsyncLoading()));
|
||||||
|
// TODO use prefs
|
||||||
|
const includePreReleases = true;
|
||||||
|
await ref
|
||||||
|
.read(updateRepositoryProvider)
|
||||||
|
.getLatestVersion(includePreReleases: includePreReleases)
|
||||||
|
.match(
|
||||||
|
(l) {
|
||||||
|
loggy.warning("failed to get latest version, $l");
|
||||||
|
state = AsyncData(
|
||||||
|
value.copyWith(
|
||||||
|
latestVersion: AsyncError(l, StackTrace.current),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
},
|
||||||
|
(r) {
|
||||||
|
state = AsyncData(
|
||||||
|
value.copyWith(latestVersion: AsyncData(r)),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
).run();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Riverpod(keepAlive: true)
|
||||||
|
AsyncValue<InstalledVersionInfo> appVersion(AppVersionRef ref) => ref.watch(
|
||||||
|
runtimeDetailsNotifierProvider
|
||||||
|
.select((value) => value.whenData((value) => value.appVersion)),
|
||||||
|
);
|
||||||
|
|
||||||
@freezed
|
@freezed
|
||||||
class RuntimeDetails with _$RuntimeDetails {
|
class RuntimeDetails with _$RuntimeDetails {
|
||||||
const RuntimeDetails._();
|
const RuntimeDetails._();
|
||||||
|
|
||||||
const factory RuntimeDetails({
|
const factory RuntimeDetails({
|
||||||
required String version,
|
required InstalledVersionInfo appVersion,
|
||||||
required String buildNumber,
|
@Default(AsyncData(null)) AsyncValue<RemoteVersionInfo?> latestVersion,
|
||||||
String? installerStore,
|
|
||||||
required String clashVersion,
|
|
||||||
}) = _RuntimeDetails;
|
}) = _RuntimeDetails;
|
||||||
|
|
||||||
String get fullVersion => version + buildNumber;
|
bool get newVersionAvailable => latestVersion.maybeWhen(
|
||||||
|
data: (data) =>
|
||||||
factory RuntimeDetails.fromJson(Map<String, dynamic> json) =>
|
data != null &&
|
||||||
_$RuntimeDetailsFromJson(json);
|
data.fullVersion.compareTo(this.appVersion.fullVersion) > 0,
|
||||||
|
orElse: () => false,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -98,7 +98,7 @@ class AppVersionLabel extends HookConsumerWidget {
|
|||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
|
|
||||||
final version = ref.watch(
|
final version = ref.watch(
|
||||||
runtimeDetailsNotifierProvider.select(
|
appVersionProvider.select(
|
||||||
(value) => switch (value) {
|
(value) => switch (value) {
|
||||||
AsyncData(:final value) => value.fullVersion,
|
AsyncData(:final value) => value.fullVersion,
|
||||||
_ => "",
|
_ => "",
|
||||||
|
|||||||
Reference in New Issue
Block a user