Show key bindings in menu
This commit is contained in:
parent
380a3b2ff7
commit
004fd617ea
3 changed files with 140 additions and 94 deletions
68
src/key_bind.rs
Normal file
68
src/key_bind.rs
Normal 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
|
||||||
|
}
|
||||||
120
src/main.rs
120
src/main.rs
|
|
@ -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,
|
||||||
}),
|
}),
|
||||||
|
|
|
||||||
46
src/menu.rs
46
src/menu.rs
|
|
@ -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,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue