Show key bindings in menu

This commit is contained in:
Jeremy Soller 2024-01-29 12:21:54 -07:00
parent 380a3b2ff7
commit 004fd617ea
No known key found for this signature in database
GPG key ID: D02FD439211AF56F
3 changed files with 140 additions and 94 deletions

68
src/key_bind.rs Normal file
View file

@ -0,0 +1,68 @@
use cosmic::iced::keyboard::{KeyCode, Modifiers};
use serde::{Deserialize, Serialize};
use std::{collections::HashMap, fmt};
use crate::Action;
#[derive(Clone, Copy, Debug, Deserialize, Eq, Hash, Ord, PartialEq, PartialOrd, Serialize)]
pub enum Modifier {
Super,
Ctrl,
Alt,
Shift,
}
#[derive(Clone, Debug, Deserialize, Eq, Hash, Ord, PartialEq, PartialOrd, Serialize)]
pub struct KeyBind {
pub modifiers: Vec<Modifier>,
pub key_code: KeyCode,
}
impl KeyBind {
pub fn matches(&self, modifiers: Modifiers, key_code: KeyCode) -> bool {
self.key_code == key_code
&& modifiers.logo() == self.modifiers.contains(&Modifier::Super)
&& modifiers.control() == self.modifiers.contains(&Modifier::Ctrl)
&& modifiers.alt() == self.modifiers.contains(&Modifier::Alt)
&& modifiers.shift() == self.modifiers.contains(&Modifier::Shift)
}
}
impl fmt::Display for KeyBind {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
for modifier in self.modifiers.iter() {
write!(f, "{:?} + ", modifier)?;
}
write!(f, "{:?}", self.key_code)
}
}
//TODO: load from config
pub fn key_binds() -> HashMap<KeyBind, Action> {
let mut key_binds = HashMap::new();
macro_rules! bind {
([$($modifier:ident),+ $(,)?], $key_code:ident, $action:ident) => {{
key_binds.insert(
KeyBind {
modifiers: vec![$(Modifier::$modifier),+],
key_code: KeyCode::$key_code,
},
Action::$action,
);
}};
}
bind!([Ctrl], C, Copy);
bind!([Ctrl], X, Cut);
bind!([Ctrl], V, Paste);
bind!([Ctrl], A, SelectAll);
bind!([Ctrl], W, TabClose);
bind!([Ctrl], T, TabNew);
bind!([Ctrl], Tab, TabNext);
bind!([Ctrl, Shift], Tab, TabPrev);
bind!([Ctrl], Q, WindowClose);
bind!([Ctrl], N, WindowNew);
key_binds
}

View file

