forked from pop-os/cosmic-settings-daemon
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Gesture configuration in shortcuts config
This commit adds a Gesture abstraction, which can define gestures by the number of fingers used, and the direction the gesture is performed in. Oftentimes, we want the gesture to be either absolute (i.e. ignoring context such as workspace direction), or relative (if workspaces are horizontal, forward would be right, backward would be left. if workspaces are vertical, forward would be up, backward would be down). The abstraction is modeled after the Binding type, and the resulting Gestures type is similar to the Shortcuts type. This is the first step in implementing configurable touchpad gestures (pop-os/cosmic-comp#396) Signed-off-by: Ryan Brue <[email protected]>
- Loading branch information
Showing
2 changed files
with
291 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,201 @@ | ||
// SPDX-License-Identifier: MPL-2.0 | ||
use std::str::FromStr; | ||
|
||
use serde::{Deserialize, Serialize}; | ||
|
||
/// Description of a gesture that can be handled by the compositor | ||
#[serde_with::serde_as] | ||
#[derive(Clone, Debug, Deserialize, PartialEq, Eq, Serialize, Hash)] | ||
#[serde(deny_unknown_fields)] | ||
pub struct Gesture { | ||
/// How many fingers are held down | ||
pub fingers: u32, | ||
pub direction: Direction, | ||
// A custom description for a custom binding | ||
#[serde(default, skip_serializing_if = "Option::is_none")] | ||
pub description: Option<String>, | ||
} | ||
|
||
/// Describes a direction, either absolute or relative | ||
#[serde_with::serde_as] | ||
#[derive(Clone, Debug, Deserialize, PartialEq, Eq, Serialize, Hash)] | ||
#[serde(deny_unknown_fields)] | ||
pub enum Direction { | ||
Relative(RelativeDirection), | ||
Absolute(AbsoluteDirection), | ||
} | ||
|
||
impl ToString for Direction { | ||
fn to_string(&self) -> String { | ||
match self { | ||
Direction::Absolute(abs) => match abs { | ||
AbsoluteDirection::AbsoluteUp => "AbsoluteUp".to_string(), | ||
AbsoluteDirection::AbsoluteDown => "AbsoluteDown".to_string(), | ||
AbsoluteDirection::AbsoluteLeft => "AbsoluteLeft".to_string(), | ||
AbsoluteDirection::AbsoluteRight => "AbsoluteRight".to_string(), | ||
}, | ||
Direction::Relative(rel) => match rel { | ||
RelativeDirection::RelativeForward => "RelativeForward".to_string(), | ||
RelativeDirection::RelativeBackward => "RelativeBackward".to_string(), | ||
RelativeDirection::RelativeLeft => "RelativeLeft".to_string(), | ||
RelativeDirection::RelativeRight => "RelativeRight".to_string(), | ||
}, | ||
} | ||
} | ||
} | ||
|
||
impl FromStr for Direction { | ||
type Err = String; | ||
|
||
fn from_str(s: &str) -> Result<Self, Self::Err> { | ||
match s { | ||
// Absolute directions | ||
"AbsoluteUp" => Ok(Direction::Absolute(AbsoluteDirection::AbsoluteUp)), | ||
"AbsoluteDown" => Ok(Direction::Absolute(AbsoluteDirection::AbsoluteDown)), | ||
"AbsoluteLeft" => Ok(Direction::Absolute(AbsoluteDirection::AbsoluteLeft)), | ||
"AbsoluteRight" => Ok(Direction::Absolute(AbsoluteDirection::AbsoluteRight)), | ||
// Relative directions | ||
"RelativeForward" => Ok(Direction::Relative(RelativeDirection::RelativeForward)), | ||
"RelativeBackward" => Ok(Direction::Relative(RelativeDirection::RelativeBackward)), | ||
"RelativeLeft" => Ok(Direction::Relative(RelativeDirection::RelativeLeft)), | ||
"RelativeRight" => Ok(Direction::Relative(RelativeDirection::RelativeRight)), | ||
_ => Err(format!("Invalid direction string")) | ||
} | ||
} | ||
} | ||
|
||
/// Describes a relative direction (typically relative to the workspace direction) | ||
#[serde_with::serde_as] | ||
#[derive(Clone, Debug, Deserialize, PartialEq, Eq, Serialize, Hash)] | ||
#[serde(deny_unknown_fields)] | ||
pub enum RelativeDirection { | ||
RelativeForward, | ||
RelativeBackward, | ||
RelativeLeft, | ||
RelativeRight, | ||
} | ||
|
||
/// Describes an absolute direction (i.e. not relative to workspace direction) | ||
#[serde_with::serde_as] | ||
#[derive(Clone, Debug, Deserialize, PartialEq, Eq, Serialize, Hash)] | ||
#[serde(deny_unknown_fields)] | ||
pub enum AbsoluteDirection { | ||
AbsoluteUp, | ||
AbsoluteDown, | ||
AbsoluteLeft, | ||
AbsoluteRight, | ||
} | ||
|
||
impl Gesture { | ||
/// Creates a new gesture from a number of fingers and a direction | ||
pub fn new(fingers: impl Into<u32>, direction: impl Into<Direction>) -> Gesture { | ||
Gesture { | ||
fingers: fingers.into(), | ||
direction: direction.into(), | ||
description: None, | ||
} | ||
} | ||
|
||
/// Returns true if the direction is absolute | ||
pub fn is_absolute(&self) -> bool { | ||
matches!(self.direction, Direction::Absolute(_)) | ||
} | ||
|
||
/// Append the binding to an existing string | ||
pub fn to_string_in_place(&self, string: &mut String) { | ||
string.push_str(&format!( | ||
"{} Finger {}", | ||
self.fingers, | ||
self.direction.to_string() | ||
)); | ||
} | ||
} | ||
|
||
impl ToString for Gesture { | ||
fn to_string(&self) -> String { | ||
let mut string = String::new(); | ||
self.to_string_in_place(&mut string); | ||
string | ||
} | ||
} | ||
|
||
|
||
impl FromStr for Gesture { | ||
type Err = String; | ||
|
||
fn from_str(value: &str) -> Result<Self, Self::Err> { | ||
let mut value_iter = value.split("+"); | ||
let n = match value_iter.next() { | ||
Some(val) => val, | ||
None => { | ||
return Err(format!("no value for the number of fingers")); | ||
}, | ||
}; | ||
let fingers = match u32::from_str(n) { | ||
Ok(a) => a, | ||
Err(_) => { | ||
return Err(format!("could not parse number of fingers")); | ||
}, | ||
}; | ||
|
||
let n2 = match value_iter.next() { | ||
Some(val) => val, | ||
None => { | ||
return Err(format!("could not parse direction")); | ||
}, | ||
}; | ||
|
||
let direction = match Direction::from_str(n2) { | ||
Ok(dir) => dir, | ||
Err(e) => { | ||
return Err(e); | ||
}, | ||
}; | ||
|
||
if let Some(n3) = value_iter.next() { | ||
return Err(format!("Extra data {} not expected", n3)); | ||
} | ||
|
||
return Ok(Self { | ||
fingers, | ||
direction, | ||
description: None, | ||
}); | ||
} | ||
} | ||
|
||
|
||
#[cfg(test)] | ||
mod tests { | ||
use crate::shortcuts::gesture::{AbsoluteDirection, Direction, RelativeDirection}; | ||
|
||
use super::Gesture; | ||
use std::str::FromStr; | ||
|
||
#[test] | ||
fn binding_from_str() { | ||
assert_eq!( | ||
Gesture::from_str("3+RelativeLeft"), | ||
Ok(Gesture::new( | ||
3 as u32, | ||
Direction::Relative(RelativeDirection::RelativeLeft) | ||
)) | ||
); | ||
|
||
assert_eq!( | ||
Gesture::from_str("5+AbsoluteUp"), | ||
Ok(Gesture::new( | ||
5 as u32, | ||
Direction::Absolute(AbsoluteDirection::AbsoluteUp) | ||
)) | ||
); | ||
|
||
assert_ne!( | ||
Gesture::from_str("4+AbsoluteLeft+More+Info"), | ||
Ok(Gesture::new( | ||
4 as u32, | ||
Direction::Absolute(AbsoluteDirection::AbsoluteLeft) | ||
)) | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters