mirror of
https://github.com/italicsjenga/muda.git
synced 2025-01-11 12:21:30 +11:00
simplify and leave some comments for the future
This commit is contained in:
parent
fbef2d8a40
commit
0fbc8cf4f4
|
@ -1,10 +1,8 @@
|
||||||
|
use crate::util::Counter;
|
||||||
|
use gtk::{prelude::*, Orientation};
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
use gtk::{prelude::*, Orientation};
|
|
||||||
|
|
||||||
use crate::util::Counter;
|
|
||||||
|
|
||||||
const COUNTER: Counter = Counter::new();
|
const COUNTER: Counter = Counter::new();
|
||||||
|
|
||||||
enum MenuEntryType {
|
enum MenuEntryType {
|
||||||
|
@ -12,18 +10,27 @@ enum MenuEntryType {
|
||||||
Text,
|
Text,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Generic shared type describing a menu entry. It can be one of [`MenuEntryType`]
|
||||||
struct MenuEntry {
|
struct MenuEntry {
|
||||||
label: String,
|
label: String,
|
||||||
enabled: bool,
|
enabled: bool,
|
||||||
entries: Option<Vec<Arc<Mutex<MenuEntry>>>>,
|
r#type: MenuEntryType,
|
||||||
etype: MenuEntryType,
|
|
||||||
item_id: Option<u64>,
|
item_id: Option<u64>,
|
||||||
menu_gtk_items: Option<Arc<Mutex<Vec<(gtk::MenuItem, gtk::Menu)>>>>,
|
// NOTE(amrbashir): because gtk doesn't allow using the same `gtk::MenuItem`
|
||||||
|
// multiple times, and thus can't be used in multiple windows, each entry
|
||||||
|
// keeps a vector of a `gtk::MenuItem` or a tuple of `gtk::MenuItem` and `gtk::Menu`
|
||||||
|
// and push to it every time `Menu::init_for_gtk_window` is called.
|
||||||
item_gtk_items: Option<Arc<Mutex<Vec<gtk::MenuItem>>>>,
|
item_gtk_items: Option<Arc<Mutex<Vec<gtk::MenuItem>>>>,
|
||||||
|
menu_gtk_items: Option<Arc<Mutex<Vec<(gtk::MenuItem, gtk::Menu)>>>>,
|
||||||
|
entries: Option<Vec<Arc<Mutex<MenuEntry>>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct InnerMenu {
|
struct InnerMenu {
|
||||||
entries: Vec<Arc<Mutex<MenuEntry>>>,
|
entries: Vec<Arc<Mutex<MenuEntry>>>,
|
||||||
|
// NOTE(amrbashir): because gtk doesn't allow using the same `gtk::MenuBar` and `gtk::Box`
|
||||||
|
// multiple times, and thus can't be used in multiple windows, entry
|
||||||
|
// keeps a vector of a tuple of `gtk::MenuBar` and `gtk::Box`
|
||||||
|
// and push to it every time `Menu::init_for_gtk_window` is called.
|
||||||
gtk_items: Vec<(gtk::MenuBar, gtk::Box)>,
|
gtk_items: Vec<(gtk::MenuBar, gtk::Box)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,23 +46,17 @@ impl Menu {
|
||||||
|
|
||||||
pub fn add_submenu(&mut self, label: impl AsRef<str>, enabled: bool) -> Submenu {
|
pub fn add_submenu(&mut self, label: impl AsRef<str>, enabled: bool) -> Submenu {
|
||||||
let label = label.as_ref().to_string();
|
let label = label.as_ref().to_string();
|
||||||
let gtk_items = Arc::new(Mutex::new(Vec::new()));
|
|
||||||
let entry = Arc::new(Mutex::new(MenuEntry {
|
let entry = Arc::new(Mutex::new(MenuEntry {
|
||||||
label: label.clone(),
|
label: label.clone(),
|
||||||
enabled,
|
enabled,
|
||||||
entries: Some(Vec::new()),
|
entries: Some(Vec::new()),
|
||||||
etype: MenuEntryType::Submenu,
|
r#type: MenuEntryType::Submenu,
|
||||||
item_id: None,
|
item_id: None,
|
||||||
menu_gtk_items: Some(gtk_items.clone()),
|
menu_gtk_items: Some(Arc::new(Mutex::new(Vec::new()))),
|
||||||
item_gtk_items: None,
|
item_gtk_items: None,
|
||||||
}));
|
}));
|
||||||
self.0.lock().entries.push(entry.clone());
|
self.0.lock().entries.push(entry.clone());
|
||||||
Submenu {
|
Submenu(entry)
|
||||||
label,
|
|
||||||
enabled,
|
|
||||||
entry,
|
|
||||||
gtk_items,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn init_for_gtk_window<W>(&self, w: &W)
|
pub fn init_for_gtk_window<W>(&self, w: &W)
|
||||||
|
@ -80,7 +81,7 @@ fn add_entries_to_menu<M: IsA<gtk::MenuShell>>(gtk_menu: &M, entries: &Vec<Arc<M
|
||||||
let gtk_item = gtk::MenuItem::with_label(&entry.label);
|
let gtk_item = gtk::MenuItem::with_label(&entry.label);
|
||||||
gtk_menu.append(>k_item);
|
gtk_menu.append(>k_item);
|
||||||
gtk_item.set_sensitive(entry.enabled);
|
gtk_item.set_sensitive(entry.enabled);
|
||||||
if let MenuEntryType::Submenu = entry.etype {
|
if let MenuEntryType::Submenu = entry.r#type {
|
||||||
let gtk_menu = gtk::Menu::new();
|
let gtk_menu = gtk::Menu::new();
|
||||||
gtk_item.set_submenu(Some(>k_menu));
|
gtk_item.set_submenu(Some(>k_menu));
|
||||||
add_entries_to_menu(>k_menu, entry.entries.as_ref().unwrap());
|
add_entries_to_menu(>k_menu, entry.entries.as_ref().unwrap());
|
||||||
|
@ -101,118 +102,77 @@ fn add_entries_to_menu<M: IsA<gtk::MenuShell>>(gtk_menu: &M, entries: &Vec<Arc<M
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Submenu {
|
pub struct Submenu(Arc<Mutex<MenuEntry>>);
|
||||||
label: String,
|
|
||||||
enabled: bool,
|
|
||||||
entry: Arc<Mutex<MenuEntry>>,
|
|
||||||
gtk_items: Arc<Mutex<Vec<(gtk::MenuItem, gtk::Menu)>>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Submenu {
|
impl Submenu {
|
||||||
pub fn set_label(&mut self, label: impl AsRef<str>) {
|
pub fn set_label(&mut self, label: impl AsRef<str>) {
|
||||||
let label = label.as_ref().to_string();
|
let label = label.as_ref().to_string();
|
||||||
for (item, _) in self.gtk_items.lock().iter() {
|
let mut entry = self.0.lock();
|
||||||
|
for (item, _) in entry.menu_gtk_items.as_ref().unwrap().lock().iter() {
|
||||||
item.set_label(&label);
|
item.set_label(&label);
|
||||||
}
|
}
|
||||||
|
entry.label = label;
|
||||||
self.label = label.clone();
|
|
||||||
self.entry.lock().label = label;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_enabled(&mut self, enabled: bool) {
|
pub fn set_enabled(&mut self, enabled: bool) {
|
||||||
for (item, _) in self.gtk_items.lock().iter() {
|
let mut entry = self.0.lock();
|
||||||
|
entry.enabled = true;
|
||||||
|
for (item, _) in entry.menu_gtk_items.as_ref().unwrap().lock().iter() {
|
||||||
item.set_sensitive(enabled);
|
item.set_sensitive(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.enabled = enabled;
|
|
||||||
self.entry.lock().enabled = enabled;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_submenu(&mut self, label: impl AsRef<str>, enabled: bool) -> Submenu {
|
pub fn add_submenu(&mut self, label: impl AsRef<str>, enabled: bool) -> Submenu {
|
||||||
let label = label.as_ref().to_string();
|
|
||||||
let gtk_items = Arc::new(Mutex::new(Vec::new()));
|
|
||||||
let entry = Arc::new(Mutex::new(MenuEntry {
|
let entry = Arc::new(Mutex::new(MenuEntry {
|
||||||
label: label.clone(),
|
label: label.as_ref().to_string(),
|
||||||
enabled,
|
enabled,
|
||||||
entries: Some(Vec::new()),
|
entries: Some(Vec::new()),
|
||||||
etype: MenuEntryType::Submenu,
|
r#type: MenuEntryType::Submenu,
|
||||||
item_id: None,
|
item_id: None,
|
||||||
menu_gtk_items: Some(gtk_items.clone()),
|
menu_gtk_items: Some(Arc::new(Mutex::new(Vec::new()))),
|
||||||
item_gtk_items: None,
|
item_gtk_items: None,
|
||||||
}));
|
}));
|
||||||
self.entry
|
self.0.lock().entries.as_mut().unwrap().push(entry.clone());
|
||||||
.lock()
|
Submenu(entry)
|
||||||
.entries
|
|
||||||
.as_mut()
|
|
||||||
.unwrap()
|
|
||||||
.push(entry.clone());
|
|
||||||
Submenu {
|
|
||||||
label,
|
|
||||||
enabled,
|
|
||||||
entry,
|
|
||||||
gtk_items,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_text_item(&mut self, label: impl AsRef<str>, enabled: bool) -> TextMenuItem {
|
pub fn add_text_item(&mut self, label: impl AsRef<str>, enabled: bool) -> TextMenuItem {
|
||||||
let id = COUNTER.next();
|
|
||||||
let label = label.as_ref().to_string();
|
|
||||||
let gtk_items = Arc::new(Mutex::new(Vec::new()));
|
|
||||||
let entry = Arc::new(Mutex::new(MenuEntry {
|
let entry = Arc::new(Mutex::new(MenuEntry {
|
||||||
label: label.clone(),
|
label: label.as_ref().to_string(),
|
||||||
enabled,
|
enabled,
|
||||||
entries: None,
|
entries: None,
|
||||||
etype: MenuEntryType::Text,
|
r#type: MenuEntryType::Text,
|
||||||
item_id: Some(id),
|
item_id: Some(COUNTER.next()),
|
||||||
menu_gtk_items: None,
|
menu_gtk_items: None,
|
||||||
item_gtk_items: Some(gtk_items.clone()),
|
item_gtk_items: Some(Arc::new(Mutex::new(Vec::new()))),
|
||||||
}));
|
}));
|
||||||
self.entry
|
self.0.lock().entries.as_mut().unwrap().push(entry.clone());
|
||||||
.lock()
|
TextMenuItem(entry)
|
||||||
.entries
|
|
||||||
.as_mut()
|
|
||||||
.unwrap()
|
|
||||||
.push(entry.clone());
|
|
||||||
TextMenuItem {
|
|
||||||
label,
|
|
||||||
enabled,
|
|
||||||
entry,
|
|
||||||
gtk_items,
|
|
||||||
id,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct TextMenuItem {
|
pub struct TextMenuItem(Arc<Mutex<MenuEntry>>);
|
||||||
label: String,
|
|
||||||
enabled: bool,
|
|
||||||
entry: Arc<Mutex<MenuEntry>>,
|
|
||||||
gtk_items: Arc<Mutex<Vec<gtk::MenuItem>>>,
|
|
||||||
id: u64,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TextMenuItem {
|
impl TextMenuItem {
|
||||||
pub fn set_label(&mut self, label: impl AsRef<str>) {
|
pub fn set_label(&mut self, label: impl AsRef<str>) {
|
||||||
let label = label.as_ref().to_string();
|
let label = label.as_ref().to_string();
|
||||||
for item in self.gtk_items.lock().iter() {
|
let mut entry = self.0.lock();
|
||||||
|
for item in entry.item_gtk_items.as_ref().unwrap().lock().iter() {
|
||||||
item.set_label(&label);
|
item.set_label(&label);
|
||||||
}
|
}
|
||||||
|
entry.label = label;
|
||||||
self.label = label.clone();
|
|
||||||
self.entry.lock().label = label;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_enabled(&mut self, enabled: bool) {
|
pub fn set_enabled(&mut self, enabled: bool) {
|
||||||
for item in self.gtk_items.lock().iter() {
|
let mut entry = self.0.lock();
|
||||||
|
for item in entry.item_gtk_items.as_ref().unwrap().lock().iter() {
|
||||||
item.set_sensitive(enabled);
|
item.set_sensitive(enabled);
|
||||||
}
|
}
|
||||||
|
entry.enabled = enabled;
|
||||||
self.enabled = enabled;
|
|
||||||
self.entry.lock().enabled = enabled;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn id(&self) -> u64 {
|
pub fn id(&self) -> u64 {
|
||||||
self.id
|
self.0.lock().item_id.unwrap()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue