mirror of
https://github.com/zed-industries/zed.git
synced 2025-01-18 08:02:27 +00:00
bb9b36dccd
Co-authored-by: Nathan Sobo <nathan@zed.dev>
498 lines
14 KiB
Rust
498 lines
14 KiB
Rust
use anyhow::anyhow;
|
|
use std::{
|
|
any::Any,
|
|
collections::{HashMap, HashSet},
|
|
fmt::Debug,
|
|
};
|
|
use tree_sitter::{Language, Node, Parser};
|
|
|
|
use crate::{Action, AnyAction};
|
|
|
|
extern "C" {
|
|
fn tree_sitter_context_predicate() -> Language;
|
|
}
|
|
|
|
pub struct Matcher {
|
|
pending: HashMap<usize, Pending>,
|
|
keymap: Keymap,
|
|
}
|
|
|
|
#[derive(Default)]
|
|
struct Pending {
|
|
keystrokes: Vec<Keystroke>,
|
|
context: Option<Context>,
|
|
}
|
|
|
|
#[derive(Default)]
|
|
pub struct Keymap(Vec<Binding>);
|
|
|
|
pub struct Binding {
|
|
keystrokes: Vec<Keystroke>,
|
|
action: Box<dyn AnyAction>,
|
|
context: Option<ContextPredicate>,
|
|
}
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
pub struct Keystroke {
|
|
pub ctrl: bool,
|
|
pub alt: bool,
|
|
pub shift: bool,
|
|
pub cmd: bool,
|
|
pub key: String,
|
|
}
|
|
|
|
#[derive(Clone, Debug, Default, Eq, PartialEq)]
|
|
pub struct Context {
|
|
pub set: HashSet<String>,
|
|
pub map: HashMap<String, String>,
|
|
}
|
|
|
|
#[derive(Debug, Eq, PartialEq)]
|
|
enum ContextPredicate {
|
|
Identifier(String),
|
|
Equal(String, String),
|
|
NotEqual(String, String),
|
|
Not(Box<ContextPredicate>),
|
|
And(Box<ContextPredicate>, Box<ContextPredicate>),
|
|
Or(Box<ContextPredicate>, Box<ContextPredicate>),
|
|
}
|
|
|
|
trait ActionArg {
|
|
fn boxed_clone(&self) -> Box<dyn Any>;
|
|
}
|
|
|
|
impl<T> ActionArg for T
|
|
where
|
|
T: 'static + Any + Clone,
|
|
{
|
|
fn boxed_clone(&self) -> Box<dyn Any> {
|
|
Box::new(self.clone())
|
|
}
|
|
}
|
|
|
|
pub enum MatchResult {
|
|
None,
|
|
Pending,
|
|
Action(Box<dyn AnyAction>),
|
|
}
|
|
|
|
impl Debug for MatchResult {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
MatchResult::None => f.debug_struct("MatchResult::None").finish(),
|
|
MatchResult::Pending => f.debug_struct("MatchResult::Pending").finish(),
|
|
MatchResult::Action(action) => f
|
|
.debug_tuple("MatchResult::Action")
|
|
.field(&action.name())
|
|
.finish(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Matcher {
|
|
pub fn new(keymap: Keymap) -> Self {
|
|
Self {
|
|
pending: HashMap::new(),
|
|
keymap,
|
|
}
|
|
}
|
|
|
|
pub fn set_keymap(&mut self, keymap: Keymap) {
|
|
self.pending.clear();
|
|
self.keymap = keymap;
|
|
}
|
|
|
|
pub fn add_bindings<T: IntoIterator<Item = Binding>>(&mut self, bindings: T) {
|
|
self.pending.clear();
|
|
self.keymap.add_bindings(bindings);
|
|
}
|
|
|
|
pub fn clear_pending(&mut self) {
|
|
self.pending.clear();
|
|
}
|
|
|
|
pub fn push_keystroke(
|
|
&mut self,
|
|
keystroke: Keystroke,
|
|
view_id: usize,
|
|
cx: &Context,
|
|
) -> MatchResult {
|
|
let pending = self.pending.entry(view_id).or_default();
|
|
|
|
if let Some(pending_ctx) = pending.context.as_ref() {
|
|
if pending_ctx != cx {
|
|
pending.keystrokes.clear();
|
|
}
|
|
}
|
|
|
|
pending.keystrokes.push(keystroke);
|
|
|
|
let mut retain_pending = false;
|
|
for binding in self.keymap.0.iter().rev() {
|
|
if binding.keystrokes.starts_with(&pending.keystrokes)
|
|
&& binding.context.as_ref().map(|c| c.eval(cx)).unwrap_or(true)
|
|
{
|
|
if binding.keystrokes.len() == pending.keystrokes.len() {
|
|
self.pending.remove(&view_id);
|
|
return MatchResult::Action(binding.action.boxed_clone());
|
|
} else {
|
|
retain_pending = true;
|
|
pending.context = Some(cx.clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
if retain_pending {
|
|
MatchResult::Pending
|
|
} else {
|
|
self.pending.remove(&view_id);
|
|
MatchResult::None
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Default for Matcher {
|
|
fn default() -> Self {
|
|
Self::new(Keymap::default())
|
|
}
|
|
}
|
|
|
|
impl Keymap {
|
|
pub fn new(bindings: Vec<Binding>) -> Self {
|
|
Self(bindings)
|
|
}
|
|
|
|
fn add_bindings<T: IntoIterator<Item = Binding>>(&mut self, bindings: T) {
|
|
self.0.extend(bindings.into_iter());
|
|
}
|
|
}
|
|
|
|
impl Binding {
|
|
pub fn new<A: Action>(keystrokes: &str, action: A, context: Option<&str>) -> Self {
|
|
let context = if let Some(context) = context {
|
|
Some(ContextPredicate::parse(context).unwrap())
|
|
} else {
|
|
None
|
|
};
|
|
|
|
Self {
|
|
keystrokes: keystrokes
|
|
.split_whitespace()
|
|
.map(|key| Keystroke::parse(key).unwrap())
|
|
.collect(),
|
|
action: Box::new(action),
|
|
context,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Keystroke {
|
|
pub fn parse(source: &str) -> anyhow::Result<Self> {
|
|
let mut ctrl = false;
|
|
let mut alt = false;
|
|
let mut shift = false;
|
|
let mut cmd = false;
|
|
let mut key = None;
|
|
|
|
let mut components = source.split("-").peekable();
|
|
while let Some(component) = components.next() {
|
|
match component {
|
|
"ctrl" => ctrl = true,
|
|
"alt" => alt = true,
|
|
"shift" => shift = true,
|
|
"cmd" => cmd = true,
|
|
_ => {
|
|
if let Some(component) = components.peek() {
|
|
if component.is_empty() && source.ends_with('-') {
|
|
key = Some(String::from("-"));
|
|
break;
|
|
} else {
|
|
return Err(anyhow!("Invalid keystroke `{}`", source));
|
|
}
|
|
} else {
|
|
key = Some(String::from(component));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(Keystroke {
|
|
ctrl,
|
|
alt,
|
|
shift,
|
|
cmd,
|
|
key: key.unwrap(),
|
|
})
|
|
}
|
|
|
|
pub fn modified(&self) -> bool {
|
|
self.ctrl || self.alt || self.shift || self.cmd
|
|
}
|
|
}
|
|
|
|
impl Context {
|
|
pub fn extend(&mut self, other: &Context) {
|
|
for v in &other.set {
|
|
self.set.insert(v.clone());
|
|
}
|
|
for (k, v) in &other.map {
|
|
self.map.insert(k.clone(), v.clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ContextPredicate {
|
|
fn parse(source: &str) -> anyhow::Result<Self> {
|
|
let mut parser = Parser::new();
|
|
let language = unsafe { tree_sitter_context_predicate() };
|
|
parser.set_language(language).unwrap();
|
|
let source = source.as_bytes();
|
|
let tree = parser.parse(source, None).unwrap();
|
|
Self::from_node(tree.root_node(), source)
|
|
}
|
|
|
|
fn from_node(node: Node, source: &[u8]) -> anyhow::Result<Self> {
|
|
let parse_error = "error parsing context predicate";
|
|
let kind = node.kind();
|
|
|
|
match kind {
|
|
"source" => Self::from_node(node.child(0).ok_or(anyhow!(parse_error))?, source),
|
|
"identifier" => Ok(Self::Identifier(node.utf8_text(source)?.into())),
|
|
"not" => {
|
|
let child = Self::from_node(
|
|
node.child_by_field_name("expression")
|
|
.ok_or(anyhow!(parse_error))?,
|
|
source,
|
|
)?;
|
|
Ok(Self::Not(Box::new(child)))
|
|
}
|
|
"and" | "or" => {
|
|
let left = Box::new(Self::from_node(
|
|
node.child_by_field_name("left")
|
|
.ok_or(anyhow!(parse_error))?,
|
|
source,
|
|
)?);
|
|
let right = Box::new(Self::from_node(
|
|
node.child_by_field_name("right")
|
|
.ok_or(anyhow!(parse_error))?,
|
|
source,
|
|
)?);
|
|
if kind == "and" {
|
|
Ok(Self::And(left, right))
|
|
} else {
|
|
Ok(Self::Or(left, right))
|
|
}
|
|
}
|
|
"equal" | "not_equal" => {
|
|
let left = node
|
|
.child_by_field_name("left")
|
|
.ok_or(anyhow!(parse_error))?
|
|
.utf8_text(source)?
|
|
.into();
|
|
let right = node
|
|
.child_by_field_name("right")
|
|
.ok_or(anyhow!(parse_error))?
|
|
.utf8_text(source)?
|
|
.into();
|
|
if kind == "equal" {
|
|
Ok(Self::Equal(left, right))
|
|
} else {
|
|
Ok(Self::NotEqual(left, right))
|
|
}
|
|
}
|
|
"parenthesized" => Self::from_node(
|
|
node.child_by_field_name("expression")
|
|
.ok_or(anyhow!(parse_error))?,
|
|
source,
|
|
),
|
|
_ => Err(anyhow!(parse_error)),
|
|
}
|
|
}
|
|
|
|
fn eval(&self, cx: &Context) -> bool {
|
|
match self {
|
|
Self::Identifier(name) => cx.set.contains(name.as_str()),
|
|
Self::Equal(left, right) => cx
|
|
.map
|
|
.get(left)
|
|
.map(|value| value == right)
|
|
.unwrap_or(false),
|
|
Self::NotEqual(left, right) => {
|
|
cx.map.get(left).map(|value| value != right).unwrap_or(true)
|
|
}
|
|
Self::Not(pred) => !pred.eval(cx),
|
|
Self::And(left, right) => left.eval(cx) && right.eval(cx),
|
|
Self::Or(left, right) => left.eval(cx) || right.eval(cx),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use crate::action;
|
|
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_keystroke_parsing() -> anyhow::Result<()> {
|
|
assert_eq!(
|
|
Keystroke::parse("ctrl-p")?,
|
|
Keystroke {
|
|
key: "p".into(),
|
|
ctrl: true,
|
|
alt: false,
|
|
shift: false,
|
|
cmd: false,
|
|
}
|
|
);
|
|
|
|
assert_eq!(
|
|
Keystroke::parse("alt-shift-down")?,
|
|
Keystroke {
|
|
key: "down".into(),
|
|
ctrl: false,
|
|
alt: true,
|
|
shift: true,
|
|
cmd: false,
|
|
}
|
|
);
|
|
|
|
assert_eq!(
|
|
Keystroke::parse("shift-cmd--")?,
|
|
Keystroke {
|
|
key: "-".into(),
|
|
ctrl: false,
|
|
alt: false,
|
|
shift: true,
|
|
cmd: true,
|
|
}
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn test_context_predicate_parsing() -> anyhow::Result<()> {
|
|
use ContextPredicate::*;
|
|
|
|
assert_eq!(
|
|
ContextPredicate::parse("a && (b == c || d != e)")?,
|
|
And(
|
|
Box::new(Identifier("a".into())),
|
|
Box::new(Or(
|
|
Box::new(Equal("b".into(), "c".into())),
|
|
Box::new(NotEqual("d".into(), "e".into())),
|
|
))
|
|
)
|
|
);
|
|
|
|
assert_eq!(
|
|
ContextPredicate::parse("!a")?,
|
|
Not(Box::new(Identifier("a".into())),)
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn test_context_predicate_eval() -> anyhow::Result<()> {
|
|
let predicate = ContextPredicate::parse("a && b || c == d")?;
|
|
|
|
let mut context = Context::default();
|
|
context.set.insert("a".into());
|
|
assert!(!predicate.eval(&context));
|
|
|
|
context.set.insert("b".into());
|
|
assert!(predicate.eval(&context));
|
|
|
|
context.set.remove("b");
|
|
context.map.insert("c".into(), "x".into());
|
|
assert!(!predicate.eval(&context));
|
|
|
|
context.map.insert("c".into(), "d".into());
|
|
assert!(predicate.eval(&context));
|
|
|
|
let predicate = ContextPredicate::parse("!a")?;
|
|
assert!(predicate.eval(&Context::default()));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn test_matcher() -> anyhow::Result<()> {
|
|
action!(A, &'static str);
|
|
action!(B);
|
|
action!(Ab);
|
|
|
|
impl PartialEq for A {
|
|
fn eq(&self, other: &Self) -> bool {
|
|
self.0 == other.0
|
|
}
|
|
}
|
|
impl Eq for A {}
|
|
impl Debug for A {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(f, "A({:?})", &self.0)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
struct ActionArg {
|
|
a: &'static str,
|
|
}
|
|
|
|
let keymap = Keymap(vec![
|
|
Binding::new("a", A("x"), Some("a")),
|
|
Binding::new("b", B, Some("a")),
|
|
Binding::new("a b", Ab, Some("a || b")),
|
|
]);
|
|
|
|
let mut ctx_a = Context::default();
|
|
ctx_a.set.insert("a".into());
|
|
|
|
let mut ctx_b = Context::default();
|
|
ctx_b.set.insert("b".into());
|
|
|
|
let mut matcher = Matcher::new(keymap);
|
|
|
|
// Basic match
|
|
assert_eq!(matcher.test_keystroke("a", 1, &ctx_a), Some(A("x")));
|
|
|
|
// Multi-keystroke match
|
|
assert_eq!(matcher.test_keystroke::<A>("a", 1, &ctx_b), None);
|
|
assert_eq!(matcher.test_keystroke("b", 1, &ctx_b), Some(Ab));
|
|
|
|
// Failed matches don't interfere with matching subsequent keys
|
|
assert_eq!(matcher.test_keystroke::<A>("x", 1, &ctx_a), None);
|
|
assert_eq!(matcher.test_keystroke("a", 1, &ctx_a), Some(A("x")));
|
|
|
|
// Pending keystrokes are cleared when the context changes
|
|
assert_eq!(matcher.test_keystroke::<A>("a", 1, &ctx_b), None);
|
|
assert_eq!(matcher.test_keystroke("b", 1, &ctx_a), Some(B));
|
|
|
|
let mut ctx_c = Context::default();
|
|
ctx_c.set.insert("c".into());
|
|
|
|
// Pending keystrokes are maintained per-view
|
|
assert_eq!(matcher.test_keystroke::<A>("a", 1, &ctx_b), None);
|
|
assert_eq!(matcher.test_keystroke::<A>("a", 2, &ctx_c), None);
|
|
assert_eq!(matcher.test_keystroke("b", 1, &ctx_b), Some(Ab));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
impl Matcher {
|
|
fn test_keystroke<A>(&mut self, keystroke: &str, view_id: usize, cx: &Context) -> Option<A>
|
|
where
|
|
A: Action + Debug + Eq,
|
|
{
|
|
if let MatchResult::Action(action) =
|
|
self.push_keystroke(Keystroke::parse(keystroke).unwrap(), view_id, cx)
|
|
{
|
|
Some(*action.boxed_clone_as_any().downcast().unwrap())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
}
|