2022-12-31 08:51:54 +01:00
|
|
|
import 'dart:async';
|
|
|
|
|
2023-05-28 19:11:34 +02:00
|
|
|
import 'package:copy_with/copy_with.dart';
|
2022-11-26 10:51:54 +01:00
|
|
|
import 'package:flutter/material.dart';
|
2022-12-31 08:51:54 +01:00
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
2022-11-26 10:51:54 +01:00
|
|
|
import 'package:flutter_colorpicker/flutter_colorpicker.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/app_localizations.dart';
|
2023-07-28 18:48:50 +02:00
|
|
|
import 'package:nc_photos/bloc_util.dart';
|
2023-08-19 18:47:56 +02:00
|
|
|
import 'package:nc_photos/controller/pref_controller.dart';
|
|
|
|
import 'package:nc_photos/exception_event.dart';
|
|
|
|
import 'package:nc_photos/exception_util.dart' as exception_util;
|
2022-11-26 10:51:54 +01:00
|
|
|
import 'package:nc_photos/k.dart' as k;
|
|
|
|
import 'package:nc_photos/mobile/android/android_info.dart';
|
2023-08-19 18:47:56 +02:00
|
|
|
import 'package:nc_photos/object_extension.dart';
|
2022-11-26 10:51:54 +01:00
|
|
|
import 'package:nc_photos/platform/k.dart' as platform_k;
|
2023-07-24 15:06:59 +02:00
|
|
|
import 'package:nc_photos/session_storage.dart';
|
2022-11-26 10:51:54 +01:00
|
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
|
|
|
import 'package:nc_photos/theme.dart';
|
2023-08-19 18:47:56 +02:00
|
|
|
import 'package:nc_photos/widget/page_visibility_mixin.dart';
|
2022-12-16 16:01:04 +01:00
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2023-05-28 19:11:34 +02:00
|
|
|
import 'package:to_string/to_string.dart';
|
2022-12-16 16:01:04 +01:00
|
|
|
|
2023-05-28 19:11:34 +02:00
|
|
|
part 'theme/bloc.dart';
|
|
|
|
part 'theme/state_event.dart';
|
2022-12-16 16:01:04 +01:00
|
|
|
part 'theme_settings.g.dart';
|
2022-11-26 10:51:54 +01:00
|
|
|
|
2023-08-19 18:47:56 +02:00
|
|
|
// typedef _BlocBuilder = BlocBuilder<_Bloc, _State>;
|
|
|
|
typedef _BlocListener = BlocListener<_Bloc, _State>;
|
|
|
|
typedef _BlocSelector<T> = BlocSelector<_Bloc, _State, T>;
|
2023-07-24 15:06:59 +02:00
|
|
|
|
2022-12-31 08:51:54 +01:00
|
|
|
class ThemeSettings extends StatelessWidget {
|
2022-11-26 10:51:54 +01:00
|
|
|
const ThemeSettings({super.key});
|
|
|
|
|
|
|
|
@override
|
2022-12-31 08:51:54 +01:00
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return BlocProvider(
|
2023-08-19 18:47:56 +02:00
|
|
|
create: (_) => _Bloc(
|
|
|
|
prefController: context.read(),
|
|
|
|
),
|
2022-12-31 08:51:54 +01:00
|
|
|
child: const _WrappedThemeSettings(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _WrappedThemeSettings extends StatefulWidget {
|
|
|
|
const _WrappedThemeSettings();
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _WrappedThemeSettingsState();
|
2022-11-26 10:51:54 +01:00
|
|
|
}
|
|
|
|
|
2022-12-16 16:01:04 +01:00
|
|
|
@npLog
|
2023-08-19 18:47:56 +02:00
|
|
|
class _WrappedThemeSettingsState extends State<_WrappedThemeSettings>
|
|
|
|
with RouteAware, PageVisibilityMixin {
|
2022-11-26 10:51:54 +01:00
|
|
|
@override
|
2022-12-31 08:51:54 +01:00
|
|
|
void initState() {
|
2022-11-26 10:51:54 +01:00
|
|
|
super.initState();
|
2023-08-19 18:47:56 +02:00
|
|
|
_bloc.add(const _Init());
|
2022-11-26 10:51:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2022-12-31 08:51:54 +01:00
|
|
|
Widget build(BuildContext context) {
|
2022-11-26 10:51:54 +01:00
|
|
|
return Scaffold(
|
2023-08-19 18:47:56 +02:00
|
|
|
body: MultiBlocListener(
|
|
|
|
listeners: [
|
|
|
|
_BlocListener(
|
|
|
|
listenWhen: (previous, current) => previous.error != current.error,
|
|
|
|
listener: (context, state) {
|
|
|
|
if (state.error != null && isPageVisible()) {
|
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content:
|
|
|
|
Text(exception_util.toUserString(state.error!.error)),
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
child: CustomScrollView(
|
|
|
|
slivers: [
|
|
|
|
SliverAppBar(
|
|
|
|
pinned: true,
|
|
|
|
title: Text(L10n.global().settingsThemeTitle),
|
|
|
|
),
|
|
|
|
SliverList(
|
|
|
|
delegate: SliverChildListDelegate(
|
|
|
|
[
|
|
|
|
const _SeedColorOption(),
|
|
|
|
if (platform_k.isAndroid &&
|
|
|
|
AndroidInfo().sdkInt >= AndroidVersion.Q)
|
|
|
|
_BlocSelector<bool>(
|
|
|
|
selector: (state) => state.isFollowSystemTheme,
|
|
|
|
builder: (_, isFollowSystemTheme) {
|
|
|
|
return SwitchListTile(
|
|
|
|
title: Text(
|
|
|
|
L10n.global().settingsFollowSystemThemeTitle),
|
|
|
|
value: isFollowSystemTheme,
|
|
|
|
onChanged: (value) {
|
|
|
|
_bloc.add(_SetFollowSystemTheme(value));
|
|
|
|
},
|
|
|
|
);
|
2022-12-31 08:51:54 +01:00
|
|
|
},
|
2023-08-19 18:47:56 +02:00
|
|
|
),
|
|
|
|
_BlocSelector<bool>(
|
|
|
|
selector: (state) => state.isUseBlackInDarkTheme,
|
|
|
|
builder: (context, isUseBlackInDarkTheme) {
|
|
|
|
return SwitchListTile(
|
|
|
|
title: Text(
|
|
|
|
L10n.global().settingsUseBlackInDarkThemeTitle),
|
|
|
|
subtitle: Text(isUseBlackInDarkTheme
|
|
|
|
? L10n.global()
|
|
|
|
.settingsUseBlackInDarkThemeTrueDescription
|
|
|
|
: L10n.global()
|
|
|
|
.settingsUseBlackInDarkThemeFalseDescription),
|
|
|
|
value: isUseBlackInDarkTheme,
|
|
|
|
onChanged: (value) {
|
|
|
|
_bloc.add(_SetUseBlackInDarkTheme(
|
|
|
|
value, Theme.of(context)));
|
|
|
|
},
|
|
|
|
);
|
2022-12-31 08:51:54 +01:00
|
|
|
},
|
2023-08-19 18:47:56 +02:00
|
|
|
),
|
|
|
|
],
|
2022-11-26 10:51:54 +01:00
|
|
|
),
|
2023-08-19 18:47:56 +02:00
|
|
|
),
|
|
|
|
],
|
2022-11-26 10:51:54 +01:00
|
|
|
),
|
2023-08-19 18:47:56 +02:00
|
|
|
),
|
2022-11-26 10:51:54 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-07-24 15:06:59 +02:00
|
|
|
late final _bloc = context.read<_Bloc>();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorOption extends StatelessWidget {
|
|
|
|
const _SeedColorOption();
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-08-19 18:47:56 +02:00
|
|
|
return _BlocSelector<int?>(
|
|
|
|
selector: (state) => state.seedColor,
|
|
|
|
builder: (context, seedColor) {
|
2023-07-24 15:06:59 +02:00
|
|
|
if (SessionStorage().isSupportDynamicColor) {
|
|
|
|
return ListTile(
|
|
|
|
title: Text(L10n.global().settingsSeedColorTitle),
|
2023-08-19 18:47:56 +02:00
|
|
|
subtitle: Text(seedColor == null
|
2023-07-24 15:06:59 +02:00
|
|
|
? L10n.global().settingsSeedColorSystemColorDescription
|
|
|
|
: L10n.global().settingsSeedColorDescription),
|
2023-08-19 18:47:56 +02:00
|
|
|
trailing: seedColor == null
|
2023-07-24 15:06:59 +02:00
|
|
|
? null
|
|
|
|
: Icon(
|
|
|
|
Icons.circle,
|
|
|
|
size: 32,
|
2023-08-19 18:47:56 +02:00
|
|
|
color: Color(seedColor),
|
2023-07-24 15:06:59 +02:00
|
|
|
),
|
|
|
|
onTap: () => _onSeedColorPressed(context),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return ListTile(
|
|
|
|
title: Text(L10n.global().settingsSeedColorTitle),
|
|
|
|
subtitle: Text(L10n.global().settingsSeedColorDescription),
|
|
|
|
trailing: Icon(
|
|
|
|
Icons.circle,
|
|
|
|
size: 32,
|
2023-08-19 18:47:56 +02:00
|
|
|
color: seedColor?.run(Color.new) ?? defaultSeedColor,
|
2023-07-24 15:06:59 +02:00
|
|
|
),
|
|
|
|
onTap: () => _onSeedColorPressed(context),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-11-26 10:51:54 +01:00
|
|
|
Future<void> _onSeedColorPressed(BuildContext context) async {
|
2023-07-24 15:06:59 +02:00
|
|
|
final result = await showDialog<int>(
|
2022-11-26 10:51:54 +01:00
|
|
|
context: context,
|
|
|
|
builder: (context) => const _SeedColorPicker(),
|
|
|
|
);
|
|
|
|
if (result == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-24 15:06:59 +02:00
|
|
|
if (context.mounted) {
|
|
|
|
context
|
|
|
|
.read<_Bloc>()
|
|
|
|
.add(_SetSeedColor(result == -1 ? null : Color(result)));
|
2022-11-26 10:51:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorPicker extends StatefulWidget {
|
|
|
|
const _SeedColorPicker();
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _SeedColorPickerState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorPickerState extends State<_SeedColorPicker> {
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Visibility(
|
|
|
|
visible: _isVisible,
|
|
|
|
child: AlertDialog(
|
|
|
|
title: Text(L10n.global().settingsSeedColorPickerTitle),
|
|
|
|
content: Wrap(
|
|
|
|
children: const [
|
|
|
|
Color(0xFFF44336),
|
|
|
|
Color(0xFF9C27B0),
|
|
|
|
Color(0xFF2196F3),
|
|
|
|
Color(0xFF4CAF50),
|
|
|
|
Color(0xFFFFC107),
|
|
|
|
null,
|
|
|
|
]
|
|
|
|
.map((c) => _SeedColorPickerItem(
|
|
|
|
seedColor: c,
|
2023-07-24 15:06:59 +02:00
|
|
|
onSelected: () => _onItemSelected(context, c?.value),
|
2022-11-26 10:51:54 +01:00
|
|
|
))
|
|
|
|
.toList(),
|
|
|
|
),
|
2023-07-24 15:06:59 +02:00
|
|
|
actions: SessionStorage().isSupportDynamicColor
|
|
|
|
? [
|
|
|
|
TextButton(
|
|
|
|
onPressed: () => _onItemSelected(context, -1),
|
|
|
|
child: Text(L10n.global()
|
|
|
|
.settingsSeedColorPickerSystemColorButtonLabel),
|
|
|
|
),
|
|
|
|
]
|
|
|
|
: null,
|
2022-11-26 10:51:54 +01:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-07-24 15:06:59 +02:00
|
|
|
Future<void> _onItemSelected(BuildContext context, int? seedColor) async {
|
2022-11-26 10:51:54 +01:00
|
|
|
if (seedColor != null) {
|
|
|
|
Navigator.of(context).pop(seedColor);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
_isVisible = false;
|
|
|
|
});
|
|
|
|
final color = await showDialog<Color>(
|
|
|
|
context: context,
|
2023-07-24 15:06:59 +02:00
|
|
|
builder: (_) => const _SeedColorCustomPicker(),
|
2022-11-26 10:51:54 +01:00
|
|
|
barrierColor: Colors.transparent,
|
|
|
|
);
|
2023-07-24 15:06:59 +02:00
|
|
|
Navigator.of(context).pop(color?.value);
|
2022-11-26 10:51:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var _isVisible = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorCustomPicker extends StatefulWidget {
|
|
|
|
const _SeedColorCustomPicker();
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _SeedColorCustomPickerState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorCustomPickerState extends State<_SeedColorCustomPicker> {
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return AlertDialog(
|
|
|
|
title: Text(L10n.global().settingsSeedColorPickerTitle),
|
|
|
|
content: SingleChildScrollView(
|
|
|
|
child: _HueRingPicker(
|
|
|
|
pickerColor: _customColor,
|
|
|
|
onColorChanged: (value) {
|
|
|
|
setState(() {
|
|
|
|
_customColor = value;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
actions: [
|
|
|
|
TextButton(
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop(_customColor);
|
|
|
|
},
|
|
|
|
child: Text(L10n.global().applyButtonLabel),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-08-19 18:47:56 +02:00
|
|
|
late var _customColor = getSeedColor(context) ?? defaultSeedColor;
|
2022-11-26 10:51:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
class _SeedColorPickerItem extends StatelessWidget {
|
|
|
|
const _SeedColorPickerItem({
|
|
|
|
required this.seedColor,
|
|
|
|
this.onSelected,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final content = SizedBox.square(
|
|
|
|
dimension: _size,
|
|
|
|
child: Center(
|
|
|
|
child: seedColor != null
|
|
|
|
? Icon(
|
|
|
|
Icons.circle,
|
|
|
|
size: _size * .9,
|
|
|
|
color: seedColor,
|
|
|
|
)
|
|
|
|
: Transform.scale(
|
|
|
|
scale: .9,
|
|
|
|
child: Stack(
|
|
|
|
alignment: Alignment.center,
|
|
|
|
children: [
|
|
|
|
Image.asset("assets/ic_custom_color_56dp.png"),
|
|
|
|
const Icon(
|
|
|
|
Icons.colorize_outlined,
|
|
|
|
size: _size * .5,
|
|
|
|
color: Colors.black87,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
if (onSelected != null) {
|
|
|
|
return InkWell(
|
|
|
|
customBorder: const CircleBorder(),
|
|
|
|
onTap: onSelected,
|
|
|
|
child: content,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return content;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final Color? seedColor;
|
|
|
|
final VoidCallback? onSelected;
|
|
|
|
|
|
|
|
static const _size = 56.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Based on the original HueRingPicker
|
|
|
|
class _HueRingPicker extends StatefulWidget {
|
|
|
|
const _HueRingPicker({
|
|
|
|
Key? key,
|
|
|
|
required this.pickerColor,
|
|
|
|
required this.onColorChanged,
|
|
|
|
// ignore: unused_element
|
|
|
|
this.colorPickerHeight = 250.0,
|
|
|
|
// ignore: unused_element
|
|
|
|
this.hueRingStrokeWidth = 20.0,
|
|
|
|
// ignore: unused_element
|
|
|
|
this.displayThumbColor = true,
|
|
|
|
// ignore: unused_element
|
|
|
|
this.pickerAreaBorderRadius = const BorderRadius.all(Radius.zero),
|
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
final Color pickerColor;
|
|
|
|
final ValueChanged<Color> onColorChanged;
|
|
|
|
final double colorPickerHeight;
|
|
|
|
final double hueRingStrokeWidth;
|
|
|
|
final bool displayThumbColor;
|
|
|
|
final BorderRadius pickerAreaBorderRadius;
|
|
|
|
|
|
|
|
@override
|
|
|
|
_HueRingPickerState createState() => _HueRingPickerState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _HueRingPickerState extends State<_HueRingPicker> {
|
|
|
|
HSVColor currentHsvColor = const HSVColor.fromAHSV(0.0, 0.0, 0.0, 0.0);
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
currentHsvColor = HSVColor.fromColor(widget.pickerColor);
|
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void didUpdateWidget(_HueRingPicker oldWidget) {
|
|
|
|
super.didUpdateWidget(oldWidget);
|
|
|
|
currentHsvColor = HSVColor.fromColor(widget.pickerColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void onColorChanging(HSVColor color) {
|
|
|
|
setState(() => currentHsvColor = color.withSaturation(1).withValue(1));
|
|
|
|
widget.onColorChanged(currentHsvColor.toColor());
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Stack(
|
|
|
|
children: <Widget>[
|
|
|
|
ClipRRect(
|
|
|
|
borderRadius: widget.pickerAreaBorderRadius,
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(18),
|
|
|
|
child: Stack(
|
|
|
|
alignment: AlignmentDirectional.center,
|
|
|
|
children: <Widget>[
|
|
|
|
ColorIndicator(
|
|
|
|
currentHsvColor,
|
|
|
|
width: 128,
|
|
|
|
height: 128,
|
|
|
|
),
|
|
|
|
SizedBox(
|
|
|
|
width: widget.colorPickerHeight,
|
|
|
|
height: widget.colorPickerHeight,
|
|
|
|
child: ColorPickerHueRing(
|
|
|
|
currentHsvColor,
|
|
|
|
onColorChanging,
|
|
|
|
displayThumbColor: widget.displayThumbColor,
|
|
|
|
strokeWidth: 26,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|