Bound the search range for the keybindings by the highest handler path

This commit is contained in:
Mikayla Maki 2023-03-10 12:12:32 -08:00
parent 9dc608dc4b
commit 00a38e4c3b
2 changed files with 46 additions and 21 deletions

View file

@ -485,7 +485,9 @@ pub struct MutableAppContext {
cx: AppContext, cx: AppContext,
action_deserializers: HashMap<&'static str, (TypeId, DeserializeActionCallback)>, action_deserializers: HashMap<&'static str, (TypeId, DeserializeActionCallback)>,
capture_actions: HashMap<TypeId, HashMap<TypeId, Vec<Box<ActionCallback>>>>, capture_actions: HashMap<TypeId, HashMap<TypeId, Vec<Box<ActionCallback>>>>,
// Entity Types -> { Action Types -> Action Handlers }
actions: HashMap<TypeId, HashMap<TypeId, Vec<Box<ActionCallback>>>>, actions: HashMap<TypeId, HashMap<TypeId, Vec<Box<ActionCallback>>>>,
// Action Types -> Action Handlers
global_actions: HashMap<TypeId, Box<GlobalActionCallback>>, global_actions: HashMap<TypeId, Box<GlobalActionCallback>>,
keystroke_matcher: KeymapMatcher, keystroke_matcher: KeymapMatcher,
next_entity_id: usize, next_entity_id: usize,
@ -1239,21 +1241,35 @@ impl MutableAppContext {
action: &dyn Action, action: &dyn Action,
) -> Option<SmallVec<[Keystroke; 2]>> { ) -> Option<SmallVec<[Keystroke; 2]>> {
let mut contexts = Vec::new(); let mut contexts = Vec::new();
for view_id in self.ancestors(window_id, view_id) { let mut highest_handler = None;
for (i, view_id) in self.ancestors(window_id, view_id).enumerate() {
if let Some(view) = self.views.get(&(window_id, view_id)) { if let Some(view) = self.views.get(&(window_id, view_id)) {
if let Some(actions) = self.actions.get(&view.as_any().type_id()) {
if actions.contains_key(&action.as_any().type_id()) {
highest_handler = Some(i);
}
}
contexts.push(view.keymap_context(self)); contexts.push(view.keymap_context(self));
} }
} }
if self.global_actions.contains_key(&action.as_any().type_id()) {
highest_handler = Some(contexts.len())
}
self.keystroke_matcher self.keystroke_matcher
.bindings_for_action_type(action.as_any().type_id()) .bindings_for_action_type(action.as_any().type_id())
.find_map(|b| { .find_map(|b| {
if b.match_dispatch_path_context(&contexts) { highest_handler
.map(|highest_handler| {
if (0..=highest_handler).any(|depth| b.match_context(&contexts[depth..])) {
Some(b.keystrokes().into()) Some(b.keystrokes().into())
} else { } else {
None None
} }
}) })
.flatten()
})
} }
pub fn available_actions( pub fn available_actions(
@ -1261,29 +1277,47 @@ impl MutableAppContext {
window_id: usize, window_id: usize,
view_id: usize, view_id: usize,
) -> impl Iterator<Item = (&'static str, Box<dyn Action>, SmallVec<[&Binding; 1]>)> { ) -> impl Iterator<Item = (&'static str, Box<dyn Action>, SmallVec<[&Binding; 1]>)> {
let mut action_types: HashSet<_> = self.global_actions.keys().copied().collect();
let mut contexts = Vec::new(); let mut contexts = Vec::new();
for view_id in self.ancestors(window_id, view_id) { let mut handler_depth_by_action_type = HashMap::<TypeId, Option<usize>>::default();
for (i, view_id) in self.ancestors(window_id, view_id).enumerate() {
if let Some(view) = self.views.get(&(window_id, view_id)) { if let Some(view) = self.views.get(&(window_id, view_id)) {
contexts.push(view.keymap_context(self)); contexts.push(view.keymap_context(self));
let view_type = view.as_any().type_id(); let view_type = view.as_any().type_id();
if let Some(actions) = self.actions.get(&view_type) { if let Some(actions) = self.actions.get(&view_type) {
action_types.extend(actions.keys().copied()); handler_depth_by_action_type.extend(
actions
.keys()
.copied()
.map(|action_type| (action_type, Some(i))),
);
} }
} }
} }
handler_depth_by_action_type.extend(
self.global_actions
.keys()
.copied()
.map(|action_type| (action_type, None)),
);
self.action_deserializers self.action_deserializers
.iter() .iter()
.filter_map(move |(name, (type_id, deserialize))| { .filter_map(move |(name, (type_id, deserialize))| {
if action_types.contains(type_id) { if let Some(action_depth) = handler_depth_by_action_type.get(type_id) {
Some(( Some((
*name, *name,
deserialize("{}").ok()?, deserialize("{}").ok()?,
self.keystroke_matcher self.keystroke_matcher
.bindings_for_action_type(*type_id) .bindings_for_action_type(*type_id)
.filter(|b| b.match_dispatch_path_context(&contexts)) .filter(|b| {
if let Some(action_depth) = *action_depth {
(0..=action_depth)
.any(|depth| b.match_context(&contexts[depth..]))
} else {
true
}
})
.collect(), .collect(),
)) ))
} else { } else {

View file

@ -42,15 +42,6 @@ impl Binding {
.unwrap_or(true) .unwrap_or(true)
} }
pub fn match_dispatch_path_context(&self, contexts: &[KeymapContext]) -> bool {
for i in 0..contexts.len() {
if self.match_context(&contexts[i..]) {
return true;
}
}
false
}
pub fn match_keys_and_context( pub fn match_keys_and_context(
&self, &self,
pending_keystrokes: &Vec<Keystroke>, pending_keystrokes: &Vec<Keystroke>,