@ -15,11 +15,14 @@ use cosmic::{
widget::{self, segmented_button}, widget::{self, segmented_button},
Application, ApplicationExt, Element, Application, ApplicationExt, Element,
}; };
use std::{any::TypeId, env, fs, path::PathBuf, process}; use std::{any::TypeId, env, fs, path::PathBuf, process, collections::HashMap};
use config::{AppTheme, Config, CONFIG_VERSION}; use config::{AppTheme, Config, CONFIG_VERSION};
mod config; mod config;
use key_bind::{key_binds, KeyBind};
mod key_bind;
mod localize; mod localize;
mod menu; mod menu;
@ -104,7 +107,7 @@ pub struct Flags {
config: Config, config: Config,
} }
#[derive(Clone, Copy, Debug)] #[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum Action { pub enum Action {
Copy, Copy,
Cut, Cut,
@ -116,23 +119,37 @@ pub enum Action {
RestoreFromTrash, RestoreFromTrash,
SelectAll, SelectAll,
Settings, Settings,
TabClose,
TabNew, TabNew,
TabNext,
TabPrev,
TabViewGrid,
TabViewList,
WindowClose,
WindowNew,
} }
impl Action { impl Action {
pub fn message(self, entity: segmented_button::Entity) -> Message { pub fn message(self, entity_opt: Option<segmented_button::Entity>) -> Message {
match self { match self {
Action::Copy => Message::Copy(Some(entity)), Action::Copy => Message::Copy(entity_opt),
Action::Cut => Message::Cut(Some(entity)), Action::Cut => Message::Cut(entity_opt),
Action::MoveToTrash => Message::MoveToTrash(Some(entity)), Action::MoveToTrash => Message::MoveToTrash(entity_opt),
Action::NewFile => Message::NewFile(Some(entity)), Action::NewFile => Message::NewFile(entity_opt),
Action::NewFolder => Message::NewFolder(Some(entity)), Action::NewFolder => Message::NewFolder(entity_opt),
Action::Paste => Message::Paste(Some(entity)), Action::Paste => Message::Paste(entity_opt),
Action::Properties => Message::ToggleContextPage(ContextPage::Properties), Action::Properties => Message::ToggleContextPage(ContextPage::Properties),
Action::RestoreFromTrash => Message::RestoreFromTrash(Some(entity)), Action::RestoreFromTrash => Message::RestoreFromTrash(entity_opt),
Action::SelectAll => Message::SelectAll(Some(entity)), Action::SelectAll => Message::SelectAll(entity_opt),
Action::Settings => Message::ToggleContextPage(ContextPage::Settings), Action::Settings => Message::ToggleContextPage(ContextPage::Settings),
Action::TabClose => Message::TabClose(entity_opt),
Action::TabNew => Message::TabNew, Action::TabNew => Message::TabNew,
Action::TabNext => Message::TabNext,
Action::TabPrev => Message::TabPrev,
Action::TabViewGrid => Message::TabMessage(entity_opt, tab::Message::View(tab::View::Grid)),
Action::TabViewList => Message::TabMessage(entity_opt, tab::Message::View(tab::View::List)),
Action::WindowClose => Message::WindowClose,
Action::WindowNew => Message::WindowNew,
} }
} }
} }
@ -145,7 +162,8 @@ pub enum Message {
Config(Config), Config(Config),
Copy(Option<segmented_button::Entity>), Copy(Option<segmented_button::Entity>),
Cut(Option<segmented_button::Entity>), Cut(Option<segmented_button::Entity>),
KeyModifiers(Modifiers), Key(Modifiers, KeyCode),
Modifiers(Modifiers),
MoveToTrash(Option<segmented_button::Entity>), MoveToTrash(Option<segmented_button::Entity>),
NewFile(Option<segmented_button::Entity>), NewFile(Option<segmented_button::Entity>),
NewFolder(Option<segmented_button::Entity>), NewFolder(Option<segmented_button::Entity>),
@ -191,6 +209,7 @@ pub struct App {
config: Config, config: Config,
app_themes: Vec<String>, app_themes: Vec<String>,
context_page: ContextPage, context_page: ContextPage,
key_binds: HashMap<KeyBind, Action>,
modifiers: Modifiers, modifiers: Modifiers,
} }
@ -367,6 +386,7 @@ impl Application for App {
config: flags.config, config: flags.config,
app_themes, app_themes,
context_page: ContextPage::Settings, context_page: ContextPage::Settings,
key_binds: key_binds(),
modifiers: Modifiers::empty(), modifiers: Modifiers::empty(),
}; };
@ -475,7 +495,15 @@ impl Application for App {
Message::Cut(entity_opt) => { Message::Cut(entity_opt) => {
log::warn!("TODO: CUT"); log::warn!("TODO: CUT");
} }
Message::KeyModifiers(modifiers) => { Message::Key(modifiers, key_code) => {
let entity = self.tab_model.active();
for (key_bind, action) in self.key_binds.iter() {
if key_bind.matches(modifiers, key_code) {
return self.update(action.message(Some(entity)));
}
}
}
Message::Modifiers(modifiers) => {
self.modifiers = modifiers; self.modifiers = modifiers;
} }
Message::MoveToTrash(entity_opt) => { Message::MoveToTrash(entity_opt) => {
@ -578,7 +606,7 @@ impl Application for App {
tab.context_menu = None; tab.context_menu = None;
} }
// Run action's message // Run action's message
return self.update(action.message(entity)); return self.update(action.message(Some(entity)));
} }
_ => {} _ => {}
} }
@ -671,7 +699,7 @@ impl Application for App {
fn header_start(&self) -> Vec<Element<Self::Message>> { fn header_start(&self) -> Vec<Element<Self::Message>> {
vec![ vec![
menu::menu_bar().into(), menu::menu_bar(&self.key_binds).into(),
//TODO: use theme defined space? //TODO: use theme defined space?
widget::horizontal_space(Length::Fixed(32.0)).into(), widget::horizontal_space(Length::Fixed(32.0)).into(),
] ]
@ -753,69 +781,13 @@ impl Application for App {
Subscription::batch([ Subscription::batch([
event::listen_with(|event, _status| match event { event::listen_with(|event, _status| match event {
Event::Keyboard(KeyEvent::KeyPressed { Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::A, key_code,
modifiers, modifiers,
}) => { }) => {
if modifiers == Modifiers::CTRL { Some(Message::Key(modifiers, key_code))
Some(Message::SelectAll(None))
} else {
None
}
}
Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::C,
modifiers,
}) => {
if modifiers == Modifiers::CTRL {
Some(Message::Copy(None))
} else {
None
}
}
Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::X,
modifiers,
}) => {
if modifiers == Modifiers::CTRL {
Some(Message::Cut(None))
} else {
None
}
}
Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::T,
modifiers,
}) => {
if modifiers == Modifiers::CTRL {
Some(Message::TabNew)
} else {
None
}
}
Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::W,
modifiers: Modifiers::CTRL,
}) => Some(Message::TabClose(None)),
Event::Keyboard(KeyEvent::KeyPressed {
key_code: key @ (KeyCode::PageUp | KeyCode::PageDown),
modifiers: Modifiers::CTRL,
}) => match key {
KeyCode::PageDown => Some(Message::TabPrev),
KeyCode::PageUp => Some(Message::TabNext),
_ => None,
},
Event::Keyboard(KeyEvent::KeyPressed {
key_code: KeyCode::V,
modifiers,
}) => {
if modifiers == Modifiers::CTRL {
Some(Message::Paste(None))
} else {
None
}
} }
Event::Keyboard(KeyEvent::ModifiersChanged(modifiers)) => { Event::Keyboard(KeyEvent::ModifiersChanged(modifiers)) => {
Some(Message::KeyModifiers(modifiers)) Some(Message::Modifiers(modifiers))
} }
_ => None, _ => None,
}), }),

