// Copyright 2022-2022 Tauri Programme within The Commons Conservancy // SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: MIT #![allow(clippy::uninlined_format_args)] //! muda is a Menu Utilities library for Desktop Applications. //! //! # Platforms supported: //! //! - Windows //! - macOS //! - Linux (gtk Only) //! //! # Platform-specific notes: //! //! - On Windows, accelerators don't work unless the win32 message loop calls //! [`TranslateAcceleratorW`](https://docs.rs/windows-sys/latest/windows_sys/Win32/UI/WindowsAndMessaging/fn.TranslateAcceleratorW.html). //! See [`Menu::init_for_hwnd`](https://docs.rs/muda/latest/muda/struct.Menu.html#method.init_for_hwnd) for more details //! //! # Dependencies (Linux Only) //! //! `gtk` is used for menus and `libxdo` is used to make the predfined `Copy`, `Cut`, `Paste` and `SelectAll` menu items work. Be sure to install following packages before building: //! //! #### Arch Linux / Manjaro: //! //! ```sh //! pacman -S gtk3 xdotool //! ``` //! //! #### Debian / Ubuntu: //! //! ```sh //! sudo apt install libgtk-3-dev libxdo-dev //! ``` //! //! # Example //! //! Create the menu and add your items //! //! ```no_run //! # use muda::{Menu, Submenu, MenuItem, accelerator::{Code, Modifiers, Accelerator}, PredefinedMenuItem}; //! let menu = Menu::new(); //! let menu_item2 = MenuItem::new("Menu item #2", false, None); //! let submenu = Submenu::with_items( //! "Submenu Outer", //! true, //! &[ //! &MenuItem::new( //! "Menu item #1", //! true, //! Some(Accelerator::new(Some(Modifiers::ALT), Code::KeyD)), //! ), //! &PredefinedMenuItem::separator(), //! &menu_item2, //! &MenuItem::new("Menu item #3", true, None), //! &PredefinedMenuItem::separator(), //! &Submenu::with_items( //! "Submenu Inner", //! true, //! &[ //! &MenuItem::new("Submenu item #1", true, None), //! &PredefinedMenuItem::separator(), //! &menu_item2, //! ], //! ).unwrap(), //! ], //! ); //! ``` //! //! Then add your root menu to a Window on Windows and Linux //! or use it as your global app menu on macOS //! //! ```no_run //! # let menu = muda::Menu::new(); //! # let window_hwnd = 0; //! # #[cfg(target_os = "linux")] //! # let gtk_window = gtk::ApplicationWindow::builder().build(); //! # #[cfg(target_os = "linux")] //! # let vertical_gtk_box = gtk::Box::new(gtk::Orientation::Vertical, 0); //! // --snip-- //! #[cfg(target_os = "windows")] //! menu.init_for_hwnd(window_hwnd); //! #[cfg(target_os = "linux")] //! menu.init_for_gtk_window(>k_window, Some(&vertical_gtk_box)); //! #[cfg(target_os = "macos")] //! menu.init_for_nsapp(); //! ``` //! //! # Context menus (Popup menus) //! //! You can also use a [`Menu`] or a [`Submenu`] show a context menu. //! //! ```no_run //! use muda::ContextMenu; //! # let menu = muda::Menu::new(); //! # let window_hwnd = 0; //! # #[cfg(target_os = "linux")] //! # let gtk_window = gtk::ApplicationWindow::builder().build(); //! # #[cfg(target_os = "macos")] //! # let nsview = 0 as *mut objc::runtime::Object; //! // --snip-- //! let position = muda::PhysicalPosition { x: 100., y: 120. }; //! #[cfg(target_os = "windows")] //! menu.show_context_menu_for_hwnd(window_hwnd, Some(position.into())); //! #[cfg(target_os = "linux")] //! menu.show_context_menu_for_gtk_window(>k_window, Some(position.into())); //! #[cfg(target_os = "macos")] //! menu.show_context_menu_for_nsview(nsview, Some(position.into())); //! ``` //! # Processing menu events //! //! You can use [`MenuEvent::receiver`] to get a reference to the [`MenuEventReceiver`] //! which you can use to listen to events when a menu item is activated //! ```no_run //! # use muda::MenuEvent; //! # //! # let save_item: muda::MenuItem = unsafe { std::mem::zeroed() }; //! if let Ok(event) = MenuEvent::receiver().try_recv() { //! match event.id { //! id if id == save_item.id() => { //! println!("Save menu item activated"); //! }, //! _ => {} //! } //! } //! ``` use std::{convert::Infallible, str::FromStr}; use crossbeam_channel::{unbounded, Receiver, Sender}; use once_cell::sync::{Lazy, OnceCell}; mod about_metadata; pub mod accelerator; mod builders; mod dpi; mod error; mod icon; mod items; mod menu; mod platform_impl; mod util; #[cfg(target_os = "macos")] #[macro_use] extern crate objc; pub use about_metadata::AboutMetadata; pub use builders::*; pub use dpi::*; pub use error::*; pub use icon::{BadIcon, Icon, NativeIcon}; pub use items::*; pub use menu::Menu; /// 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 for MenuId { fn as_ref(&self) -> &str { self.0.as_ref() } } impl From 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 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 { Ok(Self(s.to_string())) } } /// An enumeration of all available menu types, useful to match against /// the items returned from [`Menu::items`] or [`Submenu::items`] #[derive(Clone)] pub enum MenuItemKind { MenuItem(MenuItem), Submenu(Submenu), Predefined(PredefinedMenuItem), Check(CheckMenuItem), Icon(IconMenuItem), } impl MenuItemKind { /// Casts this item to a [`MenuItem`], and returns `None` if it wasn't. pub fn as_menuitem(&self) -> Option<&MenuItem> { match self { MenuItemKind::MenuItem(i) => Some(i), _ => None, } } /// Casts this item to a [`MenuItem`], and panics if it wasn't. pub fn as_menuitem_unchecked(&self) -> &MenuItem { match self { MenuItemKind::MenuItem(i) => i, _ => panic!("Not a MenuItem"), } } /// Casts this item to a [`Submenu`], and returns `None` if it wasn't. pub fn as_submenu(&self) -> Option<&Submenu> { match self { MenuItemKind::Submenu(i) => Some(i), _ => None, } } /// Casts this item to a [`Submenu`], and panics if it wasn't. pub fn as_submenu_unchecked(&self) -> &Submenu { match self { MenuItemKind::Submenu(i) => i, _ => panic!("Not a Submenu"), } } /// Casts this item to a [`PredefinedMenuItem`], and returns `None` if it wasn't. pub fn as_predefined_menuitem(&self) -> Option<&PredefinedMenuItem> { match self { MenuItemKind::Predefined(i) => Some(i), _ => None, } } /// Casts this item to a [`PredefinedMenuItem`], and panics if it wasn't. pub fn as_predefined_menuitem_unchecked(&self) -> &PredefinedMenuItem { match self { MenuItemKind::Predefined(i) => i, _ => panic!("Not a PredefinedMenuItem"), } } /// Casts this item to a [`CheckMenuItem`], and returns `None` if it wasn't. pub fn as_check_menuitem(&self) -> Option<&CheckMenuItem> { match self { MenuItemKind::Check(i) => Some(i), _ => None, } } /// Casts this item to a [`CheckMenuItem`], and panics if it wasn't. pub fn as_check_menuitem_unchecked(&self) -> &CheckMenuItem { match self { MenuItemKind::Check(i) => i, _ => panic!("Not a CheckMenuItem"), } } /// Casts this item to a [`IconMenuItem`], and returns `None` if it wasn't. pub fn as_icon_menuitem(&self) -> Option<&IconMenuItem> { match self { MenuItemKind::Icon(i) => Some(i), _ => None, } } /// Casts this item to a [`IconMenuItem`], and panics if it wasn't. pub fn as_icon_menuitem_unchecked(&self) -> &IconMenuItem { match self { MenuItemKind::Icon(i) => i, _ => panic!("Not an IconMenuItem"), } } } /// A trait that defines a generic item in a menu, which may be one of [`MenuItemKind`] /// /// # Safety /// /// This trait is ONLY meant to be implemented internally by the crate. pub unsafe trait IsMenuItem { fn kind(&self) -> MenuItemKind; fn id(&self) -> &MenuId; } #[derive(Debug, PartialEq, PartialOrd, Clone, Copy)] pub(crate) enum MenuItemType { MenuItem, Submenu, Predefined, Check, Icon, } impl Default for MenuItemType { fn default() -> Self { Self::MenuItem } } /// A helper trait with methods to help creating a context menu. pub trait ContextMenu { /// Get the popup [`HMENU`] for this menu. /// /// [`HMENU`]: windows_sys::Win32::UI::WindowsAndMessaging::HMENU #[cfg(target_os = "windows")] fn hpopupmenu(&self) -> windows_sys::Win32::UI::WindowsAndMessaging::HMENU; /// Shows this menu as a context menu inside a win32 window. /// /// - `position` is relative to the window top-left corner, if `None`, the cursor position is used. #[cfg(target_os = "windows")] fn show_context_menu_for_hwnd(&self, hwnd: isize, position: Option); /// Attach the menu subclass handler to the given hwnd /// so you can recieve events from that window using [MenuEvent::receiver] /// /// This can be used along with [`ContextMenu::hpopupmenu`] when implementing a tray icon menu. #[cfg(target_os = "windows")] fn attach_menu_subclass_for_hwnd(&self, hwnd: isize); /// Remove the menu subclass handler from the given hwnd #[cfg(target_os = "windows")] fn detach_menu_subclass_from_hwnd(&self, hwnd: isize); /// Shows this menu as a context menu inside a [`gtk::ApplicationWindow`] /// /// - `position` is relative to the window top-left corner, if `None`, the cursor position is used. #[cfg(target_os = "linux")] fn show_context_menu_for_gtk_window( &self, w: >k::ApplicationWindow, position: Option, ); /// Get the underlying gtk menu reserved for context menus. #[cfg(target_os = "linux")] fn gtk_context_menu(&self) -> gtk::Menu; /// Shows this menu as a context menu for the specified `NSView`. /// /// - `position` is relative to the window top-left corner, if `None`, the cursor position is used. #[cfg(target_os = "macos")] fn show_context_menu_for_nsview(&self, view: cocoa::base::id, position: Option); /// Get the underlying NSMenu reserved for context menus. #[cfg(target_os = "macos")] fn ns_menu(&self) -> *mut std::ffi::c_void; } /// Describes a menu event emitted when a menu item is activated #[derive(Debug, Clone)] pub struct MenuEvent { /// Id of the menu item which triggered this event pub id: MenuId, } /// A reciever that could be used to listen to menu events. pub type MenuEventReceiver = Receiver; type MenuEventHandler = Box; static MENU_CHANNEL: Lazy<(Sender, MenuEventReceiver)> = Lazy::new(unbounded); static MENU_EVENT_HANDLER: OnceCell> = OnceCell::new(); impl MenuEvent { /// Returns the id of the menu item which triggered this event pub fn id(&self) -> &MenuId { &self.id } /// Gets a reference to the event channel's [`MenuEventReceiver`] /// which can be used to listen for menu events. /// /// ## Note /// /// This will not receive any events if [`MenuEvent::set_event_handler`] has been called with a `Some` value. pub fn receiver<'a>() -> &'a MenuEventReceiver { &MENU_CHANNEL.1 } /// Set a handler to be called for new events. Useful for implementing custom event sender. /// /// ## Note /// /// Calling this function with a `Some` value, /// will not send new events to the channel associated with [`MenuEvent::receiver`] pub fn set_event_handler(f: Option) { if let Some(f) = f { let _ = MENU_EVENT_HANDLER.set(Some(Box::new(f))); } else { let _ = MENU_EVENT_HANDLER.set(None); } } pub(crate) fn send(event: MenuEvent) { if let Some(handler) = MENU_EVENT_HANDLER.get_or_init(|| None) { handler(event); } else { let _ = MENU_CHANNEL.0.send(event); } } }