Files
umbrix/lib/features/settings/about/about_page.dart
Umbrix Developer 76a374950f feat: mobile-like window size and always-visible stats
- Changed window size to mobile phone format (400x800)
- Removed width condition for ActiveProxyFooter - now always visible
- Added run-umbrix.sh launch script with icon copying
- Stats cards now display on all screen sizes
2026-01-17 13:09:20 +03:00

186 lines
7.1 KiB
Dart

import 'package:fluentui_system_icons/fluentui_system_icons.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:gap/gap.dart';
import 'package:umbrix/core/app_info/app_info_provider.dart';
import 'package:umbrix/core/directories/directories_provider.dart';
import 'package:umbrix/core/localization/translations.dart';
import 'package:umbrix/core/model/failures.dart';
import 'package:umbrix/core/widget/adaptive_icon.dart';
import 'package:umbrix/features/app_update/notifier/app_update_notifier.dart';
import 'package:umbrix/features/app_update/notifier/app_update_state.dart';
import 'package:umbrix/features/app_update/widget/new_version_dialog.dart';
import 'package:umbrix/features/common/nested_app_bar.dart';
import 'package:umbrix/features/settings/about/privacy_policy_screen.dart';
import 'package:umbrix/features/settings/about/terms_and_conditions_screen.dart';
import 'package:umbrix/gen/assets.gen.dart';
import 'package:umbrix/utils/utils.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
class AboutPage extends HookConsumerWidget {
const AboutPage({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
final t = ref.watch(translationsProvider);
final appInfo = ref.watch(appInfoProvider).requireValue;
final appUpdate = ref.watch(appUpdateNotifierProvider);
ref.listen(
appUpdateNotifierProvider,
(_, next) async {
if (!context.mounted) return;
switch (next) {
case AppUpdateStateAvailable(:final versionInfo) || AppUpdateStateIgnored(:final versionInfo):
return NewVersionDialog(
appInfo.presentVersion,
versionInfo,
canIgnore: false,
).show(context);
case AppUpdateStateError(:final error):
return CustomToast.error(t.presentShortError(error)).show(context);
case AppUpdateStateNotAvailable():
return CustomToast.success(t.appUpdate.notAvailableMsg).show(context);
}
},
);
final conditionalTiles = [
// UMBRIX: Кнопка проверки обновлений
// Для Desktop платформ (Windows/macOS/Linux) - собственный сервер обновлений
// Для Android - будет ссылка на Google Play Store в production
if (PlatformUtils.isDesktop)
ListTile(
title: Text(t.about.checkForUpdate),
leading: switch (appUpdate) {
AppUpdateStateChecking() => const SizedBox(
width: 24,
height: 24,
child: CircularProgressIndicator(strokeWidth: 2),
),
AppUpdateStateAvailable() => const Icon(
FluentIcons.arrow_download_24_filled,
color: Colors.green,
),
_ => const Icon(FluentIcons.arrow_sync_24_regular),
},
onTap: () async {
await ref.read(appUpdateNotifierProvider.notifier).check();
},
),
if (PlatformUtils.isDesktop)
ListTile(
title: Text(t.settings.general.openWorkingDir),
trailing: const Icon(FluentIcons.open_folder_24_regular),
onTap: () async {
final path = ref.watch(appDirectoriesProvider).requireValue.workingDir.uri;
await UriUtils.tryLaunch(path);
},
),
];
return Scaffold(
body: CustomScrollView(
slivers: [
NestedAppBar(
title: Text(t.about.pageTitle),
actions: [
PopupMenuButton(
icon: Icon(AdaptiveIcon(context).more),
itemBuilder: (context) {
return [
PopupMenuItem(
child: Text(t.general.addToClipboard),
onTap: () {
Clipboard.setData(
ClipboardData(text: appInfo.format()),
);
},
),
];
},
),
],
),
SliverToBoxAdapter(
child: Padding(
padding: const EdgeInsets.all(16),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
// UMBRIX: Используем наш логотип
Assets.images.umbrixLogo.image(width: 64, height: 64),
const Gap(16),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
t.general.appTitle,
style: Theme.of(context).textTheme.titleLarge,
),
const Gap(4),
Text(
"${t.about.version} ${appInfo.presentVersion}",
),
],
),
],
),
),
),
SliverList(
delegate: SliverChildListDelegate(
[
...conditionalTiles,
if (conditionalTiles.isNotEmpty) const Divider(),
// UMBRIX: Убрали "Исходный код" - наш приватный форк
// ListTile(
// title: Text(t.about.sourceCode),
// trailing: const Icon(FluentIcons.open_24_regular),
// onTap: () async {
// await UriUtils.tryLaunch(
// Uri.parse(Constants.githubUrl),
// );
// },
// ),
// UMBRIX: Telegram канал пока закомментирован
// ListTile(
// title: Text(t.about.telegramChannel),
// trailing: const Icon(FluentIcons.open_24_regular),
// onTap: () async {
// await UriUtils.tryLaunch(
// Uri.parse(Constants.telegramChannelUrl),
// );
// },
// ),
ListTile(
title: Text(t.about.termsAndConditions),
trailing: const Icon(FluentIcons.chevron_right_20_regular),
onTap: () {
Navigator.of(context).push(
MaterialPageRoute(
builder: (context) => const TermsAndConditionsScreen(),
),
);
},
),
ListTile(
title: Text(t.about.privacyPolicy),
trailing: const Icon(FluentIcons.chevron_right_20_regular),
onTap: () {
Navigator.of(context).push(
MaterialPageRoute(
builder: (context) => const PrivacyPolicyScreen(),
),
);
},
),
],
),
),
],
),
);
}
}