View file

@ -11,8 +11,9 @@ use cosmic::{
}, },
Element, Element,
}; };
use std::collections::HashMap;
use crate::{fl, tab, Action, ContextPage, Location, Message, Tab}; use crate::{fl, KeyBind, tab, Action, ContextPage, Location, Message, Tab};
macro_rules! menu_button { macro_rules! menu_button {
($($x:expr),+ $(,)?) => ( ($($x:expr),+ $(,)?) => (
@ -30,6 +31,7 @@ macro_rules! menu_button {
} }
pub fn context_menu<'a>(entity: segmented_button::Entity, tab: &Tab) -> Element<'a, Message> { pub fn context_menu<'a>(entity: segmented_button::Entity, tab: &Tab) -> Element<'a, Message> {
//TODO: show key bindings in context menu?
let menu_action = |label, action| { let menu_action = |label, action| {
menu_button!(widget::text(label)).on_press(Message::TabContextAction(entity, action)) menu_button!(widget::text(label)).on_press(Message::TabContextAction(entity, action))
}; };
@ -87,7 +89,7 @@ pub fn context_menu<'a>(entity: segmented_button::Entity, tab: &Tab) -> Element<
.into() .into()
} }
pub fn menu_bar<'a>() -> Element<'a, Message> { pub fn menu_bar<'a>(key_binds: &HashMap<KeyBind, Action>) -> Element<'a, Message> {
//TODO: port to libcosmic //TODO: port to libcosmic
let menu_root = |label| { let menu_root = |label| {
widget::button(widget::text(label)) widget::button(widget::text(label))
@ -95,20 +97,24 @@ pub fn menu_bar<'a>() -> Element<'a, Message> {
.style(theme::Button::MenuRoot) .style(theme::Button::MenuRoot)
}; };
let find_key = |message: &Message| -> String { let find_key = |action: &Action| -> String {
//TODO: hotkey config for (key_bind, key_action) in key_binds.iter() {
if action == key_action {
return key_bind.to_string();
}
}
String::new() String::new()
}; };
let menu_item = |label, message| { let menu_item = |label, action| {
let key = find_key(&message); let key = find_key(&action);
MenuTree::new( MenuTree::new(
menu_button!( menu_button!(
widget::text(label), widget::text(label),
widget::horizontal_space(Length::Fill), widget::horizontal_space(Length::Fill),
widget::text(key) widget::text(key)
) )
.on_press(message), .on_press(action.message(None)),
) )
}; };
@ -116,23 +122,23 @@ pub fn menu_bar<'a>() -> Element<'a, Message> {
MenuTree::with_children( MenuTree::with_children(
menu_root(fl!("file")), menu_root(fl!("file")),
vec![ vec![
menu_item(fl!("new-tab"), Message::TabNew), menu_item(fl!("new-tab"), Action::TabNew),
menu_item(fl!("new-window"), Message::WindowNew), menu_item(fl!("new-window"), Action::WindowNew),
menu_item(fl!("new-file"), Message::NewFile(None)), menu_item(fl!("new-file"), Action::NewFile),
menu_item(fl!("new-folder"), Message::NewFolder(None)), menu_item(fl!("new-folder"), Action::NewFolder),
MenuTree::new(horizontal_rule(1)), MenuTree::new(horizontal_rule(1)),
menu_item(fl!("close-tab"), Message::TabClose(None)), menu_item(fl!("close-tab"), Action::TabClose),
MenuTree::new(horizontal_rule(1)), MenuTree::new(horizontal_rule(1)),
menu_item(fl!("quit"), Message::WindowClose), menu_item(fl!("quit"), Action::WindowClose),
], ],
), ),
MenuTree::with_children( MenuTree::with_children(
menu_root(fl!("edit")), menu_root(fl!("edit")),
vec![ vec![
menu_item(fl!("cut"), Message::Cut(None)), menu_item(fl!("cut"), Action::Cut),
menu_item(fl!("copy"), Message::Copy(None)), menu_item(fl!("copy"), Action::Copy),
menu_item(fl!("paste"), Message::Paste(None)), menu_item(fl!("paste"), Action::Paste),
menu_item(fl!("select-all"), Message::SelectAll(None)), menu_item(fl!("select-all"), Action::SelectAll),
], ],
), ),
MenuTree::with_children( MenuTree::with_children(
@ -140,16 +146,16 @@ pub fn menu_bar<'a>() -> Element<'a, Message> {
vec![ vec![
menu_item( menu_item(
fl!("grid-view"), fl!("grid-view"),
Message::TabMessage(None, tab::Message::View(tab::View::Grid)), Action::TabViewGrid
), ),
menu_item( menu_item(
fl!("list-view"), fl!("list-view"),
Message::TabMessage(None, tab::Message::View(tab::View::List)), Action::TabViewList
), ),
MenuTree::new(horizontal_rule(1)), MenuTree::new(horizontal_rule(1)),
menu_item( menu_item(
fl!("menu-settings"), fl!("menu-settings"),
Message::ToggleContextPage(ContextPage::Settings), Action::Settings,
), ),
], ],
), ),