Merge branch 'appmenu' into 'master'
Appmenu See merge request World/hammond!33
This commit is contained in:
commit
666ab01d03
@ -257,12 +257,12 @@ Tobias Bernard
|
|||||||
</object>
|
</object>
|
||||||
</child>
|
</child>
|
||||||
<child>
|
<child>
|
||||||
<object class="GtkMenuButton" id="menu_toggle">
|
<object class="GtkMenuButton" id="menu_button">
|
||||||
<property name="visible">True</property>
|
<property name="visible">True</property>
|
||||||
<property name="can_focus">True</property>
|
<property name="can_focus">True</property>
|
||||||
<property name="receives_default">False</property>
|
<property name="receives_default">False</property>
|
||||||
<property name="valign">center</property>
|
<property name="valign">center</property>
|
||||||
<property name="popover">menu_popover</property>
|
<property name="use_popover">True</property>
|
||||||
<child>
|
<child>
|
||||||
<object class="GtkImage">
|
<object class="GtkImage">
|
||||||
<property name="visible">True</property>
|
<property name="visible">True</property>
|
||||||
@ -281,125 +281,4 @@ Tobias Bernard
|
|||||||
</packing>
|
</packing>
|
||||||
</child>
|
</child>
|
||||||
</object>
|
</object>
|
||||||
<object class="GtkPopover" id="menu_popover">
|
</interface>
|
||||||
<property name="can_focus">False</property>
|
|
||||||
<property name="relative_to">menu_toggle</property>
|
|
||||||
<child>
|
|
||||||
<object class="GtkBox">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="can_focus">False</property>
|
|
||||||
<property name="margin_left">10</property>
|
|
||||||
<property name="margin_right">10</property>
|
|
||||||
<property name="margin_top">10</property>
|
|
||||||
<property name="margin_bottom">10</property>
|
|
||||||
<property name="orientation">vertical</property>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="sensitive">False</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">False</property>
|
|
||||||
<property name="text" translatable="yes">Preferences</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">0</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton" id="update_button">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">True</property>
|
|
||||||
<property name="text" translatable="yes">Check for new episodes</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">1</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton" id="import">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">True</property>
|
|
||||||
<property name="text" translatable="yes">Import Shows</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">2</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton" id="export">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="sensitive">False</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">False</property>
|
|
||||||
<property name="text" translatable="yes">Export Shows</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">3</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkSeparator">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="can_focus">False</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">4</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton" id="about">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">False</property>
|
|
||||||
<property name="text" translatable="yes">About</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">6</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="sensitive">False</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">False</property>
|
|
||||||
<property name="text" translatable="yes">Help</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">7</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
<child>
|
|
||||||
<object class="GtkModelButton">
|
|
||||||
<property name="visible">True</property>
|
|
||||||
<property name="sensitive">False</property>
|
|
||||||
<property name="can_focus">True</property>
|
|
||||||
<property name="receives_default">False</property>
|
|
||||||
<property name="text" translatable="yes">Keyboard Shortcuts</property>
|
|
||||||
</object>
|
|
||||||
<packing>
|
|
||||||
<property name="expand">False</property>
|
|
||||||
<property name="fill">True</property>
|
|
||||||
<property name="position">8</property>
|
|
||||||
</packing>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</child>
|
|
||||||
</object>
|
|
||||||
</interface>
|
|
||||||
33
hammond-gtk/resources/gtk/help-overlay.ui
Normal file
33
hammond-gtk/resources/gtk/help-overlay.ui
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<interface>
|
||||||
|
<object class="GtkShortcutsWindow" id="help_overlay">
|
||||||
|
<property name="modal">True</property>
|
||||||
|
<child>
|
||||||
|
<object class="GtkShortcutsSection">
|
||||||
|
<property name="section-name">shortcuts</property>
|
||||||
|
<property name="max-height">12</property>
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<child>
|
||||||
|
<object class="GtkShortcutsGroup">
|
||||||
|
<property name="title" translatable="yes">General</property>
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<child>
|
||||||
|
<object class="GtkShortcutsShortcut">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="accelerator"><primary>r</property>
|
||||||
|
<property name="title" translatable="yes" context="shortcut window">Check for new episodes</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkShortcutsShortcut">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="accelerator"><primary>q</property>
|
||||||
|
<property name="title" translatable="yes" context="shortcut window">Quit the application</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</interface>
|
||||||
41
hammond-gtk/resources/gtk/menus.ui
Normal file
41
hammond-gtk/resources/gtk/menus.ui
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
<?xml version="1.0"?>
|
||||||
|
<interface>
|
||||||
|
<!-- interface-requires gtk+ 3.0 -->
|
||||||
|
<menu id="menu">
|
||||||
|
<section>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Check for new episodes</attribute>
|
||||||
|
<attribute name="action">app.refresh</attribute>
|
||||||
|
<attribute name="accel"><primary>r</attribute>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Import Shows</attribute>
|
||||||
|
<attribute name="action">app.import</attribute>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Export Shows</attribute>
|
||||||
|
<attribute name="action">app.export</attribute>
|
||||||
|
</item>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Preferences</attribute>
|
||||||
|
<attribute name="action">app.preferences</attribute>
|
||||||
|
</item>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Keyboard Shortcuts</attribute>
|
||||||
|
<attribute name="action">win.show-help-overlay</attribute>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_Help</attribute>
|
||||||
|
<attribute name="action">app.help</attribute>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<attribute name="label" translatable="yes">_About</attribute>
|
||||||
|
<attribute name="action">app.about</attribute>
|
||||||
|
</item>
|
||||||
|
</section>
|
||||||
|
</menu>
|
||||||
|
</interface>
|
||||||
@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<gresources>
|
<gresources>
|
||||||
<gresource prefix="/org/gnome/hammond/">
|
<gresource prefix="/org/gnome/Hammond/">
|
||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/episode_widget.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/episode_widget.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/show_widget.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/show_widget.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/empty_view.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/empty_view.ui</file>
|
||||||
@ -11,6 +11,8 @@
|
|||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/shows_child.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/shows_child.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/headerbar.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/headerbar.ui</file>
|
||||||
<file compressed="true" preprocess="xml-stripblanks">gtk/inapp_notif.ui</file>
|
<file compressed="true" preprocess="xml-stripblanks">gtk/inapp_notif.ui</file>
|
||||||
|
<file compressed="true" preprocess="xml-stripblanks">gtk/menus.ui</file>
|
||||||
|
<file compressed="true" preprocess="xml-stripblanks">gtk/help-overlay.ui</file>
|
||||||
<file compressed="true">gtk/style.css</file>
|
<file compressed="true">gtk/style.css</file>
|
||||||
</gresource>
|
</gresource>
|
||||||
</gresources>
|
</gresources>
|
||||||
|
|||||||
@ -1,6 +1,9 @@
|
|||||||
#![allow(new_without_default)]
|
#![allow(new_without_default)]
|
||||||
|
|
||||||
use gio::{ApplicationExt, ApplicationExtManual, ApplicationFlags, Settings, SettingsExt};
|
use gio::{
|
||||||
|
ActionMapExt, ApplicationExt, ApplicationExtManual, ApplicationFlags, Settings, SettingsExt,
|
||||||
|
SimpleAction, SimpleActionExt,
|
||||||
|
};
|
||||||
use glib;
|
use glib;
|
||||||
use gtk;
|
use gtk;
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
@ -13,12 +16,14 @@ use headerbar::Header;
|
|||||||
use settings::{self, WindowGeometry};
|
use settings::{self, WindowGeometry};
|
||||||
use stacks::{Content, PopulatedState};
|
use stacks::{Content, PopulatedState};
|
||||||
use utils;
|
use utils;
|
||||||
use widgets::{mark_all_notif, remove_show_notif};
|
use widgets::{about_dialog, mark_all_notif, remove_show_notif};
|
||||||
|
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::sync::mpsc::{channel, Receiver, Sender};
|
use std::sync::mpsc::{channel, Sender};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use std::cell::RefCell;
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub enum Action {
|
pub enum Action {
|
||||||
RefreshAllViews,
|
RefreshAllViews,
|
||||||
@ -41,12 +46,6 @@ pub enum Action {
|
|||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct App {
|
pub struct App {
|
||||||
app_instance: gtk::Application,
|
app_instance: gtk::Application,
|
||||||
window: gtk::Window,
|
|
||||||
overlay: gtk::Overlay,
|
|
||||||
header: Rc<Header>,
|
|
||||||
content: Rc<Content>,
|
|
||||||
receiver: Receiver<Action>,
|
|
||||||
sender: Sender<Action>,
|
|
||||||
settings: Settings,
|
settings: Settings,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -63,62 +62,178 @@ impl App {
|
|||||||
let cleanup_date = settings::get_cleanup_date(&settings);
|
let cleanup_date = settings::get_cleanup_date(&settings);
|
||||||
utils::cleanup(cleanup_date);
|
utils::cleanup(cleanup_date);
|
||||||
|
|
||||||
// Create the main window
|
application.connect_startup(clone!(settings => move |app| {
|
||||||
let window = gtk::Window::new(gtk::WindowType::Toplevel);
|
let (sender, receiver) = channel();
|
||||||
window.set_title("Hammond");
|
let receiver = Rc::new(RefCell::new(receiver));
|
||||||
|
|
||||||
window.connect_delete_event(clone!(application, settings, window => move |_, _| {
|
let refresh = SimpleAction::new("refresh", None);
|
||||||
WindowGeometry::from_window(&window).write(&settings);
|
refresh.connect_activate(clone!(sender => move |_, _| {
|
||||||
application.quit();
|
gtk::idle_add(clone!(sender => move || {
|
||||||
Inhibit(false)
|
let s: Option<Vec<_>> = None;
|
||||||
|
utils::refresh(s, sender.clone());
|
||||||
|
glib::Continue(false)
|
||||||
|
}));
|
||||||
|
}));
|
||||||
|
app.add_action(&refresh);
|
||||||
|
app.set_accels_for_action("app.refresh", &["<primary>r"]);
|
||||||
|
|
||||||
|
let import = SimpleAction::new("import", None);
|
||||||
|
import.connect_activate(clone!(sender, app => move |_, _| {
|
||||||
|
let window = app.get_active_window().expect("Failed to get active window");
|
||||||
|
utils::on_import_clicked(&window, &sender);
|
||||||
|
}));
|
||||||
|
app.add_action(&import);
|
||||||
|
|
||||||
|
let about = SimpleAction::new("about", None);
|
||||||
|
about.connect_activate(clone!(app => move |_, _| {
|
||||||
|
let window = app.get_active_window().expect("Failed to get active window");
|
||||||
|
about_dialog(&window);
|
||||||
|
}));
|
||||||
|
app.add_action(&about);
|
||||||
|
|
||||||
|
let quit = SimpleAction::new("quit", None);
|
||||||
|
quit.connect_activate(clone!(app => move |_, _| app.quit()));
|
||||||
|
app.add_action(&quit);
|
||||||
|
app.set_accels_for_action("app.quit", &["<primary>q"]);
|
||||||
|
|
||||||
|
app.set_accels_for_action("win.menu", &["F10"]);
|
||||||
|
|
||||||
|
app.connect_activate(clone!(sender, settings, receiver => move |app| {
|
||||||
|
// Get the current window (if any)
|
||||||
|
if let Some(window) = app.get_active_window() {
|
||||||
|
// Already open, just raise the window
|
||||||
|
window.present();
|
||||||
|
} else {
|
||||||
|
// Time to open one!
|
||||||
|
// Create the main window
|
||||||
|
let window = gtk::ApplicationWindow::new(&app);
|
||||||
|
window.set_title("Hammond");
|
||||||
|
|
||||||
|
window.connect_delete_event(clone!(app, settings => move |window, _| {
|
||||||
|
WindowGeometry::from_window(&window).write(&settings);
|
||||||
|
app.quit();
|
||||||
|
Inhibit(false)
|
||||||
|
}));
|
||||||
|
|
||||||
|
// Create a content instance
|
||||||
|
let content =
|
||||||
|
Rc::new(Content::new(sender.clone()).expect(
|
||||||
|
"Content Initialization failed."));
|
||||||
|
|
||||||
|
// Create the headerbar
|
||||||
|
let header = Rc::new(Header::new(&content, &window, &sender));
|
||||||
|
|
||||||
|
let menu = SimpleAction::new("menu", None);
|
||||||
|
menu.connect_activate(clone!(header => move |_, _| {
|
||||||
|
header.open_menu();
|
||||||
|
}));
|
||||||
|
window.add_action(&menu);
|
||||||
|
|
||||||
|
// Add the content main stack to the overlay.
|
||||||
|
let overlay = gtk::Overlay::new();
|
||||||
|
overlay.add(&content.get_stack());
|
||||||
|
|
||||||
|
// Add the overlay to the main window
|
||||||
|
window.add(&overlay);
|
||||||
|
|
||||||
|
WindowGeometry::from_settings(&settings).apply(&window);
|
||||||
|
|
||||||
|
App::setup_timed_callbacks(&sender, &settings);
|
||||||
|
|
||||||
|
window.show_all();
|
||||||
|
window.activate();
|
||||||
|
|
||||||
|
gtk::timeout_add(50, clone!(sender, receiver => move || {
|
||||||
|
// Uses receiver, content, header, sender, overlay
|
||||||
|
let act = receiver.borrow().try_recv();
|
||||||
|
//let act: Result<Action, RecvError> = Ok(Action::RefreshAllViews);
|
||||||
|
match act {
|
||||||
|
Ok(Action::RefreshAllViews) => content.update(),
|
||||||
|
Ok(Action::RefreshShowsView) => content.update_shows_view(),
|
||||||
|
Ok(Action::RefreshWidgetIfSame(id)) =>
|
||||||
|
content.update_widget_if_same(id),
|
||||||
|
Ok(Action::RefreshEpisodesView) => content.update_home(),
|
||||||
|
Ok(Action::RefreshEpisodesViewBGR) =>
|
||||||
|
content.update_home_if_background(),
|
||||||
|
Ok(Action::ReplaceWidget(pd)) => {
|
||||||
|
let shows = content.get_shows();
|
||||||
|
let mut pop = shows.borrow().populated();
|
||||||
|
pop.borrow_mut()
|
||||||
|
.replace_widget(pd.clone())
|
||||||
|
.map_err(|err| error!("Failed to update ShowWidget: {}", err))
|
||||||
|
.map_err(|_|
|
||||||
|
error!("Failed ot update ShowWidget {}", pd.title()))
|
||||||
|
.ok();
|
||||||
|
}
|
||||||
|
Ok(Action::ShowWidgetAnimated) => {
|
||||||
|
let shows = content.get_shows();
|
||||||
|
let mut pop = shows.borrow().populated();
|
||||||
|
pop.borrow_mut().switch_visible(
|
||||||
|
PopulatedState::Widget,
|
||||||
|
gtk::StackTransitionType::SlideLeft,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
Ok(Action::ShowShowsAnimated) => {
|
||||||
|
let shows = content.get_shows();
|
||||||
|
let mut pop = shows.borrow().populated();
|
||||||
|
pop.borrow_mut()
|
||||||
|
.switch_visible(PopulatedState::View,
|
||||||
|
gtk::StackTransitionType::SlideRight);
|
||||||
|
}
|
||||||
|
Ok(Action::HeaderBarShowTile(title)) =>
|
||||||
|
header.switch_to_back(&title),
|
||||||
|
Ok(Action::HeaderBarNormal) => header.switch_to_normal(),
|
||||||
|
Ok(Action::HeaderBarShowUpdateIndicator) =>
|
||||||
|
header.show_update_notification(),
|
||||||
|
Ok(Action::HeaderBarHideUpdateIndicator) =>
|
||||||
|
header.hide_update_notification(),
|
||||||
|
Ok(Action::MarkAllPlayerNotification(pd)) => {
|
||||||
|
let notif = mark_all_notif(pd, &sender);
|
||||||
|
notif.show(&overlay);
|
||||||
|
}
|
||||||
|
Ok(Action::RemoveShow(pd)) => {
|
||||||
|
let notif = remove_show_notif(pd, sender.clone());
|
||||||
|
notif.show(&overlay);
|
||||||
|
}
|
||||||
|
Ok(Action::ErrorNotification(err)) => {
|
||||||
|
error!("An error notification was triggered: {}", err);
|
||||||
|
let callback = || glib::Continue(false);
|
||||||
|
let notif = InAppNotification::new(&err, callback,
|
||||||
|
|| {}, UndoState::Hidden);
|
||||||
|
notif.show(&overlay);
|
||||||
|
}
|
||||||
|
Err(_) => (),
|
||||||
|
}
|
||||||
|
|
||||||
|
Continue(true)
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}));
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let (sender, receiver) = channel();
|
|
||||||
|
|
||||||
// Create a content instance
|
|
||||||
let content =
|
|
||||||
Rc::new(Content::new(sender.clone()).expect("Content Initialization failed."));
|
|
||||||
|
|
||||||
// Create the headerbar
|
|
||||||
let header = Rc::new(Header::new(&content, &window, &sender));
|
|
||||||
|
|
||||||
// Add the content main stack to the overlay.
|
|
||||||
let overlay = gtk::Overlay::new();
|
|
||||||
overlay.add(&content.get_stack());
|
|
||||||
|
|
||||||
// Add the overlay to the main window
|
|
||||||
window.add(&overlay);
|
|
||||||
|
|
||||||
App {
|
App {
|
||||||
app_instance: application,
|
app_instance: application,
|
||||||
window,
|
|
||||||
overlay,
|
|
||||||
header,
|
|
||||||
content,
|
|
||||||
receiver,
|
|
||||||
sender,
|
|
||||||
settings,
|
settings,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup_timed_callbacks(&self) {
|
fn setup_timed_callbacks(sender: &Sender<Action>, settings: &Settings) {
|
||||||
self.setup_dark_theme();
|
App::setup_dark_theme(&sender, settings);
|
||||||
self.setup_refresh_on_startup();
|
App::setup_refresh_on_startup(&sender, settings);
|
||||||
self.setup_auto_refresh();
|
App::setup_auto_refresh(&sender, settings);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup_dark_theme(&self) {
|
fn setup_dark_theme(_sender: &Sender<Action>, settings: &Settings) {
|
||||||
let settings = gtk::Settings::get_default().unwrap();
|
let gtk_settings = gtk::Settings::get_default().unwrap();
|
||||||
let enabled = self.settings.get_boolean("dark-theme");
|
let enabled = settings.get_boolean("dark-theme");
|
||||||
|
|
||||||
settings.set_property_gtk_application_prefer_dark_theme(enabled);
|
gtk_settings.set_property_gtk_application_prefer_dark_theme(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup_refresh_on_startup(&self) {
|
fn setup_refresh_on_startup(sender: &Sender<Action>, settings: &Settings) {
|
||||||
// Update the feeds right after the Application is initialized.
|
// Update the feeds right after the Application is initialized.
|
||||||
if self.settings.get_boolean("refresh-on-startup") {
|
let sender = sender.clone();
|
||||||
let sender = self.sender.clone();
|
if settings.get_boolean("refresh-on-startup") {
|
||||||
|
|
||||||
info!("Refresh on startup.");
|
info!("Refresh on startup.");
|
||||||
// The ui loads async, after initialization
|
// The ui loads async, after initialization
|
||||||
// so we need to delay this a bit so it won't block
|
// so we need to delay this a bit so it won't block
|
||||||
@ -131,12 +246,12 @@ impl App {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup_auto_refresh(&self) {
|
fn setup_auto_refresh(sender: &Sender<Action>, settings: &Settings) {
|
||||||
let refresh_interval = settings::get_refresh_interval(&self.settings).num_seconds() as u32;
|
let refresh_interval = settings::get_refresh_interval(&settings).num_seconds() as u32;
|
||||||
let sender = self.sender.clone();
|
|
||||||
|
|
||||||
info!("Auto-refresh every {:?} seconds.", refresh_interval);
|
info!("Auto-refresh every {:?} seconds.", refresh_interval);
|
||||||
|
|
||||||
|
let sender = sender.clone();
|
||||||
gtk::timeout_add_seconds(refresh_interval, move || {
|
gtk::timeout_add_seconds(refresh_interval, move || {
|
||||||
let s: Option<Vec<_>> = None;
|
let s: Option<Vec<_>> = None;
|
||||||
utils::refresh(s, sender.clone());
|
utils::refresh(s, sender.clone());
|
||||||
@ -146,81 +261,6 @@ impl App {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn run(self) {
|
pub fn run(self) {
|
||||||
WindowGeometry::from_settings(&self.settings).apply(&self.window);
|
|
||||||
|
|
||||||
let window = self.window.clone();
|
|
||||||
self.app_instance.connect_startup(move |app| {
|
|
||||||
build_ui(&window, app);
|
|
||||||
});
|
|
||||||
|
|
||||||
self.setup_timed_callbacks();
|
|
||||||
|
|
||||||
let content = self.content;
|
|
||||||
let headerbar = self.header;
|
|
||||||
let sender = self.sender;
|
|
||||||
let overlay = self.overlay;
|
|
||||||
let receiver = self.receiver;
|
|
||||||
gtk::timeout_add(50, move || {
|
|
||||||
match receiver.try_recv() {
|
|
||||||
Ok(Action::RefreshAllViews) => content.update(),
|
|
||||||
Ok(Action::RefreshShowsView) => content.update_shows_view(),
|
|
||||||
Ok(Action::RefreshWidgetIfSame(id)) => content.update_widget_if_same(id),
|
|
||||||
Ok(Action::RefreshEpisodesView) => content.update_home(),
|
|
||||||
Ok(Action::RefreshEpisodesViewBGR) => content.update_home_if_background(),
|
|
||||||
Ok(Action::ReplaceWidget(pd)) => {
|
|
||||||
let shows = content.get_shows();
|
|
||||||
let mut pop = shows.borrow().populated();
|
|
||||||
pop.borrow_mut()
|
|
||||||
.replace_widget(pd.clone())
|
|
||||||
.map_err(|err| error!("Failed to update ShowWidget: {}", err))
|
|
||||||
.map_err(|_| error!("Failed ot update ShowWidget {}", pd.title()))
|
|
||||||
.ok();
|
|
||||||
}
|
|
||||||
Ok(Action::ShowWidgetAnimated) => {
|
|
||||||
let shows = content.get_shows();
|
|
||||||
let mut pop = shows.borrow().populated();
|
|
||||||
pop.borrow_mut().switch_visible(
|
|
||||||
PopulatedState::Widget,
|
|
||||||
gtk::StackTransitionType::SlideLeft,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Ok(Action::ShowShowsAnimated) => {
|
|
||||||
let shows = content.get_shows();
|
|
||||||
let mut pop = shows.borrow().populated();
|
|
||||||
pop.borrow_mut()
|
|
||||||
.switch_visible(PopulatedState::View, gtk::StackTransitionType::SlideRight);
|
|
||||||
}
|
|
||||||
Ok(Action::HeaderBarShowTile(title)) => headerbar.switch_to_back(&title),
|
|
||||||
Ok(Action::HeaderBarNormal) => headerbar.switch_to_normal(),
|
|
||||||
Ok(Action::HeaderBarShowUpdateIndicator) => headerbar.show_update_notification(),
|
|
||||||
Ok(Action::HeaderBarHideUpdateIndicator) => headerbar.hide_update_notification(),
|
|
||||||
Ok(Action::MarkAllPlayerNotification(pd)) => {
|
|
||||||
let notif = mark_all_notif(pd, &sender);
|
|
||||||
notif.show(&overlay);
|
|
||||||
}
|
|
||||||
Ok(Action::RemoveShow(pd)) => {
|
|
||||||
let notif = remove_show_notif(pd, sender.clone());
|
|
||||||
notif.show(&overlay);
|
|
||||||
}
|
|
||||||
Ok(Action::ErrorNotification(err)) => {
|
|
||||||
error!("An error notification was triggered: {}", err);
|
|
||||||
let callback = || glib::Continue(false);
|
|
||||||
let notif = InAppNotification::new(&err, callback, || {}, UndoState::Hidden);
|
|
||||||
notif.show(&overlay);
|
|
||||||
}
|
|
||||||
Err(_) => (),
|
|
||||||
}
|
|
||||||
|
|
||||||
Continue(true)
|
|
||||||
});
|
|
||||||
|
|
||||||
ApplicationExtManual::run(&self.app_instance, &[]);
|
ApplicationExtManual::run(&self.app_instance, &[]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_ui(window: >k::Window, app: >k::Application) {
|
|
||||||
window.set_application(app);
|
|
||||||
window.show_all();
|
|
||||||
window.activate();
|
|
||||||
app.connect_activate(move |_| ());
|
|
||||||
}
|
|
||||||
|
|||||||
@ -21,7 +21,7 @@ pub struct InAppNotification {
|
|||||||
|
|
||||||
impl Default for InAppNotification {
|
impl Default for InAppNotification {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/inapp_notif.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/inapp_notif.ui");
|
||||||
|
|
||||||
let revealer: gtk::Revealer = builder.get_object("revealer").unwrap();
|
let revealer: gtk::Revealer = builder.get_object("revealer").unwrap();
|
||||||
let text: gtk::Label = builder.get_object("text").unwrap();
|
let text: gtk::Label = builder.get_object("text").unwrap();
|
||||||
|
|||||||
@ -1,19 +1,19 @@
|
|||||||
|
use gio::MenuModel;
|
||||||
use glib;
|
use glib;
|
||||||
use gtk;
|
use gtk;
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
|
|
||||||
use failure::Error;
|
use failure::Error;
|
||||||
use failure::ResultExt;
|
use failure::ResultExt;
|
||||||
use rayon;
|
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
use hammond_data::{dbqueries, opml, Source};
|
use hammond_data::{dbqueries, Source};
|
||||||
|
|
||||||
use std::sync::mpsc::Sender;
|
use std::sync::mpsc::Sender;
|
||||||
|
|
||||||
use app::Action;
|
use app::Action;
|
||||||
use stacks::Content;
|
use stacks::Content;
|
||||||
use utils::{self, itunes_to_rss, refresh};
|
use utils::{itunes_to_rss, refresh};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
// TODO: split this into smaller
|
// TODO: split this into smaller
|
||||||
@ -23,31 +23,28 @@ pub struct Header {
|
|||||||
switch: gtk::StackSwitcher,
|
switch: gtk::StackSwitcher,
|
||||||
back: gtk::Button,
|
back: gtk::Button,
|
||||||
show_title: gtk::Label,
|
show_title: gtk::Label,
|
||||||
about: gtk::ModelButton,
|
|
||||||
import: gtk::ModelButton,
|
|
||||||
export: gtk::ModelButton,
|
|
||||||
update_button: gtk::ModelButton,
|
|
||||||
update_box: gtk::Box,
|
update_box: gtk::Box,
|
||||||
update_label: gtk::Label,
|
update_label: gtk::Label,
|
||||||
update_spinner: gtk::Spinner,
|
update_spinner: gtk::Spinner,
|
||||||
|
menu_button: gtk::MenuButton,
|
||||||
|
app_menu: MenuModel,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Header {
|
impl Default for Header {
|
||||||
fn default() -> Header {
|
fn default() -> Header {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/headerbar.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/headerbar.ui");
|
||||||
|
|
||||||
let header = builder.get_object("headerbar").unwrap();
|
let header = builder.get_object("headerbar").unwrap();
|
||||||
let add_toggle = builder.get_object("add_toggle").unwrap();
|
let add_toggle = builder.get_object("add_toggle").unwrap();
|
||||||
let switch = builder.get_object("switch").unwrap();
|
let switch = builder.get_object("switch").unwrap();
|
||||||
let back = builder.get_object("back").unwrap();
|
let back = builder.get_object("back").unwrap();
|
||||||
let show_title = builder.get_object("show_title").unwrap();
|
let show_title = builder.get_object("show_title").unwrap();
|
||||||
let import = builder.get_object("import").unwrap();
|
|
||||||
let export = builder.get_object("export").unwrap();
|
|
||||||
let update_button = builder.get_object("update_button").unwrap();
|
|
||||||
let update_box = builder.get_object("update_notification").unwrap();
|
let update_box = builder.get_object("update_notification").unwrap();
|
||||||
let update_label = builder.get_object("update_label").unwrap();
|
let update_label = builder.get_object("update_label").unwrap();
|
||||||
let update_spinner = builder.get_object("update_spinner").unwrap();
|
let update_spinner = builder.get_object("update_spinner").unwrap();
|
||||||
let about = builder.get_object("about").unwrap();
|
let menu_button = builder.get_object("menu_button").unwrap();
|
||||||
|
let menus = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/menus.ui");
|
||||||
|
let app_menu = menus.get_object("menu").unwrap();
|
||||||
|
|
||||||
Header {
|
Header {
|
||||||
container: header,
|
container: header,
|
||||||
@ -55,27 +52,34 @@ impl Default for Header {
|
|||||||
switch,
|
switch,
|
||||||
back,
|
back,
|
||||||
show_title,
|
show_title,
|
||||||
about,
|
|
||||||
import,
|
|
||||||
export,
|
|
||||||
update_button,
|
|
||||||
update_box,
|
update_box,
|
||||||
update_label,
|
update_label,
|
||||||
update_spinner,
|
update_spinner,
|
||||||
|
menu_button,
|
||||||
|
app_menu,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Refactor components into smaller state machines
|
// TODO: Refactor components into smaller state machines
|
||||||
impl Header {
|
impl Header {
|
||||||
pub fn new(content: &Content, window: >k::Window, sender: &Sender<Action>) -> Header {
|
pub fn new(
|
||||||
|
content: &Content,
|
||||||
|
window: >k::ApplicationWindow,
|
||||||
|
sender: &Sender<Action>,
|
||||||
|
) -> Header {
|
||||||
let h = Header::default();
|
let h = Header::default();
|
||||||
h.init(content, window, &sender);
|
h.init(content, window, &sender);
|
||||||
h
|
h
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn init(&self, content: &Content, window: >k::Window, sender: &Sender<Action>) {
|
pub fn init(
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/headerbar.ui");
|
&self,
|
||||||
|
content: &Content,
|
||||||
|
window: >k::ApplicationWindow,
|
||||||
|
sender: &Sender<Action>,
|
||||||
|
) {
|
||||||
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/headerbar.ui");
|
||||||
|
|
||||||
let add_popover: gtk::Popover = builder.get_object("add_popover").unwrap();
|
let add_popover: gtk::Popover = builder.get_object("add_popover").unwrap();
|
||||||
let new_url: gtk::Entry = builder.get_object("new_url").unwrap();
|
let new_url: gtk::Entry = builder.get_object("new_url").unwrap();
|
||||||
@ -98,22 +102,6 @@ impl Header {
|
|||||||
|
|
||||||
self.add_toggle.set_popover(&add_popover);
|
self.add_toggle.set_popover(&add_popover);
|
||||||
|
|
||||||
self.update_button
|
|
||||||
.connect_clicked(clone!(sender => move |_| {
|
|
||||||
gtk::idle_add(clone!(sender => move || {
|
|
||||||
let s: Option<Vec<_>> = None;
|
|
||||||
refresh(s, sender.clone());
|
|
||||||
glib::Continue(false)
|
|
||||||
}));
|
|
||||||
}));
|
|
||||||
|
|
||||||
self.about
|
|
||||||
.connect_clicked(clone!(window => move |_| about_dialog(&window)));
|
|
||||||
|
|
||||||
self.import.connect_clicked(
|
|
||||||
clone!(window, sender => move |_| on_import_clicked(&window, &sender)),
|
|
||||||
);
|
|
||||||
|
|
||||||
// Add the Headerbar to the window.
|
// Add the Headerbar to the window.
|
||||||
window.set_titlebar(&self.container);
|
window.set_titlebar(&self.container);
|
||||||
|
|
||||||
@ -131,6 +119,8 @@ impl Header {
|
|||||||
.ok();
|
.ok();
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
self.menu_button.set_menu_model(Some(&self.app_menu));
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn switch_to_back(&self, title: &str) {
|
pub fn switch_to_back(&self, title: &str) {
|
||||||
@ -165,6 +155,10 @@ impl Header {
|
|||||||
self.update_spinner.hide();
|
self.update_spinner.hide();
|
||||||
self.update_label.hide();
|
self.update_label.hide();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn open_menu(&self) {
|
||||||
|
self.menu_button.clicked();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: THIS ALSO SUCKS!
|
// FIXME: THIS ALSO SUCKS!
|
||||||
@ -228,99 +222,3 @@ fn on_url_change(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn on_import_clicked(window: >k::Window, sender: &Sender<Action>) {
|
|
||||||
use glib::translate::ToGlib;
|
|
||||||
use gtk::{FileChooserAction, FileChooserDialog, FileFilter, ResponseType};
|
|
||||||
|
|
||||||
// let dialog = FileChooserDialog::new(title, Some(&window), FileChooserAction::Open);
|
|
||||||
// TODO: It might be better to use a FileChooserNative widget.
|
|
||||||
// Create the FileChooser Dialog
|
|
||||||
let dialog = FileChooserDialog::with_buttons(
|
|
||||||
Some("Select the file from which to you want to Import Shows."),
|
|
||||||
Some(window),
|
|
||||||
FileChooserAction::Open,
|
|
||||||
&[
|
|
||||||
("_Cancel", ResponseType::Cancel),
|
|
||||||
("_Open", ResponseType::Accept),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
|
|
||||||
// Do not show hidden(.thing) files
|
|
||||||
dialog.set_show_hidden(false);
|
|
||||||
|
|
||||||
// Set a filter to show only xml files
|
|
||||||
let filter = FileFilter::new();
|
|
||||||
FileFilterExt::set_name(&filter, Some("OPML file"));
|
|
||||||
filter.add_mime_type("application/xml");
|
|
||||||
filter.add_mime_type("text/xml");
|
|
||||||
dialog.add_filter(&filter);
|
|
||||||
|
|
||||||
dialog.connect_response(clone!(sender => move |dialog, resp| {
|
|
||||||
debug!("Dialong Response {}", resp);
|
|
||||||
if resp == ResponseType::Accept.to_glib() {
|
|
||||||
// TODO: Show an in-app notifictaion if the file can not be accessed
|
|
||||||
if let Some(filename) = dialog.get_filename() {
|
|
||||||
debug!("File selected: {:?}", filename);
|
|
||||||
|
|
||||||
rayon::spawn(clone!(sender => move || {
|
|
||||||
// Parse the file and import the feeds
|
|
||||||
if let Ok(sources) = opml::import_from_file(filename) {
|
|
||||||
// Refresh the succesfully parsed feeds to index them
|
|
||||||
utils::refresh(Some(sources), sender)
|
|
||||||
} else {
|
|
||||||
let text = String::from("Failed to parse the Imported file");
|
|
||||||
sender.send(Action::ErrorNotification(text))
|
|
||||||
.map_err(|err| error!("Action Sender: {}", err))
|
|
||||||
.ok();
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
} else {
|
|
||||||
let text = String::from("Selected File could not be accessed.");
|
|
||||||
sender.send(Action::ErrorNotification(text))
|
|
||||||
.map_err(|err| error!("Action Sender: {}", err))
|
|
||||||
.ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dialog.destroy();
|
|
||||||
}));
|
|
||||||
|
|
||||||
dialog.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Totally copied it from fractal.
|
|
||||||
// https://gitlab.gnome.org/danigm/fractal/blob/503e311e22b9d7540089d735b92af8e8f93560c5/fractal-gtk/src/app.rs#L1883-1912
|
|
||||||
fn about_dialog(window: >k::Window) {
|
|
||||||
// Feel free to add yourself if you contribured.
|
|
||||||
let authors = &[
|
|
||||||
"Constantin Nickel",
|
|
||||||
"Gabriele Musco",
|
|
||||||
"James Wykeham-Martin",
|
|
||||||
"Jordan Petridis",
|
|
||||||
"Julian Sparber",
|
|
||||||
"Rowan Lewis",
|
|
||||||
];
|
|
||||||
|
|
||||||
let dialog = gtk::AboutDialog::new();
|
|
||||||
// Waiting for a logo.
|
|
||||||
// dialog.set_logo_icon_name("org.gnome.Hammond");
|
|
||||||
dialog.set_logo_icon_name("multimedia-player");
|
|
||||||
dialog.set_comments("Podcast Client for the GNOME Desktop.");
|
|
||||||
dialog.set_copyright("© 2017, 2018 Jordan Petridis");
|
|
||||||
dialog.set_license_type(gtk::License::Gpl30);
|
|
||||||
dialog.set_modal(true);
|
|
||||||
// TODO: make it show it fetches the commit hash from which it was built
|
|
||||||
// and the version number is kept in sync automaticly
|
|
||||||
dialog.set_version("0.3.4");
|
|
||||||
dialog.set_program_name("Hammond");
|
|
||||||
// TODO: Need a wiki page first.
|
|
||||||
// dialog.set_website("https://wiki.gnome.org/Design/Apps/Potential/Podcasts");
|
|
||||||
// dialog.set_website_label("Learn more about Hammond");
|
|
||||||
dialog.set_transient_for(window);
|
|
||||||
|
|
||||||
dialog.set_artists(&["Tobias Bernard"]);
|
|
||||||
dialog.set_authors(authors);
|
|
||||||
|
|
||||||
dialog.show();
|
|
||||||
}
|
|
||||||
|
|||||||
@ -86,7 +86,7 @@ fn main() {
|
|||||||
|
|
||||||
// Add custom style
|
// Add custom style
|
||||||
let provider = gtk::CssProvider::new();
|
let provider = gtk::CssProvider::new();
|
||||||
gtk::CssProvider::load_from_resource(&provider, "/org/gnome/hammond/gtk/style.css");
|
gtk::CssProvider::load_from_resource(&provider, "/org/gnome/Hammond/gtk/style.css");
|
||||||
gtk::StyleContext::add_provider_for_screen(
|
gtk::StyleContext::add_provider_for_screen(
|
||||||
&gdk::Screen::get_default().expect("Error initializing gtk css provider."),
|
&gdk::Screen::get_default().expect("Error initializing gtk css provider."),
|
||||||
&provider,
|
&provider,
|
||||||
|
|||||||
@ -15,7 +15,7 @@ pub struct WindowGeometry {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl WindowGeometry {
|
impl WindowGeometry {
|
||||||
pub fn from_window(window: >k::Window) -> WindowGeometry {
|
pub fn from_window(window: >k::ApplicationWindow) -> WindowGeometry {
|
||||||
let position = window.get_position();
|
let position = window.get_position();
|
||||||
let size = window.get_size();
|
let size = window.get_size();
|
||||||
let left = position.0;
|
let left = position.0;
|
||||||
@ -49,7 +49,7 @@ impl WindowGeometry {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn apply(&self, window: >k::Window) {
|
pub fn apply(&self, window: >k::ApplicationWindow) {
|
||||||
if self.width > 0 && self.height > 0 {
|
if self.width > 0 && self.height > 0 {
|
||||||
window.resize(self.width, self.height);
|
window.resize(self.width, self.height);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -17,6 +17,7 @@ use serde_json::Value;
|
|||||||
|
|
||||||
// use hammond_data::feed;
|
// use hammond_data::feed;
|
||||||
use hammond_data::dbqueries;
|
use hammond_data::dbqueries;
|
||||||
|
use hammond_data::opml;
|
||||||
use hammond_data::pipeline;
|
use hammond_data::pipeline;
|
||||||
use hammond_data::utils::checkup;
|
use hammond_data::utils::checkup;
|
||||||
use hammond_data::Source;
|
use hammond_data::Source;
|
||||||
@ -333,6 +334,64 @@ fn lookup_id(id: u32) -> Result<String, Error> {
|
|||||||
.ok_or_else(|| format_err!("Failed to get url from itunes response"))
|
.ok_or_else(|| format_err!("Failed to get url from itunes response"))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn on_import_clicked(window: >k::Window, sender: &Sender<Action>) {
|
||||||
|
use glib::translate::ToGlib;
|
||||||
|
use gtk::{FileChooserAction, FileChooserNative, FileFilter, ResponseType};
|
||||||
|
|
||||||
|
// let dialog = FileChooserDialog::new(title, Some(&window), FileChooserAction::Open);
|
||||||
|
// TODO: It might be better to use a FileChooserNative widget.
|
||||||
|
// Create the FileChooser Dialog
|
||||||
|
let dialog = FileChooserNative::new(
|
||||||
|
Some("Select the file from which to you want to Import Shows."),
|
||||||
|
Some(window),
|
||||||
|
FileChooserAction::Open,
|
||||||
|
Some("_Import"),
|
||||||
|
None,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Do not show hidden(.thing) files
|
||||||
|
dialog.set_show_hidden(false);
|
||||||
|
|
||||||
|
// Set a filter to show only xml files
|
||||||
|
let filter = FileFilter::new();
|
||||||
|
FileFilterExt::set_name(&filter, Some("OPML file"));
|
||||||
|
filter.add_mime_type("application/xml");
|
||||||
|
filter.add_mime_type("text/xml");
|
||||||
|
dialog.add_filter(&filter);
|
||||||
|
|
||||||
|
dialog.connect_response(clone!(sender => move |dialog, resp| {
|
||||||
|
debug!("Dialong Response {}", resp);
|
||||||
|
if resp == ResponseType::Accept.to_glib() {
|
||||||
|
// TODO: Show an in-app notifictaion if the file can not be accessed
|
||||||
|
if let Some(filename) = dialog.get_filename() {
|
||||||
|
debug!("File selected: {:?}", filename);
|
||||||
|
|
||||||
|
rayon::spawn(clone!(sender => move || {
|
||||||
|
// Parse the file and import the feeds
|
||||||
|
if let Ok(sources) = opml::import_from_file(filename) {
|
||||||
|
// Refresh the succesfully parsed feeds to index them
|
||||||
|
refresh(Some(sources), sender)
|
||||||
|
} else {
|
||||||
|
let text = String::from("Failed to parse the Imported file");
|
||||||
|
sender.send(Action::ErrorNotification(text))
|
||||||
|
.map_err(|err| error!("Action Sender: {}", err))
|
||||||
|
.ok();
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
let text = String::from("Selected File could not be accessed.");
|
||||||
|
sender.send(Action::ErrorNotification(text))
|
||||||
|
.map_err(|err| error!("Action Sender: {}", err))
|
||||||
|
.ok();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dialog.destroy();
|
||||||
|
}));
|
||||||
|
|
||||||
|
dialog.run();
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|||||||
39
hammond-gtk/src/widgets/aboutdialog.rs
Normal file
39
hammond-gtk/src/widgets/aboutdialog.rs
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
use gtk;
|
||||||
|
use gtk::prelude::*;
|
||||||
|
|
||||||
|
// Totally copied it from fractal.
|
||||||
|
// https://gitlab.gnome.org/danigm/fractal/blob/503e311e22b9d7540089d735b92af8e8f93560c5/fractal-gtk/src/app.rs#L1883-1912
|
||||||
|
pub fn about_dialog(window: >k::Window) {
|
||||||
|
// Feel free to add yourself if you contribured.
|
||||||
|
let authors = &[
|
||||||
|
"Constantin Nickel",
|
||||||
|
"Gabriele Musco",
|
||||||
|
"James Wykeham-Martin",
|
||||||
|
"Jordan Petridis",
|
||||||
|
"Julian Sparber",
|
||||||
|
"Rowan Lewis",
|
||||||
|
"Zander Brown",
|
||||||
|
];
|
||||||
|
|
||||||
|
let dialog = gtk::AboutDialog::new();
|
||||||
|
// Waiting for a logo.
|
||||||
|
// dialog.set_logo_icon_name("org.gnome.Hammond");
|
||||||
|
dialog.set_logo_icon_name("multimedia-player");
|
||||||
|
dialog.set_comments("Podcast Client for the GNOME Desktop.");
|
||||||
|
dialog.set_copyright("© 2017, 2018 Jordan Petridis");
|
||||||
|
dialog.set_license_type(gtk::License::Gpl30);
|
||||||
|
dialog.set_modal(true);
|
||||||
|
// TODO: make it show it fetches the commit hash from which it was built
|
||||||
|
// and the version number is kept in sync automaticly
|
||||||
|
dialog.set_version("0.3.4");
|
||||||
|
dialog.set_program_name("Hammond");
|
||||||
|
// TODO: Need a wiki page first.
|
||||||
|
// dialog.set_website("https://wiki.gnome.org/Design/Apps/Potential/Podcasts");
|
||||||
|
// dialog.set_website_label("Learn more about Hammond");
|
||||||
|
dialog.set_transient_for(window);
|
||||||
|
|
||||||
|
dialog.set_artists(&["Tobias Bernard"]);
|
||||||
|
dialog.set_authors(authors);
|
||||||
|
|
||||||
|
dialog.show();
|
||||||
|
}
|
||||||
@ -7,7 +7,7 @@ pub struct EmptyView {
|
|||||||
|
|
||||||
impl Default for EmptyView {
|
impl Default for EmptyView {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/empty_view.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/empty_view.ui");
|
||||||
let view: gtk::Box = builder.get_object("empty_view").unwrap();
|
let view: gtk::Box = builder.get_object("empty_view").unwrap();
|
||||||
|
|
||||||
EmptyView { container: view }
|
EmptyView { container: view }
|
||||||
|
|||||||
@ -33,7 +33,7 @@ pub struct EpisodeWidget {
|
|||||||
|
|
||||||
impl Default for EpisodeWidget {
|
impl Default for EpisodeWidget {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/episode_widget.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/episode_widget.ui");
|
||||||
|
|
||||||
let container: gtk::Box = builder.get_object("episode_container").unwrap();
|
let container: gtk::Box = builder.get_object("episode_container").unwrap();
|
||||||
let progress: gtk::ProgressBar = builder.get_object("progress_bar").unwrap();
|
let progress: gtk::ProgressBar = builder.get_object("progress_bar").unwrap();
|
||||||
|
|||||||
@ -49,7 +49,7 @@ pub struct HomeView {
|
|||||||
|
|
||||||
impl Default for HomeView {
|
impl Default for HomeView {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/episodes_view.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/episodes_view.ui");
|
||||||
let container: gtk::Box = builder.get_object("container").unwrap();
|
let container: gtk::Box = builder.get_object("container").unwrap();
|
||||||
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
||||||
let frame_parent: gtk::Box = builder.get_object("frame_parent").unwrap();
|
let frame_parent: gtk::Box = builder.get_object("frame_parent").unwrap();
|
||||||
@ -181,7 +181,7 @@ struct HomeEpisode {
|
|||||||
impl Default for HomeEpisode {
|
impl Default for HomeEpisode {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder =
|
let builder =
|
||||||
gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/episodes_view_widget.ui");
|
gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/episodes_view_widget.ui");
|
||||||
let container: gtk::Box = builder.get_object("container").unwrap();
|
let container: gtk::Box = builder.get_object("container").unwrap();
|
||||||
let image: gtk::Image = builder.get_object("cover").unwrap();
|
let image: gtk::Image = builder.get_object("cover").unwrap();
|
||||||
let ep = EpisodeWidget::default();
|
let ep = EpisodeWidget::default();
|
||||||
@ -198,7 +198,7 @@ impl Default for HomeEpisode {
|
|||||||
impl HomeEpisode {
|
impl HomeEpisode {
|
||||||
fn new(episode: EpisodeWidgetQuery, sender: &Sender<Action>) -> HomeEpisode {
|
fn new(episode: EpisodeWidgetQuery, sender: &Sender<Action>) -> HomeEpisode {
|
||||||
let builder =
|
let builder =
|
||||||
gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/episodes_view_widget.ui");
|
gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/episodes_view_widget.ui");
|
||||||
let container: gtk::Box = builder.get_object("container").unwrap();
|
let container: gtk::Box = builder.get_object("container").unwrap();
|
||||||
let image: gtk::Image = builder.get_object("cover").unwrap();
|
let image: gtk::Image = builder.get_object("cover").unwrap();
|
||||||
let pid = episode.podcast_id();
|
let pid = episode.podcast_id();
|
||||||
|
|||||||
@ -1,3 +1,4 @@
|
|||||||
|
mod aboutdialog;
|
||||||
mod empty;
|
mod empty;
|
||||||
mod episode;
|
mod episode;
|
||||||
mod episode_states;
|
mod episode_states;
|
||||||
@ -5,6 +6,7 @@ mod home_view;
|
|||||||
mod show;
|
mod show;
|
||||||
mod shows_view;
|
mod shows_view;
|
||||||
|
|
||||||
|
pub use self::aboutdialog::about_dialog;
|
||||||
pub use self::empty::EmptyView;
|
pub use self::empty::EmptyView;
|
||||||
pub use self::episode::EpisodeWidget;
|
pub use self::episode::EpisodeWidget;
|
||||||
pub use self::home_view::HomeView;
|
pub use self::home_view::HomeView;
|
||||||
|
|||||||
@ -41,7 +41,7 @@ pub struct ShowWidget {
|
|||||||
|
|
||||||
impl Default for ShowWidget {
|
impl Default for ShowWidget {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/show_widget.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/show_widget.ui");
|
||||||
let container: gtk::Box = builder.get_object("container").unwrap();
|
let container: gtk::Box = builder.get_object("container").unwrap();
|
||||||
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
||||||
let episodes = builder.get_object("episodes").unwrap();
|
let episodes = builder.get_object("episodes").unwrap();
|
||||||
@ -79,7 +79,7 @@ impl ShowWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn init(&mut self, pd: &Arc<Podcast>, sender: &Sender<Action>) {
|
pub fn init(&mut self, pd: &Arc<Podcast>, sender: &Sender<Action>) {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/show_widget.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/show_widget.ui");
|
||||||
|
|
||||||
self.unsub
|
self.unsub
|
||||||
.connect_clicked(clone!(pd, sender => move |bttn| {
|
.connect_clicked(clone!(pd, sender => move |bttn| {
|
||||||
@ -190,7 +190,7 @@ fn populate_listbox(
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
if count == 0 {
|
if count == 0 {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/empty_show.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/empty_show.ui");
|
||||||
let container: gtk::Box = builder.get_object("empty_show").unwrap();
|
let container: gtk::Box = builder.get_object("empty_show").unwrap();
|
||||||
show.episodes.add(&container);
|
show.episodes.add(&container);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
|
|||||||
@ -28,7 +28,7 @@ pub struct ShowsView {
|
|||||||
|
|
||||||
impl Default for ShowsView {
|
impl Default for ShowsView {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/shows_view.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/shows_view.ui");
|
||||||
let container: gtk::Box = builder.get_object("fb_parent").unwrap();
|
let container: gtk::Box = builder.get_object("fb_parent").unwrap();
|
||||||
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
let scrolled_window: gtk::ScrolledWindow = builder.get_object("scrolled_window").unwrap();
|
||||||
let flowbox: gtk::FlowBox = builder.get_object("flowbox").unwrap();
|
let flowbox: gtk::FlowBox = builder.get_object("flowbox").unwrap();
|
||||||
@ -130,7 +130,7 @@ struct ShowsChild {
|
|||||||
|
|
||||||
impl Default for ShowsChild {
|
impl Default for ShowsChild {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let builder = gtk::Builder::new_from_resource("/org/gnome/hammond/gtk/shows_child.ui");
|
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/shows_child.ui");
|
||||||
|
|
||||||
let container: gtk::Box = builder.get_object("fb_child").unwrap();
|
let container: gtk::Box = builder.get_object("fb_child").unwrap();
|
||||||
let cover: gtk::Image = builder.get_object("pd_cover").unwrap();
|
let cover: gtk::Image = builder.get_object("pd_cover").unwrap();
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user