2025-02-05 14:57:07 +01:00
|
|
|
use gtk4 :: * ;
|
|
|
|
|
use gtk4 :: Align :: * ;
|
|
|
|
|
use gtk4 :: Orientation :: * ;
|
|
|
|
|
use gtk4 :: gdk :: * ;
|
|
|
|
|
use gtk4 :: gio :: * ;
|
2025-02-12 14:11:53 +01:00
|
|
|
use gtk4 :: glib :: * ;
|
2025-02-05 14:57:07 +01:00
|
|
|
use gtk4 :: pango :: * ;
|
|
|
|
|
use gtk4 :: pango :: Weight :: * ;
|
|
|
|
|
use gtk4 :: prelude :: * ;
|
|
|
|
|
use std :: cell :: * ;
|
|
|
|
|
use std :: iter :: * ;
|
|
|
|
|
use std :: path :: PathBuf ;
|
|
|
|
|
|
|
|
|
|
use crate :: ui :: collatable_container :: * ;
|
|
|
|
|
use crate :: ui :: component :: * ;
|
2024-11-26 17:20:53 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-02-12 14:11:53 +01:00
|
|
|
pub struct CollatedMediaGrid < A : MediaAdapter + 'static > {
|
2025-02-10 23:56:48 +01:00
|
|
|
media_widget_pairs : RefCell < Vec < ( A :: Overview , Button ) > > ,
|
2024-11-27 11:47:20 +01:00
|
|
|
grid_widget : FlowBox ,
|
2025-02-12 14:11:53 +01:00
|
|
|
on_media_selected : & 'static dyn Fn ( A :: Overview ) ,
|
2024-11-27 11:47:20 +01:00
|
|
|
}
|
2024-11-26 17:20:53 +01:00
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
impl < A : MediaAdapter > CollatedMediaGrid <A> {
|
2025-02-12 14:11:53 +01:00
|
|
|
pub fn new ( on_media_selected : impl Fn ( A :: Overview ) + 'static ) -> Self {
|
2025-01-31 17:38:52 +01:00
|
|
|
let grid_widget = flow_box ! (
|
2025-02-04 15:02:58 +01:00
|
|
|
@ orientation : Horizontal ;
|
|
|
|
|
@ homogeneous : true ;
|
|
|
|
|
@ selection_mode : SelectionMode :: None ;
|
2025-01-31 17:38:52 +01:00
|
|
|
) ;
|
2025-02-10 23:56:48 +01:00
|
|
|
let media_widget_pairs = RefCell :: new ( vec ! () ) ;
|
2025-02-12 14:11:53 +01:00
|
|
|
let on_media_selected = leak (on_media_selected) ;
|
2024-11-27 11:47:20 +01:00
|
|
|
|
2025-02-12 14:11:53 +01:00
|
|
|
Self { media_widget_pairs , grid_widget , on_media_selected }
|
2024-11-27 11:47:20 +01:00
|
|
|
}
|
|
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
pub async fn set_media ( & self , media : Vec < A :: Overview > , sorting : A :: Sorting ) {
|
|
|
|
|
// TODO: Check if we should use `MainContext :: invoke_local` here
|
2024-11-28 21:35:55 +01:00
|
|
|
|
2025-02-05 13:51:22 +01:00
|
|
|
let mut widgets = Vec :: new () ;
|
2025-02-10 23:56:48 +01:00
|
|
|
for media in media . as_slice () {
|
2025-02-12 14:11:53 +01:00
|
|
|
widgets . push ( self . create_media_entry (media) . await ) ;
|
2025-02-05 13:51:22 +01:00
|
|
|
}
|
2025-02-10 23:56:48 +01:00
|
|
|
self . media_widget_pairs . replace ( zip ( media , widgets ) . collect () ) ;
|
2024-11-27 11:47:20 +01:00
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
for ( _ , widget ) in self . sort_media_widget_pairs (sorting) {
|
|
|
|
|
self . grid_widget . append ( & widget ) ;
|
2024-11-26 17:20:53 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-12 14:11:53 +01:00
|
|
|
async fn create_media_entry ( & self , media : & A :: Overview ) -> Button {
|
|
|
|
|
button ! (
|
|
|
|
|
@ css_classes : & [ "flat" , "open-collection-item-button" ] ;
|
|
|
|
|
@ connect_clicked : clone ! (
|
|
|
|
|
# [strong] media ,
|
|
|
|
|
# [ strong ( rename_to = on_media_selected ) ] self . on_media_selected ,
|
|
|
|
|
move |_| on_media_selected ( media . clone () ) ,
|
|
|
|
|
) ;
|
|
|
|
|
& g_box ! (
|
|
|
|
|
@ option_children ;
|
|
|
|
|
@ orientation : Vertical ;
|
|
|
|
|
@ margin_top : 20 ;
|
|
|
|
|
@ margin_bottom : 20 ;
|
|
|
|
|
|
|
|
|
|
match media . get_poster_file_path () {
|
|
|
|
|
Some (poster_file_path) => {
|
|
|
|
|
let poster_file_path = PathBuf :: from (poster_file_path) ; // God forbid `Path` would work with `clone ! ()`
|
|
|
|
|
let poster_texture = spawn_blocking (
|
|
|
|
|
move || Texture :: from_filename (poster_file_path) . unwrap () ,
|
|
|
|
|
) . await . unwrap () ;
|
|
|
|
|
Some ( image ! (
|
|
|
|
|
@ paintable : & poster_texture ;
|
|
|
|
|
@ width_request : 300 ;
|
|
|
|
|
@ height_request : 300 ;
|
|
|
|
|
@ margin_bottom : 10 ;
|
|
|
|
|
) )
|
|
|
|
|
} ,
|
|
|
|
|
None => None ,
|
|
|
|
|
} . as_ref () ,
|
|
|
|
|
|
|
|
|
|
Some ( & label ! (
|
|
|
|
|
@ justify : Justification :: Center ;
|
|
|
|
|
@ wrap : true ;
|
|
|
|
|
@ max_width_chars : 1 ; // Not the actual limit, used instead to wrap more aggressively
|
|
|
|
|
@ attributes : & pango_attributes ! ( @ scale : SCALE_LARGE ; @ weight : Bold ; ) ;
|
|
|
|
|
media . get_name () . as_str () ,
|
|
|
|
|
) ) ,
|
|
|
|
|
|
|
|
|
|
match media . get_original_name () {
|
|
|
|
|
Some (original_name) => Some ( label ! (
|
|
|
|
|
@ justify : Justification :: Center ;
|
|
|
|
|
@ wrap : true ;
|
|
|
|
|
@ max_width_chars : 1 ; // Not the actual limit, used instead to wrap more aggressively
|
|
|
|
|
original_name . as_str () ,
|
|
|
|
|
) ) ,
|
|
|
|
|
None => None ,
|
|
|
|
|
} . as_ref () ,
|
|
|
|
|
|
|
|
|
|
Some ( & g_box ! (
|
|
|
|
|
@ option_children ;
|
|
|
|
|
@ orientation : Horizontal ;
|
|
|
|
|
@ halign : Center ;
|
|
|
|
|
@ spacing : 20 ;
|
|
|
|
|
Some ( & label ! ( media . get_release_date () . as_str () ) ) ,
|
|
|
|
|
match media . get_runtime_minutes () {
|
|
|
|
|
Some (runtime_minutes) => Some (
|
|
|
|
|
label ! ( format ! ( "{}m" , runtime_minutes ) . as_str () ) ,
|
|
|
|
|
) ,
|
|
|
|
|
None => None ,
|
|
|
|
|
} . as_ref () ,
|
|
|
|
|
) ) ,
|
|
|
|
|
) ,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
pub fn set_sorting ( & self , sorting : A :: Sorting ) {
|
2024-11-27 11:47:20 +01:00
|
|
|
self . grid_widget . remove_all () ;
|
|
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
for ( _ , widget ) in self . sort_media_widget_pairs (sorting) {
|
|
|
|
|
self . grid_widget . append ( & widget ) ;
|
2024-11-27 11:47:20 +01:00
|
|
|
}
|
|
|
|
|
}
|
2024-11-28 21:35:55 +01:00
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
fn sort_media_widget_pairs ( & self , sorting : A :: Sorting ) -> Vec < ( A :: Overview , Button ) > {
|
2024-11-28 21:35:55 +01:00
|
|
|
let mut sorted = Vec :: from (
|
2025-02-10 23:56:48 +01:00
|
|
|
self . media_widget_pairs . borrow () . as_slice () ,
|
|
|
|
|
) ;
|
2024-11-28 21:35:55 +01:00
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
sorted . sort_by ( | ( media_1 , _ ) , ( media_2 , _ ) |
|
|
|
|
|
A :: compare_by ( media_1 , media_2 , sorting ) ,
|
|
|
|
|
) ;
|
2024-11-28 21:35:55 +01:00
|
|
|
|
2025-01-31 17:09:22 +01:00
|
|
|
// See it, say it, ...
|
2024-11-28 21:35:55 +01:00
|
|
|
sorted
|
|
|
|
|
}
|
2024-11-27 15:22:55 +01:00
|
|
|
}
|
2024-11-27 11:47:20 +01:00
|
|
|
|
2025-02-10 23:56:48 +01:00
|
|
|
impl < A : MediaAdapter > Component for CollatedMediaGrid <A> {
|
2024-11-27 15:22:55 +01:00
|
|
|
fn get_widget ( & self ) -> & FlowBox { & self . grid_widget }
|
2024-11-27 11:47:20 +01:00
|
|
|
}
|