mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-23 17:26:18 +01:00
318 lines
9.8 KiB
Dart
318 lines
9.8 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:nc_photos/account.dart';
|
|
import 'package:nc_photos/api/api.dart';
|
|
import 'package:nc_photos/app_localizations.dart';
|
|
import 'package:nc_photos/entity/file.dart';
|
|
import 'package:nc_photos/entity/file_util.dart' as file_util;
|
|
import 'package:nc_photos/help_utils.dart';
|
|
import 'package:nc_photos/k.dart' as k;
|
|
import 'package:nc_photos/mobile/android/android_info.dart';
|
|
import 'package:nc_photos/mobile/android/permission_util.dart';
|
|
import 'package:nc_photos/object_extension.dart';
|
|
import 'package:nc_photos/platform/k.dart' as platform_k;
|
|
import 'package:nc_photos/pref.dart';
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
|
import 'package:nc_photos/theme.dart';
|
|
import 'package:nc_photos/widget/settings.dart';
|
|
import 'package:nc_photos/widget/stateful_slider.dart';
|
|
import 'package:nc_photos_plugin/nc_photos_plugin.dart';
|
|
import 'package:url_launcher/url_launcher.dart';
|
|
|
|
class EnhanceHandler {
|
|
const EnhanceHandler({
|
|
required this.account,
|
|
required this.file,
|
|
});
|
|
|
|
static bool isSupportedFormat(File file) =>
|
|
file_util.isSupportedImageFormat(file) && file.contentType != "image/gif";
|
|
|
|
Future<void> call(BuildContext context) async {
|
|
if (!Pref().hasShownEnhanceInfoOr()) {
|
|
await _showInfo(context);
|
|
}
|
|
|
|
if (!await _ensurePermission()) {
|
|
return;
|
|
}
|
|
|
|
final selected = await _pickAlgorithm(context);
|
|
if (selected == null) {
|
|
// user canceled
|
|
return;
|
|
}
|
|
_log.info("[call] Selected: ${selected.name}");
|
|
final args = await _getArgs(context, selected);
|
|
if (args == null) {
|
|
// user canceled
|
|
return;
|
|
}
|
|
switch (selected) {
|
|
case _Algorithm.zeroDce:
|
|
await ImageProcessor.zeroDce(
|
|
"${account.url}/${file.path}",
|
|
file.filename,
|
|
Pref().getEnhanceMaxWidthOr(),
|
|
Pref().getEnhanceMaxHeightOr(),
|
|
args["iteration"] ?? 8,
|
|
headers: {
|
|
"Authorization": Api.getAuthorizationHeaderValue(account),
|
|
},
|
|
);
|
|
break;
|
|
|
|
case _Algorithm.deepLab3Portrait:
|
|
await ImageProcessor.deepLab3Portrait(
|
|
"${account.url}/${file.path}",
|
|
file.filename,
|
|
Pref().getEnhanceMaxWidthOr(),
|
|
Pref().getEnhanceMaxHeightOr(),
|
|
args["radius"] ?? 16,
|
|
headers: {
|
|
"Authorization": Api.getAuthorizationHeaderValue(account),
|
|
},
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
|
|
Future<void> _showInfo(BuildContext context) async {
|
|
await showDialog(
|
|
context: context,
|
|
builder: (context) => AlertDialog(
|
|
title: Text(L10n.global().enhanceIntroDialogTitle),
|
|
content: Text(L10n.global().enhanceIntroDialogDescription),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
launch(enhanceUrl);
|
|
},
|
|
child: Text(L10n.global().learnMoreButtonLabel),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
Navigator.of(context).pushNamed(EnhancementSettings.routeName);
|
|
},
|
|
child: Text(L10n.global().configButtonLabel),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
Navigator.of(context).pop();
|
|
},
|
|
child: Text(MaterialLocalizations.of(context).closeButtonLabel),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
Pref().setHasShownEnhanceInfo(true);
|
|
}
|
|
|
|
Future<bool> _ensurePermission() async {
|
|
if (platform_k.isAndroid) {
|
|
if (AndroidInfo().sdkInt < AndroidVersion.R &&
|
|
!await Permission.hasWriteExternalStorage()) {
|
|
final results = await requestPermissionsForResult([
|
|
Permission.WRITE_EXTERNAL_STORAGE,
|
|
]);
|
|
if (results[Permission.WRITE_EXTERNAL_STORAGE] !=
|
|
PermissionRequestResult.granted) {
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
content: Text(L10n.global().errorNoStoragePermission),
|
|
duration: k.snackBarDurationNormal,
|
|
));
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Future<_Algorithm?> _pickAlgorithm(BuildContext context) =>
|
|
showDialog<_Algorithm>(
|
|
context: context,
|
|
builder: (context) => SimpleDialog(
|
|
children: _getOptions()
|
|
.map((o) => SimpleDialogOption(
|
|
padding: const EdgeInsets.all(0),
|
|
child: ListTile(
|
|
title: Text(o.title),
|
|
subtitle: o.subtitle?.run((t) => Text(t)),
|
|
trailing: o.link != null
|
|
? SizedBox(
|
|
height: double.maxFinite,
|
|
child: TextButton(
|
|
child: Text(L10n.global().detailsTooltip),
|
|
onPressed: () {
|
|
launch(o.link!);
|
|
},
|
|
),
|
|
)
|
|
: null,
|
|
onTap: () {
|
|
Navigator.of(context).pop(o.algorithm);
|
|
},
|
|
),
|
|
))
|
|
.toList(),
|
|
),
|
|
);
|
|
|
|
List<_Option> _getOptions() => [
|
|
if (platform_k.isAndroid)
|
|
_Option(
|
|
title: L10n.global().enhanceLowLightTitle,
|
|
subtitle: "Zero-DCE",
|
|
link: enhanceZeroDceUrl,
|
|
algorithm: _Algorithm.zeroDce,
|
|
),
|
|
if (platform_k.isAndroid)
|
|
_Option(
|
|
title: L10n.global().enhancePortraitBlurTitle,
|
|
subtitle: "DeepLap v3",
|
|
link: enhanceDeepLabPortraitBlurUrl,
|
|
algorithm: _Algorithm.deepLab3Portrait,
|
|
),
|
|
];
|
|
|
|
Future<Map<String, dynamic>?> _getArgs(
|
|
BuildContext context, _Algorithm selected) async {
|
|
switch (selected) {
|
|
case _Algorithm.zeroDce:
|
|
return _getZeroDceArgs(context);
|
|
|
|
case _Algorithm.deepLab3Portrait:
|
|
return _getDeepLab3PortraitArgs(context);
|
|
}
|
|
}
|
|
|
|
Future<Map<String, dynamic>?> _getZeroDceArgs(BuildContext context) async {
|
|
var current = .8;
|
|
final iteration = await showDialog<int>(
|
|
context: context,
|
|
builder: (context) => AppTheme(
|
|
child: AlertDialog(
|
|
title: Text(L10n.global().enhanceLowLightParamBrightnessLabel),
|
|
contentPadding: const EdgeInsets.fromLTRB(24.0, 20.0, 24.0, 0),
|
|
content: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
Row(
|
|
mainAxisSize: MainAxisSize.max,
|
|
children: [
|
|
Icon(
|
|
Icons.brightness_low,
|
|
color: AppTheme.getSecondaryTextColor(context),
|
|
),
|
|
Expanded(
|
|
child: StatefulSlider(
|
|
initialValue: current,
|
|
onChangeEnd: (value) {
|
|
current = value;
|
|
},
|
|
),
|
|
),
|
|
Icon(
|
|
Icons.brightness_high,
|
|
color: AppTheme.getSecondaryTextColor(context),
|
|
),
|
|
],
|
|
),
|
|
],
|
|
),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
final iteration = (current * 10).round().clamp(1, 10);
|
|
Navigator.of(context).pop(iteration);
|
|
},
|
|
child: Text(L10n.global().enhanceButtonLabel),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
_log.info("[_getZeroDceArgs] iteration: $iteration");
|
|
return iteration?.run((it) => {"iteration": it});
|
|
}
|
|
|
|
Future<Map<String, dynamic>?> _getDeepLab3PortraitArgs(
|
|
BuildContext context) async {
|
|
var current = .5;
|
|
final radius = await showDialog<int>(
|
|
context: context,
|
|
builder: (context) => AppTheme(
|
|
child: AlertDialog(
|
|
title: Text(L10n.global().enhancePortraitBlurParamBlurLabel),
|
|
contentPadding: const EdgeInsets.fromLTRB(24.0, 20.0, 24.0, 0),
|
|
content: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
Row(
|
|
mainAxisSize: MainAxisSize.max,
|
|
children: [
|
|
Icon(
|
|
Icons.circle,
|
|
size: 20,
|
|
color: AppTheme.getSecondaryTextColor(context),
|
|
),
|
|
Expanded(
|
|
child: StatefulSlider(
|
|
initialValue: current,
|
|
onChangeEnd: (value) {
|
|
current = value;
|
|
},
|
|
),
|
|
),
|
|
Icon(
|
|
Icons.blur_on,
|
|
color: AppTheme.getSecondaryTextColor(context),
|
|
),
|
|
],
|
|
),
|
|
],
|
|
),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
final radius = (current * 25).round().clamp(1, 25);
|
|
Navigator.of(context).pop(radius);
|
|
},
|
|
child: Text(L10n.global().enhanceButtonLabel),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
_log.info("[_getDeepLab3PortraitArgs] radius: $radius");
|
|
return radius?.run((it) => {"radius": it});
|
|
}
|
|
|
|
final Account account;
|
|
final File file;
|
|
|
|
static final _log = Logger("widget.handler.enhance_handler.EnhanceHandler");
|
|
}
|
|
|
|
enum _Algorithm {
|
|
zeroDce,
|
|
deepLab3Portrait,
|
|
}
|
|
|
|
class _Option {
|
|
const _Option({
|
|
required this.title,
|
|
this.subtitle,
|
|
this.link,
|
|
required this.algorithm,
|
|
});
|
|
|
|
final String title;
|
|
final String? subtitle;
|
|
final String? link;
|
|
final _Algorithm algorithm;
|
|
}
|