mirror of
https://github.com/lifegpc/eh_downloader_flutter.git
synced 2026-06-06 05:49:03 +08:00
312 lines
9.6 KiB
Dart
312 lines
9.6 KiB
Dart
import 'dart:typed_data';
|
|
import 'dart:ui';
|
|
import 'package:dio/dio.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:path/path.dart';
|
|
import 'package:palette_generator/palette_generator.dart';
|
|
import '../api/client.dart';
|
|
import '../api/file.dart';
|
|
import '../api/gallery.dart';
|
|
import '../globals.dart';
|
|
import '../utils.dart';
|
|
import '../utils/clipboard.dart';
|
|
import '../viewer/single.dart';
|
|
import 'image.dart';
|
|
|
|
final _log = Logger("Thumbnail");
|
|
|
|
class Thumbnail extends StatefulWidget {
|
|
const Thumbnail(ExtendedPMeta pMeta,
|
|
{Key? key,
|
|
int? max,
|
|
int? width,
|
|
int? height,
|
|
int? fileId,
|
|
this.gid,
|
|
this.index,
|
|
this.files,
|
|
this.gdata})
|
|
: _pMeta = pMeta,
|
|
_max = max ?? 1200,
|
|
_width = width,
|
|
_height = height,
|
|
_fileId = fileId,
|
|
super(key: key);
|
|
final ExtendedPMeta _pMeta;
|
|
final int _max;
|
|
final int? _width;
|
|
final int? _height;
|
|
final int? _fileId;
|
|
final int? gid;
|
|
final int? index;
|
|
final EhFiles? files;
|
|
final GalleryData? gdata;
|
|
|
|
int get height => _height != null
|
|
? _height!
|
|
: _pMeta.height > _pMeta.width
|
|
? _max
|
|
: _max * _pMeta.height ~/ _pMeta.width;
|
|
int get width => _width != null
|
|
? _width!
|
|
: _pMeta.width > _pMeta.height
|
|
? _max
|
|
: _max * _pMeta.width ~/ _pMeta.height;
|
|
|
|
@override
|
|
State<Thumbnail> createState() => _Thumbnail();
|
|
}
|
|
|
|
enum _ThumbnailMenu {
|
|
copyImage,
|
|
copyImgUrl,
|
|
saveAs,
|
|
}
|
|
|
|
class _Thumbnail extends State<Thumbnail> {
|
|
Uint8List? _data;
|
|
bool _isLoading = false;
|
|
Object? _error;
|
|
int? _fileId;
|
|
bool _showNsfw = false;
|
|
String? _uri;
|
|
CancelToken? _cancel;
|
|
String? _fileName;
|
|
String _dir = "";
|
|
Color? _iconColor;
|
|
double? _iconSize;
|
|
bool _disposed = false;
|
|
Future<void> _fetchData() async {
|
|
try {
|
|
_cancel = CancelToken();
|
|
_isLoading = true;
|
|
if (_fileId == null) {
|
|
final token = widget._pMeta.token;
|
|
_fileId = (await api.getFiles([token], cancel: _cancel))
|
|
.unwrap()
|
|
.files[token]![0]!
|
|
.id;
|
|
}
|
|
final re = await api.getThumbnail(_fileId!,
|
|
max: widget._max,
|
|
width: widget._width,
|
|
height: widget._height,
|
|
method: ThumbnailMethod.contain,
|
|
align: ThumbnailAlign.center,
|
|
cancel: _cancel);
|
|
if (re.response.statusCode != 200) {
|
|
throw Exception(
|
|
'Failed to get thumbnail: ${re.response.statusCode} ${re.response.statusMessage}');
|
|
}
|
|
_uri = re.response.realUri.toString();
|
|
final data = Uint8List.fromList(re.data);
|
|
if (!_cancel!.isCancelled) {
|
|
setState(() {
|
|
_isLoading = false;
|
|
_data = data;
|
|
_cancel = null;
|
|
});
|
|
updateIconColor();
|
|
}
|
|
} catch (e) {
|
|
if (!_cancel!.isCancelled) {
|
|
_log.warning("Failed to get file data:", e);
|
|
setState(() {
|
|
_isLoading = false;
|
|
_error = e;
|
|
_cancel = null;
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
@override
|
|
void initState() {
|
|
_data = null;
|
|
_isLoading = false;
|
|
_error = null;
|
|
_fileId = widget._fileId;
|
|
_showNsfw = false;
|
|
_uri = null;
|
|
_fileName = "${basenameWithoutExtension(widget._pMeta.name)}_thumb";
|
|
_dir = isAndroid && widget.gid != null ? widget.gid!.toString() : "";
|
|
super.initState();
|
|
}
|
|
|
|
Future<void> updateIconColor() async {
|
|
if (_data == null) return;
|
|
try {
|
|
final img = await instantiateImageCodec(_data!);
|
|
try {
|
|
final frame = await img.getNextFrame();
|
|
final i = frame.image;
|
|
try {
|
|
final iconSize = _iconSize ?? 24.0;
|
|
final pattle = await PaletteGenerator.fromImage(i,
|
|
region: Rect.fromCenter(
|
|
center: Offset(i.width / 2, i.height / 2),
|
|
width: iconSize,
|
|
height: iconSize));
|
|
if (!_disposed) {
|
|
setState(() {
|
|
_iconColor = pattle.colors.first.computeLuminance() > 0.5
|
|
? Colors.black
|
|
: Colors.white;
|
|
});
|
|
}
|
|
} finally {
|
|
i.dispose();
|
|
}
|
|
} finally {
|
|
img.dispose();
|
|
}
|
|
} catch (e) {
|
|
_log.warning("Failed to generate icon's color from image data:", e);
|
|
}
|
|
}
|
|
|
|
bool get showNsfw => _showNsfw || (prefs.getBool("showNsfw") ?? false);
|
|
|
|
@override
|
|
void dispose() {
|
|
_disposed = true;
|
|
_cancel?.cancel();
|
|
super.dispose();
|
|
}
|
|
|
|
Future<void> onItemSelected(_ThumbnailMenu v) async {
|
|
switch (v) {
|
|
case _ThumbnailMenu.copyImage:
|
|
try {
|
|
copyImageToClipboard(_data!, ImageFmt.jpg);
|
|
} catch (err) {
|
|
_log.warning("Failed to copy image to clipboard:", err);
|
|
}
|
|
break;
|
|
case _ThumbnailMenu.copyImgUrl:
|
|
try {
|
|
copyTextToClipboard(_uri!);
|
|
} catch (err) {
|
|
_log.warning("Failed to copy image url to clipboard:", err);
|
|
}
|
|
break;
|
|
case _ThumbnailMenu.saveAs:
|
|
try {
|
|
await platformPath.saveFile(_fileName!, "image/jpeg", _data!,
|
|
dir: _dir);
|
|
} catch (err) {
|
|
_log.warning("Failed to save image:", err);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final isLoading = _data == null && _error == null;
|
|
final isNsfw = widget._pMeta.isNsfw;
|
|
if (isLoading && !_isLoading) _fetchData();
|
|
_iconSize ??= Theme.of(context).iconTheme.size;
|
|
final iconSize = MediaQuery.of(context).size.width < 400
|
|
? 14.0
|
|
: Theme.of(context).iconTheme.size;
|
|
final moreVertMenu = Positioned(
|
|
right: 0,
|
|
top: 0,
|
|
width: iconSize,
|
|
height: iconSize,
|
|
child: PopupMenuButton(
|
|
child: Icon(Icons.more_vert, size: iconSize),
|
|
onSelected: (v) {
|
|
onItemSelected(v);
|
|
},
|
|
itemBuilder: (context) {
|
|
var list = <PopupMenuEntry<_ThumbnailMenu>>[
|
|
PopupMenuItem(
|
|
value: _ThumbnailMenu.copyImage,
|
|
child: Text(AppLocalizations.of(context)!.copyImage)),
|
|
PopupMenuItem(
|
|
value: _ThumbnailMenu.copyImgUrl,
|
|
child: Text(AppLocalizations.of(context)!.copyImgUrl)),
|
|
PopupMenuItem(
|
|
value: _ThumbnailMenu.saveAs,
|
|
child: Text(AppLocalizations.of(context)!.saveAs)),
|
|
];
|
|
return list;
|
|
}));
|
|
final timg = _data != null
|
|
? ImageWithContextMenu(_data!,
|
|
uri: _uri, fileName: _fileName, dir: _dir)
|
|
: null;
|
|
final img = widget.gid != null && widget.index != null && _data != null
|
|
? GestureDetector(
|
|
onTap: () {
|
|
context.push("/gallery/${widget.gid}/page/${widget.index}",
|
|
extra: SinglePageViewerExtra(
|
|
data: widget.gdata, files: widget.files));
|
|
},
|
|
child: timg)
|
|
: timg;
|
|
return SizedBox(
|
|
width: widget.width.toDouble(),
|
|
height: widget.height.toDouble(),
|
|
child: isLoading
|
|
? const Center(child: CircularProgressIndicator())
|
|
: _data != null
|
|
? isNsfw && !showNsfw
|
|
? Stack(
|
|
children: [
|
|
SizedBox(
|
|
width: widget.width.toDouble(),
|
|
height: widget.height.toDouble(),
|
|
child: ImageFiltered(
|
|
imageFilter: ImageFilter.blur(
|
|
sigmaX: 10,
|
|
sigmaY: 10,
|
|
tileMode: TileMode.decal),
|
|
child: img)),
|
|
SizedBox(
|
|
width: widget.width.toDouble(),
|
|
height: widget.height.toDouble(),
|
|
child: Center(
|
|
child: IconButton(
|
|
onPressed: () {
|
|
setState(() {
|
|
_showNsfw = true;
|
|
});
|
|
},
|
|
icon: Icon(Icons.visibility,
|
|
color: _iconColor ?? Colors.black),
|
|
),
|
|
)),
|
|
moreVertMenu
|
|
],
|
|
)
|
|
: Stack(children: [
|
|
SizedBox(
|
|
width: widget.width.toDouble(),
|
|
height: widget.height.toDouble(),
|
|
child: img),
|
|
moreVertMenu
|
|
])
|
|
: Center(
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
SelectableText("Error $_error"),
|
|
ElevatedButton.icon(
|
|
onPressed: () {
|
|
_fetchData();
|
|
setState(() {
|
|
_error = null;
|
|
});
|
|
},
|
|
icon: const Icon(Icons.refresh),
|
|
label: Text(AppLocalizations.of(context)!.retry))
|
|
])));
|
|
}
|
|
}
|