mirror of
https://github.com/venera-app/venera.git
synced 2025-09-27 15:57:25 +00:00
156 lines
4.6 KiB
Dart
156 lines
4.6 KiB
Dart
import 'dart:async' show Future, StreamController;
|
|
import 'package:crypto/crypto.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:venera/foundation/app.dart';
|
|
import 'package:venera/foundation/comic_source/comic_source.dart';
|
|
import 'package:venera/foundation/comic_type.dart';
|
|
import 'package:venera/foundation/local.dart';
|
|
import 'package:venera/network/images.dart';
|
|
import 'package:venera/utils/io.dart';
|
|
import '../history.dart';
|
|
import 'base_image_provider.dart';
|
|
import 'image_favorites_provider.dart' as image_provider;
|
|
|
|
class ImageFavoritesProvider
|
|
extends BaseImageProvider<image_provider.ImageFavoritesProvider> {
|
|
/// Image provider for imageFavorites
|
|
const ImageFavoritesProvider(this.imageFavorite);
|
|
|
|
final ImageFavorite imageFavorite;
|
|
|
|
int get page => imageFavorite.page;
|
|
|
|
String get sourceKey => imageFavorite.sourceKey;
|
|
|
|
String get cid => imageFavorite.id;
|
|
|
|
String get eid => imageFavorite.eid;
|
|
|
|
@override
|
|
Future<Uint8List> load(
|
|
StreamController<ImageChunkEvent>? chunkEvents,
|
|
void Function()? checkStop,
|
|
) async {
|
|
var imageKey = imageFavorite.imageKey;
|
|
var localImage = await getImageFromLocal();
|
|
checkStop?.call();
|
|
if (localImage != null) {
|
|
return localImage;
|
|
}
|
|
var cacheImage = await readFromCache();
|
|
checkStop?.call();
|
|
if (cacheImage != null) {
|
|
return cacheImage;
|
|
}
|
|
var gotImageKey = false;
|
|
if (imageKey == "") {
|
|
imageKey = await getImageKey();
|
|
checkStop?.call();
|
|
gotImageKey = true;
|
|
}
|
|
Uint8List image;
|
|
try {
|
|
image = await getImageFromNetwork(imageKey, chunkEvents, checkStop);
|
|
} catch (e) {
|
|
if (gotImageKey) {
|
|
rethrow;
|
|
} else {
|
|
imageKey = await getImageKey();
|
|
image = await getImageFromNetwork(imageKey, chunkEvents, checkStop);
|
|
}
|
|
}
|
|
await writeToCache(image);
|
|
return image;
|
|
}
|
|
|
|
Future<void> writeToCache(Uint8List image) async {
|
|
var fileName = md5.convert(key.codeUnits).toString();
|
|
var file = File(FilePath.join(App.cachePath, 'image_favorites', fileName));
|
|
if (!file.existsSync()) {
|
|
file.createSync(recursive: true);
|
|
}
|
|
await file.writeAsBytes(image);
|
|
}
|
|
|
|
Future<Uint8List?> readFromCache() async {
|
|
var fileName = md5.convert(key.codeUnits).toString();
|
|
var file = File(FilePath.join(App.cachePath, 'image_favorites', fileName));
|
|
if (!file.existsSync()) {
|
|
return null;
|
|
}
|
|
return await file.readAsBytes();
|
|
}
|
|
|
|
/// Delete a image favorite cache
|
|
static Future<void> deleteFromCache(ImageFavorite imageFavorite) async {
|
|
var fileName = md5.convert(imageFavorite.imageKey.codeUnits).toString();
|
|
var file = File(FilePath.join(App.cachePath, 'image_favorites', fileName));
|
|
if (file.existsSync()) {
|
|
await file.delete();
|
|
}
|
|
}
|
|
|
|
Future<Uint8List?> getImageFromLocal() async {
|
|
var localComic =
|
|
LocalManager().find(sourceKey, ComicType.fromKey(sourceKey));
|
|
if (localComic == null) {
|
|
return null;
|
|
}
|
|
var epIndex = localComic.chapters?.ids.toList().indexOf(eid) ?? -1;
|
|
if (epIndex == -1 && localComic.hasChapters) {
|
|
return null;
|
|
}
|
|
var images = await LocalManager().getImages(
|
|
sourceKey,
|
|
ComicType.fromKey(sourceKey),
|
|
epIndex,
|
|
);
|
|
var data = await File(images[page]).readAsBytes();
|
|
return data;
|
|
}
|
|
|
|
Future<Uint8List> getImageFromNetwork(
|
|
String imageKey,
|
|
StreamController<ImageChunkEvent>? chunkEvents,
|
|
void Function()? checkStop,
|
|
) async {
|
|
await for (var progress
|
|
in ImageDownloader.loadComicImage(imageKey, sourceKey, cid, eid)) {
|
|
checkStop?.call();
|
|
if (chunkEvents != null) {
|
|
chunkEvents.add(ImageChunkEvent(
|
|
cumulativeBytesLoaded: progress.currentBytes,
|
|
expectedTotalBytes: progress.totalBytes,
|
|
));
|
|
}
|
|
if (progress.imageBytes != null) {
|
|
return progress.imageBytes!;
|
|
}
|
|
}
|
|
throw "Error: Empty response body.";
|
|
}
|
|
|
|
Future<String> getImageKey() async {
|
|
String sourceKey = imageFavorite.sourceKey;
|
|
String cid = imageFavorite.id;
|
|
String eid = imageFavorite.eid;
|
|
var page = imageFavorite.page;
|
|
var comicSource = ComicSource.find(sourceKey);
|
|
if (comicSource == null) {
|
|
throw "Error: Comic source not found.";
|
|
}
|
|
var res = await comicSource.loadComicPages!(cid, eid);
|
|
return res.data[page - 1];
|
|
}
|
|
|
|
@override
|
|
Future<ImageFavoritesProvider> obtainKey(ImageConfiguration configuration) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
String get key =>
|
|
"ImageFavorites ${imageFavorite.imageKey}@${imageFavorite.sourceKey}@${imageFavorite.id}@${imageFavorite.eid}";
|
|
}
|