mirror of
https://github.com/italicsjenga/muda.git
synced 2025-01-11 04:11:32 +11:00
feat: add MenuId::new
(#94)
This commit is contained in:
parent
14683ec1ba
commit
32be0c5884
5
.changes/clone.md
Normal file
5
.changes/clone.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
"muda": "patch"
|
||||||
|
---
|
||||||
|
|
||||||
|
On Windows, reduce some unneccassry string cloning.
|
5
.changes/menu-id-new.md
Normal file
5
.changes/menu-id-new.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
"muda": "patch"
|
||||||
|
---
|
||||||
|
|
||||||
|
Add `MenuId::new` convenience method.
|
|
@ -20,7 +20,7 @@ mod test {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn it_returns_same_id() {
|
fn it_returns_same_id() {
|
||||||
let id = MenuId("1".into());
|
let id = MenuId::new("1");
|
||||||
assert_eq!(id, MenuItem::with_id(id.clone(), "", true, None).id());
|
assert_eq!(id, MenuItem::with_id(id.clone(), "", true, None).id());
|
||||||
assert_eq!(id, Submenu::with_id(id.clone(), "", true).id());
|
assert_eq!(id, Submenu::with_id(id.clone(), "", true).id());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
|
52
src/lib.rs
52
src/lib.rs
|
@ -126,8 +126,6 @@
|
||||||
//! }
|
//! }
|
||||||
//! ```
|
//! ```
|
||||||
|
|
||||||
use std::{convert::Infallible, str::FromStr};
|
|
||||||
|
|
||||||
use crossbeam_channel::{unbounded, Receiver, Sender};
|
use crossbeam_channel::{unbounded, Receiver, Sender};
|
||||||
use once_cell::sync::{Lazy, OnceCell};
|
use once_cell::sync::{Lazy, OnceCell};
|
||||||
|
|
||||||
|
@ -139,6 +137,7 @@ mod error;
|
||||||
mod icon;
|
mod icon;
|
||||||
mod items;
|
mod items;
|
||||||
mod menu;
|
mod menu;
|
||||||
|
mod menu_id;
|
||||||
mod platform_impl;
|
mod platform_impl;
|
||||||
mod util;
|
mod util;
|
||||||
|
|
||||||
|
@ -153,54 +152,7 @@ pub use error::*;
|
||||||
pub use icon::{BadIcon, Icon, NativeIcon};
|
pub use icon::{BadIcon, Icon, NativeIcon};
|
||||||
pub use items::*;
|
pub use items::*;
|
||||||
pub use menu::Menu;
|
pub use menu::Menu;
|
||||||
|
pub use menu_id::MenuId;
|
||||||
/// An unique id that is associated with a menu item.
|
|
||||||
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Default, Hash)]
|
|
||||||
pub struct MenuId(pub String);
|
|
||||||
|
|
||||||
impl AsRef<str> for MenuId {
|
|
||||||
fn as_ref(&self) -> &str {
|
|
||||||
self.0.as_ref()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<String> for MenuId {
|
|
||||||
fn from(value: String) -> Self {
|
|
||||||
Self(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<&str> for MenuId {
|
|
||||||
fn from(value: &str) -> Self {
|
|
||||||
Self(value.to_string())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PartialEq<&str> for MenuId {
|
|
||||||
fn eq(&self, other: &&str) -> bool {
|
|
||||||
other == &self.0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PartialEq<String> for MenuId {
|
|
||||||
fn eq(&self, other: &String) -> bool {
|
|
||||||
other == &self.0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PartialEq<&MenuId> for MenuId {
|
|
||||||
fn eq(&self, other: &&MenuId) -> bool {
|
|
||||||
other.0 == self.0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FromStr for MenuId {
|
|
||||||
type Err = Infallible;
|
|
||||||
|
|
||||||
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
|
||||||
Ok(Self(s.to_string()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// An enumeration of all available menu types, useful to match against
|
/// An enumeration of all available menu types, useful to match against
|
||||||
/// the items returned from [`Menu::items`] or [`Submenu::items`]
|
/// the items returned from [`Menu::items`] or [`Submenu::items`]
|
||||||
|
|
62
src/menu_id.rs
Normal file
62
src/menu_id.rs
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
use std::{convert::Infallible, str::FromStr};
|
||||||
|
|
||||||
|
/// An unique id that is associated with a menu item.
|
||||||
|
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Default, Hash)]
|
||||||
|
pub struct MenuId(pub String);
|
||||||
|
|
||||||
|
impl MenuId {
|
||||||
|
/// Create a new menu id.
|
||||||
|
pub fn new<S: AsRef<str>>(id: S) -> Self {
|
||||||
|
Self(id.as_ref().to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsRef<str> for MenuId {
|
||||||
|
fn as_ref(&self) -> &str {
|
||||||
|
self.0.as_ref()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<String> for MenuId {
|
||||||
|
fn from(value: String) -> Self {
|
||||||
|
Self::new(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&str> for MenuId {
|
||||||
|
fn from(value: &str) -> Self {
|
||||||
|
Self::new(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromStr for MenuId {
|
||||||
|
type Err = Infallible;
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
|
||||||
|
Ok(Self::new(s))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq<&str> for MenuId {
|
||||||
|
fn eq(&self, other: &&str) -> bool {
|
||||||
|
other == &self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq<String> for MenuId {
|
||||||
|
fn eq(&self, other: &String) -> bool {
|
||||||
|
other == &self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq<&String> for MenuId {
|
||||||
|
fn eq(&self, other: &&String) -> bool {
|
||||||
|
other == &&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq<&MenuId> for MenuId {
|
||||||
|
fn eq(&self, other: &&MenuId) -> bool {
|
||||||
|
other.0 == self.0
|
||||||
|
}
|
||||||
|
}
|
|
@ -723,7 +723,7 @@ impl MenuChild {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.ns_menu_items
|
self.ns_menu_items
|
||||||
.entry(menu_id.clone())
|
.entry(menu_id)
|
||||||
.or_insert_with(Vec::new)
|
.or_insert_with(Vec::new)
|
||||||
.push(ns_menu_item);
|
.push(ns_menu_item);
|
||||||
|
|
||||||
|
@ -762,7 +762,7 @@ impl MenuChild {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.ns_menu_items
|
self.ns_menu_items
|
||||||
.entry(menu_id.clone())
|
.entry(menu_id)
|
||||||
.or_insert_with(Vec::new)
|
.or_insert_with(Vec::new)
|
||||||
.push(ns_menu_item);
|
.push(ns_menu_item);
|
||||||
|
|
||||||
|
@ -792,7 +792,7 @@ impl MenuChild {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.ns_menu_items
|
self.ns_menu_items
|
||||||
.entry(menu_id.clone())
|
.entry(menu_id)
|
||||||
.or_insert_with(Vec::new)
|
.or_insert_with(Vec::new)
|
||||||
.push(ns_menu_item);
|
.push(ns_menu_item);
|
||||||
|
|
||||||
|
@ -825,7 +825,7 @@ impl MenuChild {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.ns_menu_items
|
self.ns_menu_items
|
||||||
.entry(menu_id.clone())
|
.entry(menu_id)
|
||||||
.or_insert_with(Vec::new)
|
.or_insert_with(Vec::new)
|
||||||
.push(ns_menu_item);
|
.push(ns_menu_item);
|
||||||
|
|
||||||
|
|
|
@ -43,23 +43,21 @@ use windows_sys::Win32::{
|
||||||
|
|
||||||
static COUNTER: Counter = Counter::new_with_start(1000);
|
static COUNTER: Counter = Counter::new_with_start(1000);
|
||||||
|
|
||||||
type AccelWrapper = (HACCEL, HashMap<u32, Accel>);
|
|
||||||
|
|
||||||
macro_rules! inner_menu_child_and_flags {
|
macro_rules! inner_menu_child_and_flags {
|
||||||
($item:ident) => {{
|
($item:ident) => {{
|
||||||
let mut flags = 0;
|
let mut flags = 0;
|
||||||
let child = match $item.kind() {
|
let child = match $item.kind() {
|
||||||
MenuItemKind::Submenu(i) => {
|
MenuItemKind::Submenu(i) => {
|
||||||
flags |= MF_POPUP;
|
flags |= MF_POPUP;
|
||||||
i.inner.clone()
|
i.inner
|
||||||
}
|
}
|
||||||
MenuItemKind::MenuItem(i) => {
|
MenuItemKind::MenuItem(i) => {
|
||||||
flags |= MF_STRING;
|
flags |= MF_STRING;
|
||||||
i.inner.clone()
|
i.inner
|
||||||
}
|
}
|
||||||
|
|
||||||
MenuItemKind::Predefined(i) => {
|
MenuItemKind::Predefined(i) => {
|
||||||
let child = i.inner.clone();
|
let child = i.inner;
|
||||||
let child_ = child.borrow();
|
let child_ = child.borrow();
|
||||||
match child_.predefined_item_type.as_ref().unwrap() {
|
match child_.predefined_item_type.as_ref().unwrap() {
|
||||||
PredefinedMenuItemType::None => return Ok(()),
|
PredefinedMenuItemType::None => return Ok(()),
|
||||||
|
@ -74,7 +72,7 @@ macro_rules! inner_menu_child_and_flags {
|
||||||
child
|
child
|
||||||
}
|
}
|
||||||
MenuItemKind::Check(i) => {
|
MenuItemKind::Check(i) => {
|
||||||
let child = i.inner.clone();
|
let child = i.inner;
|
||||||
flags |= MF_STRING;
|
flags |= MF_STRING;
|
||||||
if child.borrow().checked {
|
if child.borrow().checked {
|
||||||
flags |= MF_CHECKED;
|
flags |= MF_CHECKED;
|
||||||
|
@ -83,7 +81,7 @@ macro_rules! inner_menu_child_and_flags {
|
||||||
}
|
}
|
||||||
MenuItemKind::Icon(i) => {
|
MenuItemKind::Icon(i) => {
|
||||||
flags |= MF_STRING;
|
flags |= MF_STRING;
|
||||||
i.inner.clone()
|
i.inner
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -91,9 +89,12 @@ macro_rules! inner_menu_child_and_flags {
|
||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AccelWrapper = (HACCEL, HashMap<u32, Accel>);
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct Menu {
|
pub(crate) struct Menu {
|
||||||
id: MenuId,
|
id: MenuId,
|
||||||
|
internal_id: u32,
|
||||||
hmenu: HMENU,
|
hmenu: HMENU,
|
||||||
hpopupmenu: HMENU,
|
hpopupmenu: HMENU,
|
||||||
hwnds: Vec<HWND>,
|
hwnds: Vec<HWND>,
|
||||||
|
@ -107,17 +108,17 @@ impl Drop for Menu {
|
||||||
let _ = self.remove_for_hwnd(hwnd);
|
let _ = self.remove_for_hwnd(hwnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn remove_from_children_stores(id: &MenuId, children: &Vec<Rc<RefCell<MenuChild>>>) {
|
fn remove_from_children_stores(internal_id: u32, children: &Vec<Rc<RefCell<MenuChild>>>) {
|
||||||
for child in children {
|
for child in children {
|
||||||
let mut child_ = child.borrow_mut();
|
let mut child_ = child.borrow_mut();
|
||||||
child_.root_menu_haccel_stores.remove(id);
|
child_.root_menu_haccel_stores.remove(&internal_id);
|
||||||
if child_.item_type == MenuItemType::Submenu {
|
if child_.item_type == MenuItemType::Submenu {
|
||||||
remove_from_children_stores(id, child_.children.as_ref().unwrap());
|
remove_from_children_stores(internal_id, child_.children.as_ref().unwrap());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
remove_from_children_stores(&self.id, &self.children);
|
remove_from_children_stores(self.internal_id, &self.children);
|
||||||
|
|
||||||
for child in &self.children {
|
for child in &self.children {
|
||||||
let child_ = child.borrow();
|
let child_ = child.borrow();
|
||||||
|
@ -141,8 +142,10 @@ impl Drop for Menu {
|
||||||
|
|
||||||
impl Menu {
|
impl Menu {
|
||||||
pub fn new(id: Option<MenuId>) -> Self {
|
pub fn new(id: Option<MenuId>) -> Self {
|
||||||
|
let internal_id = COUNTER.next();
|
||||||
Self {
|
Self {
|
||||||
id: id.unwrap_or_else(|| MenuId(COUNTER.next().to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
|
internal_id,
|
||||||
hmenu: unsafe { CreateMenu() },
|
hmenu: unsafe { CreateMenu() },
|
||||||
hpopupmenu: unsafe { CreatePopupMenu() },
|
hpopupmenu: unsafe { CreatePopupMenu() },
|
||||||
haccel_store: Rc::new(RefCell::new((0, HashMap::new()))),
|
haccel_store: Rc::new(RefCell::new((0, HashMap::new()))),
|
||||||
|
@ -162,7 +165,7 @@ impl Menu {
|
||||||
child
|
child
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
.root_menu_haccel_stores
|
.root_menu_haccel_stores
|
||||||
.insert(self.id.clone(), self.haccel_store.clone());
|
.insert(self.internal_id, self.haccel_store.clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -407,7 +410,7 @@ pub(crate) struct MenuChild {
|
||||||
text: String,
|
text: String,
|
||||||
enabled: bool,
|
enabled: bool,
|
||||||
parents_hemnu: Vec<HMENU>,
|
parents_hemnu: Vec<HMENU>,
|
||||||
root_menu_haccel_stores: HashMap<MenuId, Rc<RefCell<AccelWrapper>>>,
|
root_menu_haccel_stores: HashMap<u32, Rc<RefCell<AccelWrapper>>>,
|
||||||
|
|
||||||
// menu item fields
|
// menu item fields
|
||||||
internal_id: u32,
|
internal_id: u32,
|
||||||
|
@ -461,7 +464,7 @@ impl MenuChild {
|
||||||
enabled,
|
enabled,
|
||||||
parents_hemnu: Vec::new(),
|
parents_hemnu: Vec::new(),
|
||||||
internal_id,
|
internal_id,
|
||||||
id: id.unwrap_or_else(|| MenuId(internal_id.to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
accelerator,
|
accelerator,
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
predefined_item_type: None,
|
predefined_item_type: None,
|
||||||
|
@ -483,7 +486,7 @@ impl MenuChild {
|
||||||
children: Some(Vec::new()),
|
children: Some(Vec::new()),
|
||||||
hmenu: unsafe { CreateMenu() },
|
hmenu: unsafe { CreateMenu() },
|
||||||
internal_id,
|
internal_id,
|
||||||
id: id.unwrap_or_else(|| MenuId(internal_id.to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
hpopupmenu: unsafe { CreatePopupMenu() },
|
hpopupmenu: unsafe { CreatePopupMenu() },
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
predefined_item_type: None,
|
predefined_item_type: None,
|
||||||
|
@ -501,7 +504,7 @@ impl MenuChild {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
parents_hemnu: Vec::new(),
|
parents_hemnu: Vec::new(),
|
||||||
internal_id,
|
internal_id,
|
||||||
id: MenuId(internal_id.to_string()),
|
id: MenuId::new(internal_id.to_string()),
|
||||||
accelerator: item_type.accelerator(),
|
accelerator: item_type.accelerator(),
|
||||||
predefined_item_type: Some(item_type),
|
predefined_item_type: Some(item_type),
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
|
@ -527,7 +530,7 @@ impl MenuChild {
|
||||||
enabled,
|
enabled,
|
||||||
parents_hemnu: Vec::new(),
|
parents_hemnu: Vec::new(),
|
||||||
internal_id,
|
internal_id,
|
||||||
id: id.unwrap_or_else(|| MenuId(internal_id.to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
accelerator,
|
accelerator,
|
||||||
checked,
|
checked,
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
|
@ -553,7 +556,7 @@ impl MenuChild {
|
||||||
enabled,
|
enabled,
|
||||||
parents_hemnu: Vec::new(),
|
parents_hemnu: Vec::new(),
|
||||||
internal_id,
|
internal_id,
|
||||||
id: id.unwrap_or_else(|| MenuId(internal_id.to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
accelerator,
|
accelerator,
|
||||||
icon,
|
icon,
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
|
@ -579,7 +582,7 @@ impl MenuChild {
|
||||||
enabled,
|
enabled,
|
||||||
parents_hemnu: Vec::new(),
|
parents_hemnu: Vec::new(),
|
||||||
internal_id,
|
internal_id,
|
||||||
id: id.unwrap_or_else(|| MenuId(internal_id.to_string())),
|
id: id.unwrap_or_else(|| MenuId::new(internal_id.to_string())),
|
||||||
accelerator,
|
accelerator,
|
||||||
root_menu_haccel_stores: HashMap::new(),
|
root_menu_haccel_stores: HashMap::new(),
|
||||||
predefined_item_type: None,
|
predefined_item_type: None,
|
||||||
|
|
Loading…
Reference in a new issue