Headerbar: Factor out the AddPopover widget.

This commit is contained in:
Jordan Petridis 2018-06-27 20:15:42 +03:00
parent b0fc9ef05e
commit 91aae6a9f5
No known key found for this signature in database
GPG Key ID: CEABAD9F5683B9A6
2 changed files with 122 additions and 100 deletions

View File

@ -151,9 +151,10 @@ Tobias Bernard
<object class="GtkMenuButton" id="add_toggle"> <object class="GtkMenuButton" id="add_toggle">
<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">True</property>
<property name="tooltip_text" translatable="yes">Add a new feed</property> <property name="tooltip_text" translatable="yes">Add a new feed</property>
<property name="valign">center</property> <property name="valign">center</property>
<property name="popover">add_popover</property>
<child> <child>
<object class="GtkImage" id="add-button-image2"> <object class="GtkImage" id="add-button-image2">
<property name="visible">True</property> <property name="visible">True</property>
@ -262,7 +263,6 @@ Tobias Bernard
<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="use_popover">True</property>
<child> <child>
<object class="GtkImage"> <object class="GtkImage">
<property name="visible">True</property> <property name="visible">True</property>
@ -281,4 +281,4 @@ Tobias Bernard
</packing> </packing>
</child> </child>
</object> </object>
</interface> </interface>

View File

@ -1,13 +1,12 @@
use gio::MenuModel; use gio::MenuModel;
use glib;
use gtk; use gtk;
use gtk::prelude::*; use gtk::prelude::*;
use crossbeam_channel::Sender;
use failure::Error; use failure::Error;
use failure::ResultExt; use rayon;
use url::Url; use url::Url;
use crossbeam_channel::Sender;
use hammond_data::{dbqueries, Source}; use hammond_data::{dbqueries, Source};
use app::Action; use app::Action;
@ -20,13 +19,13 @@ use std::rc::Rc;
// TODO: split this into smaller // TODO: split this into smaller
pub struct Header { pub struct Header {
container: gtk::HeaderBar, container: gtk::HeaderBar,
add_toggle: gtk::MenuButton,
switch: gtk::StackSwitcher, switch: gtk::StackSwitcher,
back: gtk::Button, back: gtk::Button,
show_title: gtk::Label, show_title: gtk::Label,
menu_button: gtk::MenuButton, menu_button: gtk::MenuButton,
app_menu: MenuModel, app_menu: MenuModel,
updater: UpdateIndicator, updater: UpdateIndicator,
add: AddPopover,
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -52,21 +51,100 @@ impl UpdateIndicator {
} }
} }
#[derive(Debug, Clone)]
struct AddPopover {
container: gtk::Popover,
result: gtk::Label,
entry: gtk::Entry,
add: gtk::Button,
toggle: gtk::MenuButton,
}
impl AddPopover {
// FIXME: THIS ALSO SUCKS!
fn on_add_clicked(&self, sender: &Sender<Action>) -> Result<(), Error> {
let url = self
.entry
.get_text()
.ok_or_else(|| format_err!("GtkEntry blew up somehow."))?;
debug!("Url: {}", url);
let url = if url.contains("itunes.com") || url.contains("apple.com") {
info!("Detected itunes url.");
let foo = itunes_to_rss(&url)?;
info!("Resolved to {}", foo);
foo
} else {
url.to_owned()
};
self.entry.set_text("");
rayon::spawn(clone!(sender => move || {
if let Ok(source) = Source::from_url(&url) {
refresh(Some(vec![source]), sender.clone());
} else {
error!("Failed to convert, url: {}, to a source entry", url);
}
}));
self.container.hide();
Ok(())
}
// FIXME: THIS SUCKS! REFACTOR ME.
fn on_entry_changed(&self) -> Result<(), Error> {
let uri = self
.entry
.get_text()
.ok_or_else(|| format_err!("GtkEntry blew up somehow."))?;
debug!("Url: {}", uri);
let url = Url::parse(&uri);
// TODO: refactor to avoid duplication
match url {
Ok(u) => {
if !dbqueries::source_exists(u.as_str())? {
self.add.set_sensitive(true);
self.result.hide();
self.result.set_label("");
} else {
self.add.set_sensitive(false);
self.result.set_label("Show already exists.");
self.result.show();
}
Ok(())
}
Err(err) => {
self.add.set_sensitive(false);
if !uri.is_empty() {
self.result.set_label("Invalid url.");
self.result.show();
error!("Error: {}", err);
} else {
self.result.hide();
}
Ok(())
}
}
}
}
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 menus = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/menus.ui"); let menus = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/menus.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 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 menu_button = builder.get_object("menu_button").unwrap();
let app_menu = menus.get_object("menu").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 menu_button = builder.get_object("menu_button").unwrap();
let app_menu = menus.get_object("menu").unwrap();
let updater = UpdateIndicator { let updater = UpdateIndicator {
container: update_box, container: update_box,
@ -74,15 +152,28 @@ impl Default for Header {
spinner: update_spinner, spinner: update_spinner,
}; };
let add_toggle = builder.get_object("add_toggle").unwrap();
let add_popover = builder.get_object("add_popover").unwrap();
let new_url = builder.get_object("new_url").unwrap();
let add_button = builder.get_object("add_button").unwrap();
let result = builder.get_object("result_label").unwrap();
let add = AddPopover {
container: add_popover,
entry: new_url,
toggle: add_toggle,
add: add_button,
result,
};
Header { Header {
container: header, container: header,
add_toggle,
switch, switch,
back, back,
show_title, show_title,
menu_button, menu_button,
app_menu, app_menu,
updater, updater,
add,
} }
} }
} }
@ -95,46 +186,39 @@ impl Header {
sender: &Sender<Action>, sender: &Sender<Action>,
) -> Rc<Self> { ) -> Rc<Self> {
let h = Rc::new(Header::default()); let h = Rc::new(Header::default());
h.init(content, window, &sender); Self::init(&h, content, window, &sender);
h h
} }
pub fn init( pub fn init(
&self, s: &Rc<Self>,
content: &Content, content: &Content,
window: &gtk::ApplicationWindow, window: &gtk::ApplicationWindow,
sender: &Sender<Action>, sender: &Sender<Action>,
) { ) {
let builder = gtk::Builder::new_from_resource("/org/gnome/Hammond/gtk/headerbar.ui"); let weak = Rc::downgrade(s);
let add_popover: gtk::Popover = builder.get_object("add_popover").unwrap(); s.switch.set_stack(&content.get_stack());
let new_url: gtk::Entry = builder.get_object("new_url").unwrap();
let add_button: gtk::Button = builder.get_object("add_button").unwrap();
let result_label: gtk::Label = builder.get_object("result_label").unwrap();
self.switch.set_stack(&content.get_stack());
new_url.connect_changed(clone!(add_button => move |url| { s.add.entry.connect_changed(clone!(weak => move |_| {
on_url_change(url, &result_label, &add_button) weak.upgrade().map(|h| {
.map_err(|err| error!("Error: {}", err)) h.add.on_entry_changed()
.ok(); .map_err(|err| error!("Error: {}", err))
.ok();
});
})); }));
add_button.connect_clicked(clone!(add_popover, new_url, sender => move |_| { s.add.add.connect_clicked(clone!(weak, sender => move |_| {
on_add_bttn_clicked(&new_url, sender.clone()) weak.upgrade().map(|h| h.add.on_add_clicked(&sender));
.map_err(|err| error!("Error: {}", err))
.ok();
add_popover.hide();
})); }));
self.add_toggle.set_popover(&add_popover);
// Add the Headerbar to the window. // Add the Headerbar to the window.
window.set_titlebar(&self.container); window.set_titlebar(&s.container);
let switch = &self.switch; let switch = &s.switch;
let add_toggle = &self.add_toggle; let add_toggle = &s.add.toggle;
let show_title = &self.show_title; let show_title = &s.show_title;
self.back.connect_clicked( s.back.connect_clicked(
clone!(switch, add_toggle, show_title, sender => move |back| { clone!(switch, add_toggle, show_title, sender => move |back| {
switch.show(); switch.show();
add_toggle.show(); add_toggle.show();
@ -144,12 +228,12 @@ impl Header {
}), }),
); );
self.menu_button.set_menu_model(Some(&self.app_menu)); s.menu_button.set_menu_model(Some(&s.app_menu));
} }
pub fn switch_to_back(&self, title: &str) { pub fn switch_to_back(&self, title: &str) {
self.switch.hide(); self.switch.hide();
self.add_toggle.hide(); self.add.toggle.hide();
self.back.show(); self.back.show();
self.set_show_title(title); self.set_show_title(title);
self.show_title.show(); self.show_title.show();
@ -157,7 +241,7 @@ impl Header {
pub fn switch_to_normal(&self) { pub fn switch_to_normal(&self) {
self.switch.show(); self.switch.show();
self.add_toggle.show(); self.add.toggle.show();
self.back.hide(); self.back.hide();
self.show_title.hide(); self.show_title.hide();
} }
@ -178,65 +262,3 @@ impl Header {
self.menu_button.clicked(); self.menu_button.clicked();
} }
} }
// FIXME: THIS ALSO SUCKS!
fn on_add_bttn_clicked(entry: &gtk::Entry, sender: Sender<Action>) -> Result<(), Error> {
let url = entry.get_text().unwrap_or_default();
let url = if url.contains("itunes.com") || url.contains("apple.com") {
info!("Detected itunes url.");
let foo = itunes_to_rss(&url)?;
info!("Resolved to {}", foo);
foo
} else {
url.to_owned()
};
let source = Source::from_url(&url).context("Failed to convert url to a Source entry.")?;
entry.set_text("");
gtk::idle_add(move || {
refresh(Some(vec![source.clone()]), sender.clone());
glib::Continue(false)
});
Ok(())
}
// FIXME: THIS SUCKS!
fn on_url_change(
entry: &gtk::Entry,
result: &gtk::Label,
add_button: &gtk::Button,
) -> Result<(), Error> {
let uri = entry
.get_text()
.ok_or_else(|| format_err!("GtkEntry blew up somehow."))?;
debug!("Url: {}", uri);
let url = Url::parse(&uri);
// TODO: refactor to avoid duplication
match url {
Ok(u) => {
if !dbqueries::source_exists(u.as_str())? {
add_button.set_sensitive(true);
result.hide();
result.set_label("");
} else {
add_button.set_sensitive(false);
result.set_label("Show already exists.");
result.show();
}
Ok(())
}
Err(err) => {
add_button.set_sensitive(false);
if !uri.is_empty() {
result.set_label("Invalid url.");
result.show();
error!("Error: {}", err);
} else {
result.hide();
}
Ok(())
}
}
}