feat: ffmpeg image loader

This commit is contained in:
2024-10-22 23:03:49 +01:00
parent d21a45c941
commit c62fbfe510
30 changed files with 331 additions and 925 deletions

View File

@ -1,27 +1,47 @@
use egui::Image;
use crate::services::ffmpeg_loader::FfmpegLoader;
use crate::theme::NEUTRAL_800;
use anyhow::Error;
use eframe::epaint::Color32;
use egui::load::SizedTexture;
use egui::{ColorImage, Context, Image, ImageData, TextureHandle, TextureOptions};
use itertools::Itertools;
use log::{error, info};
use lru::LruCache;
use nostr_sdk::util::hex;
use sha2::{Digest, Sha256};
use std::collections::HashSet;
use std::fs;
use std::num::NonZeroUsize;
use std::ops::Deref;
use std::path::PathBuf;
use std::sync::Arc;
use tokio::sync::Mutex;
use std::sync::{Arc, Mutex};
type ImageCacheStore = Arc<Mutex<LruCache<String, TextureHandle>>>;
pub struct ImageCache {
ctx: egui::Context,
ctx: Context,
dir: PathBuf,
fetch_lock: Arc<Mutex<HashSet<String>>>,
placeholder: TextureHandle,
cache: ImageCacheStore,
fetch_cache: Arc<Mutex<HashSet<String>>>,
}
impl ImageCache {
pub fn new(data_path: PathBuf, ctx: egui::Context) -> Self {
pub fn new(data_path: PathBuf, ctx: Context) -> Self {
let out = data_path.join("cache/images");
fs::create_dir_all(&out).unwrap();
let placeholder = ctx.load_texture(
"placeholder",
ImageData::from(ColorImage::new([1, 1], NEUTRAL_800)),
TextureOptions::default(),
);
Self {
ctx,
dir: out,
fetch_lock: Arc::new(Mutex::new(HashSet::new())),
placeholder,
cache: Arc::new(Mutex::new(LruCache::new(NonZeroUsize::new(100).unwrap()))),
fetch_cache: Arc::new(Mutex::new(HashSet::new())),
}
}
@ -42,29 +62,61 @@ impl ImageCache {
U: Into<String>,
{
let u = url.into();
if let Ok(mut c) = self.cache.lock() {
if let Some(i) = c.get(&u) {
return Image::from_texture(i);
}
}
let path = self.find(&u);
if !path.exists() && !u.is_empty() {
let path = path.clone();
let fl = self.fetch_lock.clone();
let cache = self.cache.clone();
let ctx = self.ctx.clone();
let fetch_cache = self.fetch_cache.clone();
let placeholder = self.placeholder.clone();
tokio::spawn(async move {
if fl.lock().await.insert(u.clone()) {
if fetch_cache.lock().unwrap().insert(u.clone()) {
info!("Fetching image: {}", &u);
if let Ok(data) = reqwest::get(&u)
.await {
tokio::fs::create_dir_all(path.parent().unwrap()).await.unwrap();
if let Err(e) = tokio::fs::write(path, data.bytes().await.unwrap()).await {
if let Ok(data) = reqwest::get(&u).await {
tokio::fs::create_dir_all(path.parent().unwrap())
.await
.unwrap();
let img_data = data.bytes().await.unwrap();
if let Err(e) = tokio::fs::write(path.clone(), img_data).await {
error!("Failed to write file: {}", e);
}
// forget cached url
for t in ctx.loaders().texture.lock().iter() {
t.forget(&u);
}
let t = Self::load_image(&ctx, path, &u)
.await
.unwrap_or(placeholder);
cache.lock().unwrap().put(u.clone(), t);
ctx.request_repaint();
}
}
});
} else if path.exists() {
let path = path.clone();
let ctx = self.ctx.clone();
let cache = self.cache.clone();
let placeholder = self.placeholder.clone();
tokio::spawn(async move {
let t = Self::load_image(&ctx, path, &u)
.await
.unwrap_or(placeholder);
cache.lock().unwrap().put(u.clone(), t);
ctx.request_repaint();
});
}
Image::from_uri(format!("file://{}", path.to_str().unwrap()))
Image::from_texture(&self.placeholder)
}
}
async fn load_image(ctx: &Context, path: PathBuf, key: &str) -> Option<TextureHandle> {
let mut loader = FfmpegLoader::new();
match loader.load_image(path) {
Ok(i) => Some(ctx.load_texture(key, ImageData::from(i), TextureOptions::default())),
Err(e) => {
println!("Failed to load image: {}", e);
None
}
}
}
}