1
0
mirror of https://github.com/git-touch/git-touch synced 2025-02-07 23:28:54 +01:00

354 lines
9.2 KiB
Dart
Raw Normal View History

import 'dart:io';
2019-09-23 17:08:51 +08:00
import 'dart:async';
import 'package:fimber/fimber.dart';
2019-12-12 20:29:56 +08:00
import 'package:fluro/fluro.dart';
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
2019-12-25 10:46:31 +08:00
import 'package:git_touch/utils/utils.dart';
2019-11-03 23:33:24 +08:00
import 'package:git_touch/widgets/action_button.dart';
2020-01-27 10:47:34 +08:00
import 'package:primer/primer.dart';
import 'package:shared_preferences/shared_preferences.dart';
class DialogOption<T> {
final T value;
final Widget widget;
DialogOption({this.value, this.widget});
}
2019-09-19 21:10:50 +08:00
class AppThemeType {
static const material = 0;
static const cupertino = 1;
2019-09-19 21:10:50 +08:00
static const values = [AppThemeType.material, AppThemeType.cupertino];
}
2020-01-13 21:07:28 +08:00
class AppBrightnessType {
2020-01-14 18:13:07 +08:00
static const followSystem = 0;
2020-01-13 21:07:28 +08:00
static const light = 1;
static const dark = 2;
static const values = [
2020-01-14 18:13:07 +08:00
AppBrightnessType.followSystem,
2020-01-13 21:07:28 +08:00
AppBrightnessType.light,
AppBrightnessType.dark
];
}
2019-09-29 15:35:33 +08:00
class PickerItem<T> {
final T value;
final String text;
PickerItem(this.value, {@required this.text});
}
class PickerGroupItem<T> {
final T value;
final List<PickerItem<T>> items;
final Function(T value) onChange;
final Function(T value) onClose;
PickerGroupItem({
@required this.value,
@required this.items,
this.onChange,
this.onClose,
});
}
2019-10-02 14:58:11 +08:00
class SelectorItem<T> {
T value;
String text;
SelectorItem({@required this.value, @required this.text});
}
2019-09-29 16:01:03 +08:00
// No animation. For replacing route
// TODO: Go back
class StaticRoute extends PageRouteBuilder {
final WidgetBuilder builder;
StaticRoute({this.builder})
: super(
pageBuilder: (BuildContext context, Animation<double> animation,
Animation<double> secondaryAnimation) {
return builder(context);
},
transitionsBuilder: (BuildContext context,
Animation<double> animation,
Animation<double> secondaryAnimation,
Widget child) {
return child;
},
);
}
2019-11-05 21:22:41 +08:00
class Palette {
2020-01-14 18:13:07 +08:00
final Color primary;
final Color text;
final Color secondaryText;
final Color tertiaryText;
final Color background;
final Color grayBackground;
final Color border;
const Palette({
2019-11-05 21:22:41 +08:00
this.primary,
this.text,
this.secondaryText,
this.tertiaryText,
this.background,
2019-12-20 22:41:38 +08:00
this.grayBackground,
2019-11-05 21:22:41 +08:00
this.border,
});
}
class ThemeModel with ChangeNotifier {
int _theme;
int get theme => _theme;
bool get ready => _theme != null;
2020-01-16 12:45:04 +08:00
Brightness systemBrightness = Brightness.light;
void setSystemBrightness(Brightness v) {
if (v != systemBrightness) {
Future.microtask(() {
systemBrightness = v;
notifyListeners();
});
2020-01-14 18:13:07 +08:00
}
}
2020-01-16 12:45:04 +08:00
int _brightnessValue = AppBrightnessType.followSystem;
int get brighnessValue => _brightnessValue;
2020-01-14 18:13:07 +08:00
// could be null
Brightness get brightness {
2020-01-13 21:07:28 +08:00
switch (_brightnessValue) {
case AppBrightnessType.light:
return Brightness.light;
case AppBrightnessType.dark:
return Brightness.dark;
default:
2020-01-16 12:45:04 +08:00
return systemBrightness;
2020-01-13 21:07:28 +08:00
}
}
Future<void> setBrightness(int v) async {
_brightnessValue = v;
final prefs = await SharedPreferences.getInstance();
2020-01-16 12:52:47 +08:00
await prefs.setInt(StorageKeys.iBrightness, v);
2020-01-13 21:07:28 +08:00
Fimber.d('write brightness: $v');
2019-11-05 21:22:41 +08:00
notifyListeners();
}
2019-12-12 20:29:56 +08:00
final router = Router();
2020-01-14 18:13:07 +08:00
final paletteLight = Palette(
2020-01-27 10:47:34 +08:00
primary: PrimerColors.blue500,
2020-01-14 18:13:07 +08:00
text: Colors.black,
secondaryText: Colors.grey.shade800,
tertiaryText: Colors.grey.shade600,
background: Colors.white,
grayBackground: Colors.grey.shade100,
2020-01-27 10:47:34 +08:00
border: Colors.grey.shade300,
2020-01-14 18:13:07 +08:00
);
final paletteDark = Palette(
2020-01-27 10:47:34 +08:00
primary: PrimerColors.blue500,
2020-01-14 18:13:07 +08:00
text: Colors.grey.shade300,
secondaryText: Colors.grey.shade400,
tertiaryText: Colors.grey.shade500,
background: Colors.black,
grayBackground: Colors.grey.shade900,
border: Colors.grey.shade700,
);
2020-01-27 15:11:51 +08:00
Palette get palette {
2020-01-16 12:45:04 +08:00
switch (brightness) {
2019-11-05 21:22:41 +08:00
case Brightness.dark:
2020-01-14 18:13:07 +08:00
return paletteDark;
2020-01-27 15:11:51 +08:00
case Brightness.light:
2019-11-05 21:22:41 +08:00
default:
2020-01-27 15:11:51 +08:00
return paletteLight;
2019-11-05 21:22:41 +08:00
}
}
2019-11-05 15:09:54 +08:00
Future<void> init() async {
2020-01-13 21:07:28 +08:00
final prefs = await SharedPreferences.getInstance();
2020-01-16 12:52:47 +08:00
final v = prefs.getInt(StorageKeys.iTheme);
Fimber.d('read theme: $v');
2019-09-19 21:10:50 +08:00
if (AppThemeType.values.contains(v)) {
_theme = v;
} else if (Platform.isIOS) {
2019-09-19 21:10:50 +08:00
_theme = AppThemeType.cupertino;
} else {
2019-09-19 21:10:50 +08:00
_theme = AppThemeType.material;
}
2020-01-16 12:52:47 +08:00
final b = prefs.getInt(StorageKeys.iBrightness);
2020-01-13 21:07:28 +08:00
Fimber.d('read brightness: $b');
if (AppBrightnessType.values.contains(b)) {
_brightnessValue = b;
}
notifyListeners();
}
Future<void> setTheme(int v) async {
_theme = v;
2020-01-13 21:07:28 +08:00
final prefs = await SharedPreferences.getInstance();
2020-01-16 12:52:47 +08:00
await prefs.setInt(StorageKeys.iTheme, v);
Fimber.d('write theme: $v');
notifyListeners();
}
2019-12-25 10:46:31 +08:00
push(BuildContext context, String url, {bool replace = false}) {
2020-02-01 10:32:29 +08:00
// Fimber.d(url);
2019-12-25 10:46:31 +08:00
if (url.startsWith('/')) {
return router.navigateTo(
context,
url,
2020-04-06 13:18:23 +08:00
transition: theme == AppThemeType.cupertino
? TransitionType.cupertino
: TransitionType.material,
2019-12-25 10:46:31 +08:00
replace: replace,
);
} else {
launchUrl(url);
}
2019-12-12 20:29:56 +08:00
}
2020-02-08 15:06:36 +08:00
Future<void> showWarning(BuildContext context, String message) async {
showCupertinoDialog(
context: context,
builder: (context) {
return CupertinoAlertDialog(
title: Text(message),
actions: <Widget>[
CupertinoDialogAction(
child: const Text('OK'),
onPressed: () {
Navigator.pop(context, true);
},
),
],
);
},
);
}
2019-10-03 12:55:17 +08:00
Future<bool> showConfirm(BuildContext context, Widget content) {
2020-02-08 12:53:56 +08:00
return showCupertinoDialog(
context: context,
builder: (context) {
return CupertinoAlertDialog(
title: content,
actions: <Widget>[
CupertinoDialogAction(
child: const Text('cancel'),
isDefaultAction: true,
onPressed: () {
Navigator.pop(context, false);
},
),
CupertinoDialogAction(
child: const Text('OK'),
onPressed: () {
Navigator.pop(context, true);
},
),
],
);
2020-02-08 12:53:56 +08:00
},
);
// default:
// return showDialog(
// context: context,
// builder: (BuildContext context) {
// return AlertDialog(
// content: content,
// actions: <Widget>[
// FlatButton(
// child: const Text('CANCEL'),
// onPressed: () {
// Navigator.pop(context, false);
// },
// ),
// FlatButton(
// child: const Text('OK'),
// onPressed: () {
// Navigator.pop(context, true);
// },
// )
// ],
// );
// },
// );
}
2019-10-02 14:58:11 +08:00
static Timer _debounce;
2019-09-29 15:35:33 +08:00
String _selectedItem;
2019-09-23 17:08:51 +08:00
showPicker(BuildContext context, PickerGroupItem<String> groupItem) async {
2020-02-08 12:53:56 +08:00
await showCupertinoModalPopup(
context: context,
builder: (context) {
return Container(
height: 216,
child: CupertinoPicker(
backgroundColor: palette.background,
children: <Widget>[
for (var v in groupItem.items)
Text(v.text, style: TextStyle(color: palette.text)),
],
itemExtent: 40,
scrollController: FixedExtentScrollController(
initialItem: groupItem.items
.toList()
.indexWhere((v) => v.value == groupItem.value)),
onSelectedItemChanged: (index) {
_selectedItem = groupItem.items[index].value;
if (groupItem.onChange != null) {
if (_debounce?.isActive ?? false) {
_debounce.cancel();
}
_debounce = Timer(const Duration(milliseconds: 500), () {
groupItem.onChange(_selectedItem);
});
}
},
),
2019-09-23 17:08:51 +08:00
);
2020-02-08 12:53:56 +08:00
},
);
if (groupItem.onClose != null) {
groupItem.onClose(_selectedItem);
2019-09-23 17:08:51 +08:00
}
}
2019-11-03 23:33:24 +08:00
showActions(BuildContext context, List<ActionItem> actionItems) async {
2019-12-12 14:02:48 +08:00
if (actionItems == null) return;
2019-11-03 23:33:24 +08:00
final value = await showCupertinoModalPopup<int>(
context: context,
builder: (BuildContext context) {
return CupertinoActionSheet(
title: Text('Actions'),
actions: actionItems.asMap().entries.map((entry) {
return CupertinoActionSheetAction(
child: Text(entry.value.text),
2020-02-01 18:30:32 +08:00
isDestructiveAction: entry.value.isDestructiveAction,
2019-11-03 23:33:24 +08:00
onPressed: () {
Navigator.pop(context, entry.key);
},
);
}).toList(),
cancelButton: CupertinoActionSheetAction(
child: const Text('Cancel'),
isDefaultAction: true,
onPressed: () {
Navigator.pop(context);
},
),
);
},
);
if (value != null) {
2020-01-01 20:59:20 +08:00
actionItems[value].onTap(context);
2019-11-03 23:33:24 +08:00
}
}
}