132 lines
4.2 KiB
Rust
132 lines
4.2 KiB
Rust
use gpui::{Context, Element, Entity, Render, Subscription, WeakEntity, Window, div};
|
|
use ui::text_for_keystrokes;
|
|
use workspace::{StatusItemView, item::ItemHandle, ui::prelude::*};
|
|
|
|
use crate::{Vim, VimEvent, VimGlobals};
|
|
|
|
/// The ModeIndicator displays the current mode in the status bar.
|
|
pub struct ModeIndicator {
|
|
vim: Option<WeakEntity<Vim>>,
|
|
pending_keys: Option<String>,
|
|
vim_subscription: Option<Subscription>,
|
|
}
|
|
|
|
impl ModeIndicator {
|
|
/// Construct a new mode indicator in this window.
|
|
pub fn new(window: &mut Window, cx: &mut Context<Self>) -> Self {
|
|
cx.observe_pending_input(window, |this: &mut Self, window, cx| {
|
|
this.update_pending_keys(window, cx);
|
|
cx.notify();
|
|
})
|
|
.detach();
|
|
|
|
let handle = cx.entity().clone();
|
|
let window_handle = window.window_handle();
|
|
cx.observe_new::<Vim>(move |_, window, cx| {
|
|
let Some(window) = window else {
|
|
return;
|
|
};
|
|
if window.window_handle() != window_handle {
|
|
return;
|
|
}
|
|
let vim = cx.entity().clone();
|
|
handle.update(cx, |_, cx| {
|
|
cx.subscribe(&vim, |mode_indicator, vim, event, cx| match event {
|
|
VimEvent::Focused => {
|
|
mode_indicator.vim_subscription =
|
|
Some(cx.observe(&vim, |_, _, cx| cx.notify()));
|
|
mode_indicator.vim = Some(vim.downgrade());
|
|
}
|
|
})
|
|
.detach()
|
|
})
|
|
})
|
|
.detach();
|
|
|
|
Self {
|
|
vim: None,
|
|
pending_keys: None,
|
|
vim_subscription: None,
|
|
}
|
|
}
|
|
|
|
fn update_pending_keys(&mut self, window: &mut Window, cx: &App) {
|
|
self.pending_keys = window
|
|
.pending_input_keystrokes()
|
|
.map(|keystrokes| text_for_keystrokes(keystrokes, cx));
|
|
}
|
|
|
|
fn vim(&self) -> Option<Entity<Vim>> {
|
|
self.vim.as_ref().and_then(|vim| vim.upgrade())
|
|
}
|
|
|
|
fn current_operators_description(&self, vim: Entity<Vim>, cx: &mut Context<Self>) -> String {
|
|
let recording = Vim::globals(cx)
|
|
.recording_register
|
|
.map(|reg| format!("recording @{reg} "))
|
|
.into_iter();
|
|
|
|
let vim = vim.read(cx);
|
|
recording
|
|
.chain(
|
|
cx.global::<VimGlobals>()
|
|
.pre_count
|
|
.map(|count| format!("{}", count)),
|
|
)
|
|
.chain(vim.selected_register.map(|reg| format!("\"{reg}")))
|
|
.chain(
|
|
vim.operator_stack
|
|
.iter()
|
|
.map(|item| item.status().to_string()),
|
|
)
|
|
.chain(
|
|
cx.global::<VimGlobals>()
|
|
.post_count
|
|
.map(|count| format!("{}", count)),
|
|
)
|
|
.collect::<Vec<_>>()
|
|
.join("")
|
|
}
|
|
}
|
|
|
|
impl Render for ModeIndicator {
|
|
fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
|
let vim = self.vim();
|
|
let Some(vim) = vim else {
|
|
return div().into_any();
|
|
};
|
|
|
|
let vim_readable = vim.read(cx);
|
|
let label = if let Some(label) = vim_readable.status_label.clone() {
|
|
label
|
|
} else {
|
|
let mode = if vim_readable.temp_mode {
|
|
format!("(insert) {}", vim_readable.mode)
|
|
} else {
|
|
vim_readable.mode.to_string()
|
|
};
|
|
|
|
let current_operators_description = self.current_operators_description(vim.clone(), cx);
|
|
let pending = self
|
|
.pending_keys
|
|
.as_ref()
|
|
.unwrap_or(¤t_operators_description);
|
|
format!("{} -- {} --", pending, mode).into()
|
|
};
|
|
|
|
Label::new(label)
|
|
.size(LabelSize::Small)
|
|
.line_height_style(LineHeightStyle::UiLabel)
|
|
.into_any_element()
|
|
}
|
|
}
|
|
|
|
impl StatusItemView for ModeIndicator {
|
|
fn set_active_pane_item(
|
|
&mut self,
|
|
_active_pane_item: Option<&dyn ItemHandle>,
|
|
_window: &mut Window,
|
|
_cx: &mut Context<Self>,
|
|
) {
|
|
}
|
|
}
|