204 lines
6.3 KiB
Rust
204 lines
6.3 KiB
Rust
use gtk::prelude::*;
|
|
use gtk;
|
|
use gdk_pixbuf::Pixbuf;
|
|
|
|
use std::fs;
|
|
|
|
use hammond_data::models::Podcast;
|
|
use hammond_downloader::downloader;
|
|
use hammond_data::index_feed::Database;
|
|
use hammond_data::dbqueries;
|
|
|
|
use widgets::episode::episodes_listbox;
|
|
use podcasts_view::update_podcasts_view;
|
|
|
|
// http://gtk-rs.org/tuto/closures
|
|
macro_rules! clone {
|
|
(@param _) => ( _ );
|
|
(@param $x:ident) => ( $x );
|
|
($($n:ident),+ => move || $body:expr) => (
|
|
{
|
|
$( let $n = $n.clone(); )+
|
|
move || $body
|
|
}
|
|
);
|
|
($($n:ident),+ => move |$($p:tt),+| $body:expr) => (
|
|
{
|
|
$( let $n = $n.clone(); )+
|
|
move |$(clone!(@param $p),)+| $body
|
|
}
|
|
);
|
|
}
|
|
|
|
fn podcast_widget(db: &Database, stack: >k::Stack, pd: &Podcast) -> gtk::Box {
|
|
// Adapted from gnome-music AlbumWidget
|
|
let pd_widget_source = include_str!("../../gtk/podcast_widget.ui");
|
|
let pd_widget_buidler = gtk::Builder::new_from_string(pd_widget_source);
|
|
let pd_widget: gtk::Box = pd_widget_buidler.get_object("podcast_widget").unwrap();
|
|
|
|
let cover: gtk::Image = pd_widget_buidler.get_object("cover").unwrap();
|
|
let title_label: gtk::Label = pd_widget_buidler.get_object("title_label").unwrap();
|
|
// let desc_label: gtk::Label = pd_widget_buidler.get_object("description_label").unwrap();
|
|
let desc_text_view: gtk::TextView = pd_widget_buidler.get_object("desc_text_view").unwrap();
|
|
let view: gtk::Viewport = pd_widget_buidler.get_object("view").unwrap();
|
|
let unsub_button: gtk::Button = pd_widget_buidler.get_object("unsub_button").unwrap();
|
|
let played_button: gtk::Button = pd_widget_buidler
|
|
.get_object("mark_all_played_button")
|
|
.unwrap();
|
|
|
|
// TODO: should spawn a thread to avoid locking the UI probably.
|
|
unsub_button.connect_clicked(clone!(db, stack, pd => move |bttn| {
|
|
on_unsub_button_clicked(&db, &stack, &pd, bttn);
|
|
}));
|
|
|
|
title_label.set_text(pd.title());
|
|
let listbox = episodes_listbox(db, pd.title());
|
|
if let Ok(l) = listbox {
|
|
view.add(&l);
|
|
}
|
|
|
|
{
|
|
let buff = desc_text_view.get_buffer().unwrap();
|
|
buff.set_text(pd.description());
|
|
}
|
|
|
|
let img = get_pixbuf_from_path(pd.image_uri(), pd.title());
|
|
if let Some(i) = img {
|
|
cover.set_from_pixbuf(&i);
|
|
}
|
|
|
|
played_button.connect_clicked(clone!(db, stack, pd => move |_| {
|
|
on_played_button_clicked(&db, &stack, &pd);
|
|
}));
|
|
|
|
show_played_button(db, pd, &played_button);
|
|
|
|
pd_widget
|
|
}
|
|
|
|
fn on_unsub_button_clicked(
|
|
db: &Database,
|
|
stack: >k::Stack,
|
|
pd: &Podcast,
|
|
unsub_button: >k::Button,
|
|
) {
|
|
let res = dbqueries::remove_feed(db, pd);
|
|
if res.is_ok() {
|
|
info!("{} was removed succesfully.", pd.title());
|
|
// hack to get away without properly checking for none.
|
|
// if pressed twice would panic.
|
|
unsub_button.hide();
|
|
|
|
let dl_fold = downloader::get_download_folder(pd.title());
|
|
if let Ok(fold) = dl_fold {
|
|
let res3 = fs::remove_dir_all(&fold);
|
|
if res3.is_ok() {
|
|
info!("All the content at, {} was removed succesfully", &fold);
|
|
}
|
|
};
|
|
}
|
|
update_podcasts_view(db, stack);
|
|
stack.set_visible_child_name("pd_grid")
|
|
}
|
|
|
|
fn on_played_button_clicked(db: &Database, stack: >k::Stack, pd: &Podcast) {
|
|
{
|
|
let tempdb = db.lock().unwrap();
|
|
let _ = dbqueries::update_none_to_played_now(&tempdb, pd);
|
|
}
|
|
|
|
update_podcast_widget(db, stack, pd);
|
|
}
|
|
|
|
fn show_played_button(db: &Database, pd: &Podcast, played_button: >k::Button) {
|
|
let new_episodes = {
|
|
let tempdb = db.lock().unwrap();
|
|
dbqueries::get_pd_unplayed_episodes(&tempdb, pd)
|
|
};
|
|
|
|
if let Ok(n) = new_episodes {
|
|
if !n.is_empty() {
|
|
played_button.show()
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn create_flowbox_child(db: &Database, pd: &Podcast) -> gtk::FlowBoxChild {
|
|
let build_src = include_str!("../../gtk/podcasts_child.ui");
|
|
let builder = gtk::Builder::new_from_string(build_src);
|
|
|
|
// Copy of gnome-music AlbumWidget
|
|
let box_: gtk::Box = builder.get_object("fb_child").unwrap();
|
|
let pd_title: gtk::Label = builder.get_object("pd_title").unwrap();
|
|
let pd_cover: gtk::Image = builder.get_object("pd_cover").unwrap();
|
|
let banner: gtk::Image = builder.get_object("banner").unwrap();
|
|
let banner_title: gtk::Label = builder.get_object("banner_label").unwrap();
|
|
|
|
pd_title.set_text(pd.title());
|
|
|
|
let cover = get_pixbuf_from_path(pd.image_uri(), pd.title());
|
|
if let Some(img) = cover {
|
|
pd_cover.set_from_pixbuf(&img);
|
|
};
|
|
|
|
configure_banner(db, pd, &banner, &banner_title);
|
|
|
|
let fbc = gtk::FlowBoxChild::new();
|
|
fbc.add(&box_);
|
|
// info!("flowbox child created");
|
|
fbc
|
|
}
|
|
|
|
fn configure_banner(db: &Database, pd: &Podcast, banner: >k::Image, banner_title: >k::Label) {
|
|
let bann = Pixbuf::new_from_file_at_scale("assets/banner.png", 256, 256, true);
|
|
if let Ok(b) = bann {
|
|
banner.set_from_pixbuf(&b);
|
|
|
|
let new_episodes = {
|
|
let tempdb = db.lock().unwrap();
|
|
dbqueries::get_pd_unplayed_episodes(&tempdb, pd)
|
|
};
|
|
|
|
if let Ok(n) = new_episodes {
|
|
if !n.is_empty() {
|
|
banner_title.set_text(&n.len().to_string());
|
|
banner.show();
|
|
banner_title.show();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_flowbox_child_activate(db: &Database, stack: >k::Stack, parent: &Podcast) {
|
|
let old = stack.get_child_by_name("pdw").unwrap();
|
|
let pdw = podcast_widget(db, stack, parent);
|
|
|
|
stack.remove(&old);
|
|
stack.add_named(&pdw, "pdw");
|
|
stack.set_visible_child(&pdw);
|
|
|
|
// aggresive memory cleanup
|
|
// probably not needed
|
|
old.destroy();
|
|
println!("Hello World!, child activated");
|
|
}
|
|
|
|
fn get_pixbuf_from_path(img_path: Option<&str>, pd_title: &str) -> Option<Pixbuf> {
|
|
let img_path = downloader::cache_image(pd_title, img_path);
|
|
if let Some(i) = img_path {
|
|
Pixbuf::new_from_file_at_scale(&i, 256, 256, true).ok()
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
pub fn update_podcast_widget(db: &Database, stack: >k::Stack, pd: &Podcast) {
|
|
let old = stack.get_child_by_name("pdw").unwrap();
|
|
let pdw = podcast_widget(db, stack, pd);
|
|
let vis = stack.get_visible_child_name().unwrap();
|
|
|
|
stack.remove(&old);
|
|
stack.add_named(&pdw, "pdw");
|
|
stack.set_visible_child_name(&vis);
|
|
}
|