From 8de378dc10460b234c7230bdc390d92938c3fa95 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?W=C3=B6lfchen?= Date: Fri, 11 Oct 2024 11:27:24 +0000 Subject: [PATCH] feat: button improvements (#1187) * feat: add keyboard support for button presses Co-authored-by: Julian Schuler <31921487+julianschuler@users.noreply.github.com> * chore: attribution * fix: activate eventbox on button release; minor fixes * docs: improve button and eventbox documentation Co-authored-by: Julian Schuler <31921487+julianschuler@users.noreply.github.com> --------- Co-authored-by: Julian Schuler <31921487+julianschuler@users.noreply.github.com> --- CHANGELOG.md | 1 + crates/eww/src/widgets/widget_definitions.rs | 61 ++++++++++++-------- 2 files changed, 37 insertions(+), 25 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cf59e14..90db61f3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,7 @@ All notable changes to eww will be listed here, starting at changes since versio - Add `min` and `max` function calls to simplexpr (By: ovalkonia) - Add `flip-x`, `flip-y`, `vertical` options to the graph widget to determine its direction - Add `transform-origin-x`/`transform-origin-y` properties to transform widget (By: mario-kr) +- Add keyboard support for button presses (By: julianschuler) ## [0.6.0] (21.04.2024) diff --git a/crates/eww/src/widgets/widget_definitions.rs b/crates/eww/src/widgets/widget_definitions.rs index 394237a0..29c14a08 100644 --- a/crates/eww/src/widgets/widget_definitions.rs +++ b/crates/eww/src/widgets/widget_definitions.rs @@ -38,16 +38,16 @@ use yuck::{ /// thus not connecting a new handler unless the condition is met. macro_rules! connect_signal_handler { ($widget:ident, if $cond:expr, $connect_expr:expr) => {{ + const KEY:&str = std::concat!("signal-handler:", std::line!()); unsafe { - let key = ::std::concat!("signal-handler:", ::std::line!()); - let old = $widget.data::(key); + let old = $widget.data::(KEY); if let Some(old) = old { let a = old.as_ref().as_raw(); $widget.disconnect(gtk::glib::SignalHandlerId::from_glib(a)); } - $widget.set_data::(key, $connect_expr); + $widget.set_data::(KEY, $connect_expr); } }}; ($widget:ident, $connect_expr:expr) => {{ @@ -495,7 +495,6 @@ fn build_gtk_input(bargs: &mut BuilderArgs) -> Result { prop(value: as_string) { gtk_widget.set_text(&value); }, - // @prop onchange - Command to run when the text changes. The placeholder `{}` will be replaced by the value // @prop timeout - timeout of the command. Default: "200ms" prop(timeout: as_duration = Duration::from_millis(200), onchange: as_string) { @@ -520,7 +519,7 @@ fn build_gtk_input(bargs: &mut BuilderArgs) -> Result { const WIDGET_NAME_BUTTON: &str = "button"; /// @widget button -/// @desc A button +/// @desc A button containing any widget as it's child. Events are triggered on release. fn build_gtk_button(bargs: &mut BuilderArgs) -> Result { let gtk_widget = gtk::Button::new(); @@ -528,15 +527,22 @@ fn build_gtk_button(bargs: &mut BuilderArgs) -> Result { prop( // @prop timeout - timeout of the command. Default: "200ms" timeout: as_duration = Duration::from_millis(200), - // @prop onclick - a command that get's run when the button is clicked + // @prop onclick - command to run when the button is activated either by leftclicking or keyboard onclick: as_string = "", - // @prop onmiddleclick - a command that get's run when the button is middleclicked + // @prop onmiddleclick - command to run when the button is middleclicked onmiddleclick: as_string = "", - // @prop onrightclick - a command that get's run when the button is rightclicked + // @prop onrightclick - command to run when the button is rightclicked onrightclick: as_string = "" ) { - gtk_widget.add_events(gdk::EventMask::BUTTON_PRESS_MASK); - connect_signal_handler!(gtk_widget, gtk_widget.connect_button_press_event(move |_, evt| { + // animate button upon right-/middleclick (if gtk theme supports it) + // since we do this, we can't use `connect_clicked` as that would always run `onclick` as well + connect_signal_handler!(gtk_widget, gtk_widget.connect_button_press_event(move |button, _| { + button.emit_activate(); + glib::Propagation::Proceed + })); + let onclick_ = onclick.clone(); + // mouse click events + connect_signal_handler!(gtk_widget, gtk_widget.connect_button_release_event(move |_, evt| { match evt.button() { 1 => run_command(timeout, &onclick, &[] as &[&str]), 2 => run_command(timeout, &onmiddleclick, &[] as &[&str]), @@ -545,8 +551,18 @@ fn build_gtk_button(bargs: &mut BuilderArgs) -> Result { } glib::Propagation::Proceed })); + // keyboard events + connect_signal_handler!(gtk_widget, gtk_widget.connect_key_release_event(move |_, evt| { + match evt.scancode() { + // return + 36 => run_command(timeout, &onclick_, &[] as &[&str]), + // space + 65 => run_command(timeout, &onclick_, &[] as &[&str]), + _ => {}, + } + glib::Propagation::Proceed + })); } - }); Ok(gtk_widget) } @@ -786,26 +802,26 @@ fn build_gtk_event_box(bargs: &mut BuilderArgs) -> Result { // Support :hover selector gtk_widget.connect_enter_notify_event(|gtk_widget, evt| { if evt.detail() != NotifyType::Inferior { - gtk_widget.clone().set_state_flags(gtk::StateFlags::PRELIGHT, false); + gtk_widget.set_state_flags(gtk::StateFlags::PRELIGHT, false); } glib::Propagation::Proceed }); gtk_widget.connect_leave_notify_event(|gtk_widget, evt| { if evt.detail() != NotifyType::Inferior { - gtk_widget.clone().unset_state_flags(gtk::StateFlags::PRELIGHT); + gtk_widget.unset_state_flags(gtk::StateFlags::PRELIGHT); } glib::Propagation::Proceed }); // Support :active selector gtk_widget.connect_button_press_event(|gtk_widget, _| { - gtk_widget.clone().set_state_flags(gtk::StateFlags::ACTIVE, false); + gtk_widget.set_state_flags(gtk::StateFlags::ACTIVE, false); glib::Propagation::Proceed }); gtk_widget.connect_button_release_event(|gtk_widget, _| { - gtk_widget.clone().unset_state_flags(gtk::StateFlags::ACTIVE); + gtk_widget.unset_state_flags(gtk::StateFlags::ACTIVE); glib::Propagation::Proceed }); @@ -916,21 +932,18 @@ fn build_gtk_event_box(bargs: &mut BuilderArgs) -> Result { }; })); }, - - // TODO the fact that we have the same code here as for button is ugly, as we want to keep consistency - prop( // @prop timeout - timeout of the command. Default: "200ms" timeout: as_duration = Duration::from_millis(200), - // @prop onclick - a command that get's run when the button is clicked + // @prop onclick - command to run when the widget is clicked onclick: as_string = "", - // @prop onmiddleclick - a command that get's run when the button is middleclicked + // @prop onmiddleclick - command to run when the widget is middleclicked onmiddleclick: as_string = "", - // @prop onrightclick - a command that get's run when the button is rightclicked + // @prop onrightclick - command to run when the widget is rightclicked onrightclick: as_string = "" ) { gtk_widget.add_events(gdk::EventMask::BUTTON_PRESS_MASK); - connect_signal_handler!(gtk_widget, gtk_widget.connect_button_press_event(move |_, evt| { + connect_signal_handler!(gtk_widget, gtk_widget.connect_button_release_event(move |_, evt| { match evt.button() { 1 => run_command(timeout, &onclick, &[] as &[&str]), 2 => run_command(timeout, &onmiddleclick, &[] as &[&str]), @@ -941,7 +954,6 @@ fn build_gtk_event_box(bargs: &mut BuilderArgs) -> Result { })); } }); - Ok(gtk_widget) } @@ -1182,8 +1194,7 @@ fn build_gtk_stack(bargs: &mut BuilderArgs) -> Result { const WIDGET_NAME_TRANSFORM: &str = "transform"; /// @widget transform -/// @desc A widget that applies transformations to its content. They are applied in the following -/// order: rotate->translate->scale) +/// @desc A widget that applies transformations to its content. They are applied in the following order: rotate -> translate -> scale fn build_transform(bargs: &mut BuilderArgs) -> Result { let w = Transform::new(); def_widget!(bargs, _g, w, {