From 86357f8ed7b462b1967fea112744e5e000c5a090 Mon Sep 17 00:00:00 2001 From: MjKey Date: Sat, 10 Aug 2024 19:33:29 +0300 Subject: [PATCH] 2.0.3 --- assets/styles.json | 25 +++ lang/en.json | 31 +++- lang/ru.json | 31 +++- lib/main.dart | 226 +++++++++++++++++++++-- lib/socket_service.dart | 7 +- lib/style_screen.dart | 400 ++++++++++++++++++++++++++++++++++++++++ pubspec.yaml | 94 ++-------- setup.iss | 37 +++- 8 files changed, 757 insertions(+), 94 deletions(-) create mode 100644 assets/styles.json create mode 100644 lib/style_screen.dart diff --git a/assets/styles.json b/assets/styles.json new file mode 100644 index 0000000..9022c64 --- /dev/null +++ b/assets/styles.json @@ -0,0 +1,25 @@ +{ + "styles": [ + { + "name": "Kode Mono Black", + "img": "", + "css": "@import url(https://fonts.googleapis.com/css2?family=Kode+Mono:wght@400..700&display=swap);body{background-color:#fff0;margin:0 auto;overflow:hidden}#timer{font-family:'Kode Mono',monospace;font-size:48px;margin-top:50px;text-align:center;color:#000;}" + }, + { + "name": "Kode Mono White", + "img": "", + "css": "@import url(https://fonts.googleapis.com/css2?family=Kode+Mono:wght@400..700&display=swap);body{background-color:#fff0;margin:0 auto;overflow:hidden}#timer{font-family:'Kode Mono',monospace;font-size:48px;margin-top:50px;text-align:center;color:#fff}" + }, + { + "name": "Pop Art", + "img": "", + "css": "@import url(https://fonts.googleapis.com/css2?family=Raleway:wght@900&display=swap);body{background-color:#fff0;margin:0 auto;overflow:hidden}#timer{font-family:'Raleway',sans-serif;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);font-size:120px;letter-spacing:.1em;-webkit-text-fill-color:#fff0;-webkit-text-stroke-width:3px;-webkit-text-stroke-color:#fff;text-shadow:8px 8px #ff1f8f,20px 20px #000}" + }, + { + "name": "Neon Border", + "img": "", + "css": "@import url(https://fonts.googleapis.com/css2?family=Kode+Mono:wght@400..700&display=swap);body{background-color:#fff0;margin:0 auto;overflow:hidden;text-align:center}#timer{font-family:'Kode Mono';font-size:3em;display:inline-block;text-shadow:-1px -1px 10px #fff,-1px -1px 10px #fff,0 0 20px #ff00de,0 0 40px #ff00de,0 0 60px #ff00de,0 0 80px #ff00de;padding:30px;box-shadow:0 0 10px #ff00de,inset 0 0 10px #ff00de,0 0 20px 2px #ff00de,inset 0 0 20px 2px #ff00de}" + } + ] + } + \ No newline at end of file diff --git a/lang/en.json b/lang/en.json index 9493413..762c223 100644 --- a/lang/en.json +++ b/lang/en.json @@ -46,5 +46,34 @@ "support":"Support the author", "port_settings": "Port Settings", "port": "Port", - "err_port":"Enter the correct port values" + "err_port":"Enter the correct port values", + "m_styles": "Timer Styles for OBS", + "m_g_styles": "Create Timer Styles", + "timer_styles_title": "Styles for Timer for OBS", + "copy_css": "Copy CSS", + "css_copied": "CSS code copied", + "help": "How to apply styles", + "help_content": "Instructions for applying styles for OBS...", + "style_create": "You can create your own CSS ", + "style_cr_link": ">>HERE<<", + "css_generator_title": "CSS Generator", + "pick_color": "Pick a color", + "select": "Select", + "background": "Background", + "text_shadow": "Text Shadow", + "border": "Border", + "font_size": "Font Size", + "text_color": "Text Color", + "font_family": "Font Family", + "background_color": "Background Color", + "text_shadow_color": "Text Shadow Color", + "text_shadow_offset_x": "Text Shadow Offset X", + "text_shadow_offset_y": "Text Shadow Offset Y", + "border_width": "Border Width", + "border_radius": "Border Radius", + "border_color": "Border Color", + "pick": "Pick", + "width_block": "Width block", + "height_block": "Height block", + "custom_font": "Font Path [C://gachi/myfon.ttf] (leave empty to use Google Fonts)" } \ No newline at end of file diff --git a/lang/ru.json b/lang/ru.json index 1539afe..0fa6b21 100644 --- a/lang/ru.json +++ b/lang/ru.json @@ -46,5 +46,34 @@ "support":"Поддержать автора", "port_settings": "Настройки портов", "port": "порт", - "err_port":"Введите корректные значения портов" + "err_port":"Введите корректные значения портов", + "m_styles": "Стили таймера для OBS", + "m_g_styles": "Создать стиль таймера", + "timer_styles_title": "Стили для таймера для OBS", + "css_copied": "CSS-код скопирован", + "help": "Как применять стили?", + "help_content": "Инструкции по применению стилей в OBS...", + "style_create": "Создать свой CSS можно ", + "style_cr_link": ">>ТУТ<<", + "css_generator_title": "Генератор CSS", + "pick_color": "Выберите цвет", + "select": "Выбрать", + "background": "Фон", + "text_shadow": "Тень текста", + "border": "Рамка", + "font_size": "Размер шрифта", + "text_color": "Цвет текста", + "font_family": "Google шрифт", + "background_color": "Цвет фона", + "text_shadow_color": "Цвет тени текста", + "text_shadow_offset_x": "Смещение тени X", + "text_shadow_offset_y": "Смещение тени Y", + "border_width": "Ширина рамки", + "border_radius": "Радиус рамки", + "border_color": "Цвет рамки", + "copy_css": "Скопировать CSS", + "pick": "Выбрать", + "width_block": "Ширина блока", + "height_block": "Высота рамки", + "custom_font": "Путь до шрифта [C://pupka/font.ttf] (оставьте поле пустым, чтобы использовать шрифт Google)" } \ No newline at end of file diff --git a/lib/main.dart b/lib/main.dart index 2ca4945..3e5bcfb 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -1,4 +1,4 @@ -// ignore_for_file: depend_on_referenced_packages, library_private_types_in_public_api, use_build_context_synchronously +// ignore_for_file: depend_on_referenced_packages, library_private_types_in_public_api, use_build_context_synchronously, deprecated_member_use import 'dart:io'; import 'dart:async'; @@ -24,6 +24,9 @@ import 'package:network_info_plus/network_info_plus.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:web_socket_channel/web_socket_channel.dart'; import 'package:flutter_localizations/flutter_localizations.dart'; +import 'package:webview_windows/webview_windows.dart'; +import 'package:translator/translator.dart'; +import 'style_screen.dart'; class LocalizationProvider with ChangeNotifier { Map _localizedStrings = {}; @@ -61,6 +64,30 @@ class LocalizationProvider with ChangeNotifier { } } +class TimerStyle { + final String name; + final String img; + final String css; + + TimerStyle(this.name, this.img, this.css); + + factory TimerStyle.fromJson(Map json) { + return TimerStyle( + json['name'] as String, + json['img'] as String, + json['css'] as String, + ); + } +} + +Future> _loadStyles() async { + final String response = await rootBundle.loadString('assets/styles.json'); + final data = await json.decode(response); + return (data['styles'] as List) + .map((style) => TimerStyle.fromJson(style)) + .toList(); +} + class ThemeProvider with ChangeNotifier { bool _isDarkMode = false; @@ -144,7 +171,7 @@ void main() async { LogManager.log(Level.INFO, 'Запуск..'); await windowManager.ensureInitialized(); if (Platform.isWindows) { - WindowManager.instance.setMinimumSize(const Size(700, 600)); + WindowManager.instance.setMinimumSize(const Size(700, 725)); WindowManager.instance.setMaximumSize(const Size(700, 1080)); LogManager.log(Level.INFO, 'Размеры окна заданы'); } @@ -213,6 +240,23 @@ class _MainScreenState extends State { int _httpPort = 8080; int _wsPort = 4040; + final String changelog = ''' + ★ Добавлен выбор стилей для таймера + ★ Добавлен генератр стилей (CSS) для таймера + ☆ Фикс получения локального IP [issues/1] + ☆ Немножко поправлен код + + Задумки (в планах на будущее): + ✦ Добавить поддержку: + ✧ Donate.Stream + ✧ Donatty + ✧ Donatepay + ✧ StreamElements (Вряд ли) + '''; + String translatedChangelog = ''; + + final version = "2.0.3"; + @override void initState() { super.initState(); @@ -222,6 +266,16 @@ class _MainScreenState extends State { _startWebServer(); _startWebSocketServer(); _getLocalIpAddress(); + _translateChangelog(); + } + + void _translateChangelog() async { + final translator = GoogleTranslator(); + final translated = + await translator.translate(changelog, from: 'ru', to: 'en'); + setState(() { + translatedChangelog = translated.text; + }); } void _incrementCounter() { @@ -292,8 +346,9 @@ class _MainScreenState extends State { }); } - void _showAuthorInfo() { + void _showAuthorInfo() async { LogManager.log(Level.INFO, 'Отображение информации об авторе'); + SharedPreferences prefs = await SharedPreferences.getInstance(); showDialog( context: context, builder: (BuildContext context) { @@ -340,7 +395,7 @@ class _MainScreenState extends State { style: theme.textTheme.bodyLarge, children: [ TextSpan( - text: '2.0.2', + text: version, style: GoogleFonts.lato( textStyle: const TextStyle( color: Color.fromARGB(255, 243, 33, 226), @@ -371,11 +426,15 @@ class _MainScreenState extends State { )), ), ), - const Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text('• Сделал покарсивше чутка'), - Text('• Испавил QR код в тёмной версии') + Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Text( + prefs.getString('language') == 'ru' + ? changelog + : translatedChangelog, + textAlign: TextAlign.left, + ), ], ), const SizedBox(height: 20), @@ -880,6 +939,135 @@ class _MainScreenState extends State { await _initSocketService(); } + void _openTextStyleGenerator() { + Navigator.of(context).push(MaterialPageRoute( + builder: (context) => const StyleGeneratorScreen(), + )); + } + + void _showYouTubeHelpDialog() { + final WebviewController controller = WebviewController(); + + showDialog( + context: context, + builder: (BuildContext context) { + return AlertDialog( + title: Text(context.read().translate('help')), + content: SizedBox( + width: 600, + height: 400, + child: FutureBuilder( + future: controller.initialize(), + builder: (BuildContext context, AsyncSnapshot snapshot) { + if (snapshot.connectionState == ConnectionState.done) { + controller.loadUrl('https://mjkey.ru/yt.html'); + return Webview(controller); + } else { + return const Center(child: CircularProgressIndicator()); + } + }, + ), + ), + actions: [ + TextButton( + child: + Text(context.read().translate('close')), + onPressed: () { + Navigator.of(context).pop(); + }, + ), + ], + ); + }, + ).then((_) { + controller.dispose(); + }); + } + + void _showTimerStylesDialog() async { + List styles = await _loadStyles(); + final localizationProvider = + Provider.of(context, listen: false); + + showDialog( + context: context, + builder: (BuildContext context) { + final theme = Theme.of(context); + return AlertDialog( + title: Text(localizationProvider.translate('timer_styles_title')), + content: SingleChildScrollView( + child: Column( + children: styles.map((style) { + return Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Column( + children: [ + Image.memory(base64Decode(style.img.split(',')[1]), + height: 50), + ], + ), + const SizedBox(width: 10), + Column( + children: [ + ElevatedButton( + onPressed: () { + Clipboard.setData(ClipboardData(text: style.css)); + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text(localizationProvider + .translate('css_copied'))), + ); + }, + child: + Text(localizationProvider.translate('copy_css')), + ), + ], + ), + ], + ); + }).toList(), + ), + ), + actions: [ + RichText( + text: TextSpan( + text: + '${context.read().translate('style_create')} ', + style: theme.textTheme.bodySmall, + children: [ + TextSpan( + text: context + .read() + .translate('style_cr_link'), + style: const TextStyle( + color: Colors.blue, + decoration: TextDecoration.underline, + ), + recognizer: TapGestureRecognizer() + ..onTap = () async { + _openTextStyleGenerator(); + }, + ), + ], + ), + ), + IconButton( + icon: const Icon(Icons.help_outline), + onPressed: _showYouTubeHelpDialog, + ), + TextButton( + child: Text(localizationProvider.translate('close')), + onPressed: () { + Navigator.of(context).pop(); + }, + ), + ], + ); + }, + ); + } + void _saveStatistics() async { LogManager.log(Level.INFO, 'Сохранение статистик'); SharedPreferences prefs = await SharedPreferences.getInstance(); @@ -1390,8 +1578,8 @@ class _MainScreenState extends State { // GestureDetector( // onTap: _incrementCounter, // child: Image.asset( - // 'assets/pepe.gif', // Укажи ссылку на свою гифку - // height: 50, // Настроить размер + // 'assets/pepe.gif', + // height: 50, // ),), // ], // ), @@ -1440,6 +1628,12 @@ class _MainScreenState extends State { case 'port_settings': _showPortSettingsDialog(); break; + case 'timer_styles': + _showTimerStylesDialog(); + break; + case 'gen_timer_styles': + _openTextStyleGenerator(); + break; } }, itemBuilder: (BuildContext context) { @@ -1479,6 +1673,16 @@ class _MainScreenState extends State { child: Text(context .read() .translate('clear_stat'))), + PopupMenuItem( + value: 'timer_styles', + child: Text(context + .read() + .translate('m_styles'))), + PopupMenuItem( + value: 'gen_timer_styles', + child: Text(context + .read() + .translate('m_g_styles'))), ]; }, ), diff --git a/lib/socket_service.dart b/lib/socket_service.dart index 134e8ff..d5cb2cb 100644 --- a/lib/socket_service.dart +++ b/lib/socket_service.dart @@ -1,3 +1,5 @@ +// ignore_for_file: library_prefixes + import 'package:socket_io_client/socket_io_client.dart' as IO; import 'package:logging/logging.dart'; @@ -25,7 +27,6 @@ class SocketService { }); socket.on('connect_error', (error) { - print('Connection error: $error'); _logger.severe('Ошибка подключения socket: $error'); }); @@ -49,7 +50,7 @@ class SocketService { Future connect() async { _logger.info('Подключение к socket'); socket.connect(); - await Future.delayed(Duration(seconds: 2)); + await Future.delayed(const Duration(seconds: 2)); if (socket.connected) { _logger.info('Socket успешно подключен'); } else { @@ -61,7 +62,7 @@ class SocketService { _logger.info('Закрытие socket соединения'); socket.disconnect(); socket.dispose(); - await Future.delayed(Duration(milliseconds: 500)); + await Future.delayed(const Duration(milliseconds: 500)); _logger.info('Socket соединение закрыто'); } } diff --git a/lib/style_screen.dart b/lib/style_screen.dart new file mode 100644 index 0000000..56ecd20 --- /dev/null +++ b/lib/style_screen.dart @@ -0,0 +1,400 @@ +// ignore_for_file: library_private_types_in_public_api + +import 'package:flutter/material.dart'; +import 'package:flutter_colorpicker/flutter_colorpicker.dart'; +import 'package:google_fonts/google_fonts.dart'; +import 'package:flutter/services.dart'; +import 'package:provider/provider.dart'; +import 'main.dart'; + +class StyleGeneratorScreen extends StatefulWidget { + const StyleGeneratorScreen({super.key}); + + @override + _StyleGeneratorScreenState createState() => _StyleGeneratorScreenState(); +} + +class _StyleGeneratorScreenState extends State { + double _fontSize = 40; + Color _textColor = Colors.black; + Color _backgroundColor = Colors.white; + Color _textShadowColor = Colors.black; + double _textShadowOffsetX = 2.0; + double _textShadowOffsetY = 2.0; + double _borderRadius = 5.0; + double _borderWidth = 1.0; + Color _borderColor = Colors.black; + String _fontFamily = 'Roboto'; + double _widthBlock = 190.0; + double _heightBlock = 75.0; + String _fontPath = ''; + + bool _showBackground = true; + bool _showTextShadow = true; + bool _showBorder = true; + + TextStyle _buildTextStyle() { + return GoogleFonts.getFont( + _fontFamily, + fontSize: _fontSize, + color: _textColor, + shadows: _showTextShadow + ? [ + Shadow( + offset: Offset(_textShadowOffsetX, _textShadowOffsetY), + color: _textShadowColor, + ), + ] + : [], + ); + } + + String getCSS() { + String fontFace = _fontPath.isNotEmpty + ? ''' +@font-face { + font-family: 'CustomFont'; + src: url('file://${Uri.encodeComponent(_fontPath)}'); +} + ''' + : ''' +@import url('https://fonts.googleapis.com/css2?family=${Uri.encodeComponent(_fontFamily)}&display=swap'); + '''; + + return ''' +$fontFace + +body { + background-color: #fff0; + margin: 0 auto; + overflow: hidden; + display: flex; + justify-content: center; + font-family: ${_fontPath.isNotEmpty ? 'CustomFont' : _fontFamily}; +} + +#timer { + color: ${_colorToHex(_textColor)}; + font-size: ${_fontSize}px; + ${_showBackground ? 'background-color: ${_colorToHex(_backgroundColor)};' : ''} + ${_showTextShadow ? 'text-shadow: ${_textShadowOffsetX}px ${_textShadowOffsetY}px ${_colorToHex(_textShadowColor)};' : ''} + ${_showBorder ? 'border-radius: ${_borderRadius}px;' : ''} + ${_showBorder ? 'border: ${_borderWidth}px solid ${_colorToHex(_borderColor)};' : ''} + width: ${_widthBlock}px; + height: ${_heightBlock}px; + display: flex; + flex-wrap: nowrap; + align-items: center; + justify-content: center; + font-family: ${_fontPath.isNotEmpty ? 'CustomFont' : _fontFamily}; +} + '''; + } + + String _colorToHex(Color color) { + return '#${color.value.toRadixString(16).substring(2).toUpperCase()}'; + } + + void _pickColor(Color currentColor, ValueChanged onColorChanged) { + showDialog( + context: context, + builder: (context) { + return AlertDialog( + title: Text( + context.read().translate('pick_color')), + content: SingleChildScrollView( + child: ColorPicker( + pickerColor: currentColor, + onColorChanged: onColorChanged, + ), + ), + actions: [ + ElevatedButton( + child: Text( + context.read().translate('select')), + onPressed: () { + Navigator.of(context).pop(); + }, + ), + ], + ); + }, + ); + } + + void _copyCSS() { + Clipboard.setData(ClipboardData(text: getCSS())); + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text( + context.read().translate('css_copied'))), + ); + } + + @override + Widget build(BuildContext context) { + final loc = context.read(); + + return Scaffold( + appBar: AppBar( + title: Text(loc.translate('css_generator_title')), + centerTitle: true, + ), + body: Padding( + padding: const EdgeInsets.all(16.0), + child: Column( + children: [ + Expanded( + child: SingleChildScrollView( + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Flexible( + fit: FlexFit.loose, + child: Center( + child: Container( + width: _widthBlock, + height: _heightBlock, + decoration: BoxDecoration( + color: _showBackground + ? _backgroundColor + : Colors.transparent, + borderRadius: _showBorder + ? BorderRadius.circular(_borderRadius) + : null, + border: _showBorder + ? Border.all( + color: _borderColor, width: _borderWidth) + : null, + ), + child: Center( + child: Text( + '00:00:00', + style: _buildTextStyle(), + ), + ), + ), + ), + ), + const SizedBox(height: 20), + _buildToggleRow( + loc.translate('background'), _showBackground, (val) { + setState(() { + _showBackground = val; + }); + }), + _buildToggleRow( + loc.translate('text_shadow'), _showTextShadow, (val) { + setState(() { + _showTextShadow = val; + }); + }), + _buildToggleRow(loc.translate('border'), _showBorder, + (val) { + setState(() { + _showBorder = val; + }); + }), + const SizedBox(height: 20), + _buildSliderRow( + loc.translate('font_size'), _fontSize, 10, 100, (val) { + setState(() { + _fontSize = val; + }); + }), + const SizedBox(height: 10), + _buildSliderRow( + loc.translate('width_block'), _widthBlock, 10, 800, + (val) { + setState(() { + _widthBlock = val; + }); + }), + const SizedBox(height: 10), + _buildSliderRow( + loc.translate('height_block'), _heightBlock, 10, 800, + (val) { + setState(() { + _heightBlock = val; + }); + }), + const SizedBox(height: 10), + _buildColorPickerRow( + loc.translate('text_color'), _textColor, (color) { + setState(() { + _textColor = color; + }); + }), + const SizedBox(height: 10), + _buildFontPickerRow( + loc.translate('font_family'), _fontFamily, (font) { + setState(() { + if (font != null) { + _fontFamily = font; + } + }); + }), + const SizedBox(height: 10), + TextField( + decoration: InputDecoration( + labelText: loc.translate('custom_font'), + ), + onChanged: (value) { + setState(() { + _fontPath = value; + }); + }, + ), + const SizedBox(height: 10), + if (_showBackground) + _buildColorPickerRow( + loc.translate('background_color'), _backgroundColor, + (color) { + setState(() { + _backgroundColor = color; + }); + }), + const SizedBox(height: 10), + if (_showTextShadow) ...[ + _buildColorPickerRow( + loc.translate('text_shadow_color'), _textShadowColor, + (color) { + setState(() { + _textShadowColor = color; + }); + }), + const SizedBox(height: 10), + _buildSliderRow(loc.translate('text_shadow_offset_x'), + _textShadowOffsetX, -10, 10, (val) { + setState(() { + _textShadowOffsetX = val; + }); + }), + const SizedBox(height: 10), + _buildSliderRow(loc.translate('text_shadow_offset_y'), + _textShadowOffsetY, -10, 10, (val) { + setState(() { + _textShadowOffsetY = val; + }); + }), + ], + const SizedBox(height: 10), + if (_showBorder) ...[ + _buildSliderRow( + loc.translate('border_width'), _borderWidth, 0, 10, + (val) { + setState(() { + _borderWidth = val; + }); + }), + const SizedBox(height: 10), + _buildSliderRow( + loc.translate('border_radius'), _borderRadius, 0, 50, + (val) { + setState(() { + _borderRadius = val; + }); + }), + const SizedBox(height: 10), + _buildColorPickerRow( + loc.translate('border_color'), _borderColor, (color) { + setState(() { + _borderColor = color; + }); + }), + ], + const SizedBox(height: 20), + ElevatedButton( + onPressed: _copyCSS, + style: ElevatedButton.styleFrom( + padding: + const EdgeInsets.symmetric(horizontal: 24, vertical: 12), + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(8.0), + ), + ), + child: Text(loc.translate('copy_css')), + ), + ], + ), + ), + ), + ], + ), + ), + ); + } + + Widget _buildSliderRow(String label, double value, double min, double max, + ValueChanged onChanged) { + return Row( + children: [ + Text(label, style: const TextStyle(fontSize: 16)), + Expanded( + child: Slider( + value: value, + min: min, + max: max, + onChanged: onChanged, + ), + ), + Text(value.toStringAsFixed(1), style: const TextStyle(fontSize: 16)), + ], + ); + } + + Widget _buildToggleRow( + String label, bool value, ValueChanged onChanged) { + return Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text(label, style: const TextStyle(fontSize: 16)), + Switch( + value: value, + onChanged: onChanged, + ), + ], + ); + } + + Widget _buildColorPickerRow( + String label, Color currentColor, ValueChanged onColorChanged) { + return Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text(label, style: const TextStyle(fontSize: 16)), + ElevatedButton( + onPressed: () => _pickColor(currentColor, onColorChanged), + style: ElevatedButton.styleFrom( + backgroundColor: currentColor, + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + ), + child: Text(context.read().translate('pick'), + style: const TextStyle(color: Colors.white)), + ), + ], + ); + } + + Widget _buildFontPickerRow( + String label, String currentFont, ValueChanged onFontChanged) { + final fontNames = GoogleFonts.asMap().keys.toList(); + + return Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text(label, style: const TextStyle(fontSize: 16)), + DropdownButton( + value: currentFont, + items: fontNames.map((String font) { + return DropdownMenuItem( + value: font, + child: Text(font, style: TextStyle(fontFamily: font)), + ); + }).toList(), + onChanged: onFontChanged, + ), + ], + ); + } +} diff --git a/pubspec.yaml b/pubspec.yaml index f66c4c2..5134c7b 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -1,107 +1,47 @@ -name: donat_timer -description: "Donaton Timer." -# The following line prevents the package from being accidentally published to -# pub.dev using `flutter pub publish`. This is preferred for private packages. -publish_to: 'none' # Remove this line if you wish to publish to pub.dev - -# The following defines the version and build number for your application. -# A version number is three numbers separated by dots, like 1.2.43 -# followed by an optional build number separated by a +. -# Both the version and the builder number may be overridden in flutter -# build by specifying --build-name and --build-number, respectively. -# In Android, build-name is used as versionName while build-number used as versionCode. -# Read more about Android versioning at https://developer.android.com/studio/publish/versioning -# In iOS, build-name is used as CFBundleShortVersionString while build-number is used as CFBundleVersion. -# Read more about iOS versioning at -# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html -# In Windows, build-name is used as the major, minor, and patch parts -# of the product and file versions while build-number is used as the build suffix. -version: 2.0.2+1 +name: DonathonTimer +description: "Donathon Timer - A powerful and customizable timer application designed for live streaming and event management." +publish_to: 'none' +version: 2.0.3+1 environment: sdk: '>=3.4.3 <4.0.0' -# Dependencies specify other packages that your package needs in order to work. -# To automatically upgrade your package dependencies to the latest versions -# consider running `flutter pub upgrade --major-versions`. Alternatively, -# dependencies can be manually updated by changing the version numbers below to -# the latest version available on pub.dev. To see which dependencies have newer -# versions available, run `flutter pub outdated`. dependencies: flutter_localizations: sdk: flutter flutter: sdk: flutter intl: ^0.19.0 - provider: ^6.0.0 - http: ^1.2.1 + provider: ^6.1.2 + http: ^1.2.2 logging: ^1.2.0 - shared_preferences: ^2.2.3 - socket_io_client: ^1.0.2 - web_socket_channel: ^2.1.0 - cupertino_icons: ^1.0.6 - http_server: ^1.0.0 + shared_preferences: ^2.3.1 + socket_io_client: ^2.0.3+1 + web_socket_channel: ^3.0.1 + cupertino_icons: ^1.0.8 shelf: ^1.4.1 flutter_colorpicker: ^1.1.0 window_manager: ^0.3.9 - path_provider: ^2.1.3 + path_provider: ^2.1.4 url_launcher: ^6.3.0 qr_flutter: ^4.1.0 - network_info_plus: ^5.0.3 + network_info_plus: ^6.0.0 audioplayers: ^6.0.0 google_fonts: ^6.2.1 + desktop_webview_window: ^0.2.3 + webview_windows: ^0.4.0 + translator: ^1.0.0 dev_dependencies: flutter_test: sdk: flutter - # The "flutter_lints" package below contains a set of recommended lints to - # encourage good coding practices. The lint set provided by the package is - # activated in the `analysis_options.yaml` file located at the root of your - # package. See that file for information about deactivating specific lint - # rules and activating additional ones. flutter_lints: ^4.0.0 -# For information on the generic Dart part of this file, see the -# following page: https://dart.dev/tools/pub/pubspec - -# The following section is specific to Flutter packages. flutter: - - # The following line ensures that the Material Icons font is - # included with your application, so that you can use the icons in - # the material Icons class. uses-material-design: true - - # To add assets to your application, add an assets section, like this: assets: + - assets/styles.json - assets/pepe.gif - assets/pepe.mp3 - - lang/ - # - images/a_dot_ham.jpeg - - # An image asset can refer to one or more resolution-specific "variants", see - # https://flutter.dev/assets-and-images/#resolution-aware - - # For details regarding adding assets from package dependencies, see - # https://flutter.dev/assets-and-images/#from-packages - - # To add custom fonts to your application, add a fonts section here, - # in this "flutter" section. Each entry in this list should have a - # "family" key with the font family name, and a "fonts" key with a - # list giving the asset and other descriptors for the font. For - # example: - # fonts: - # - family: Schyler - # fonts: - # - asset: fonts/Schyler-Regular.ttf - # - asset: fonts/Schyler-Italic.ttf - # style: italic - # - family: Trajan Pro - # fonts: - # - asset: fonts/TrajanPro.ttf - # - asset: fonts/TrajanPro_Bold.ttf - # weight: 700 - # - # For details regarding fonts from package dependencies, - # see https://flutter.dev/custom-fonts/#from-packages + - lang/ \ No newline at end of file diff --git a/setup.iss b/setup.iss index 8d07437..1a89400 100644 --- a/setup.iss +++ b/setup.iss @@ -1,12 +1,17 @@ [Setup] AppName=Донатон Таймер -AppVersion=2.0.1 +AppVersion=2.0.3 DefaultDirName={pf}\DTimer DefaultGroupName=Донатон OutputBaseFilename=DTimer Setup Compression=lzma SolidCompression=yes SetupIconFile=Донатон Таймер.ico +WizardStyle=modern + +[Languages] +Name: "ru"; MessagesFile: "compiler:Languages\Russian.isl" +Name: "en"; MessagesFile: "compiler:Default.isl" [Files] Source: "build\windows\x64\runner\Release\*"; DestDir: "{app}"; Flags: ignoreversion recursesubdirs createallsubdirs @@ -19,3 +24,33 @@ Name: "{userdesktop}\Донатон Таймер"; Filename: "{app}\donat_timer. [Run] Filename: "{app}\donat_timer.exe"; Description: "Запустить Таймер Донатона"; Flags: nowait postinstall skipifsilent +[Code] +var + DeleteSettingsAndFiles: Boolean; + +procedure InitializeUninstallProgressForm(); +begin + DeleteSettingsAndFiles := False; + if MsgBox(ExpandConstant('{cm:DeleteSettingsPrompt}'), mbConfirmation, MB_YESNO) = IDYES then + begin + DeleteSettingsAndFiles := True; + end; +end; + +procedure CurUninstallStepChanged(CurUninstallStep: TUninstallStep); +begin + if (CurUninstallStep = usPostUninstall) and DeleteSettingsAndFiles then + begin + DelTree(ExpandConstant('{userappdata}\MjKey Studio'), True, True, True); + DelTree(ExpandConstant('{app}'), True, True, True); + end; +end; + +[UninstallDelete] +Type: filesandordirs; Name: "{app}" + +[CustomMessages] +ru.SetupWindowTitle=Установка Донатон Таймера +ru.DeleteSettingsPrompt=Вы хотите удалить настройки и все файлы программы? +en.SetupWindowTitle=Installation of Donathon Timer +en.DeleteSettingsPrompt=Do you want to delete settings and all program files? \ No newline at end of file