
## Before https://github.com/user-attachments/assets/affb8407-a20e-4258-a8f7-b271da5d7b77 ## After https://github.com/user-attachments/assets/654fe7b9-330a-40c8-8885-72e69fa85b0a Release Notes: - Hide blinking cursor when window is deactivated ([4710](https://github.com/zed-industries/zed/issues/4710))
108 lines
3 KiB
Rust
108 lines
3 KiB
Rust
use crate::EditorSettings;
|
|
use gpui::ModelContext;
|
|
use settings::Settings;
|
|
use settings::SettingsStore;
|
|
use smol::Timer;
|
|
use std::time::Duration;
|
|
|
|
pub struct BlinkManager {
|
|
blink_interval: Duration,
|
|
|
|
blink_epoch: usize,
|
|
blinking_paused: bool,
|
|
visible: bool,
|
|
enabled: bool,
|
|
}
|
|
|
|
impl BlinkManager {
|
|
pub fn new(blink_interval: Duration, cx: &mut ModelContext<Self>) -> Self {
|
|
// Make sure we blink the cursors if the setting is re-enabled
|
|
cx.observe_global::<SettingsStore>(move |this, cx| {
|
|
this.blink_cursors(this.blink_epoch, cx)
|
|
})
|
|
.detach();
|
|
|
|
Self {
|
|
blink_interval,
|
|
|
|
blink_epoch: 0,
|
|
blinking_paused: false,
|
|
visible: true,
|
|
enabled: false,
|
|
}
|
|
}
|
|
|
|
fn next_blink_epoch(&mut self) -> usize {
|
|
self.blink_epoch += 1;
|
|
self.blink_epoch
|
|
}
|
|
|
|
pub fn pause_blinking(&mut self, cx: &mut ModelContext<Self>) {
|
|
self.show_cursor(cx);
|
|
|
|
let epoch = self.next_blink_epoch();
|
|
let interval = self.blink_interval;
|
|
cx.spawn(|this, mut cx| async move {
|
|
Timer::after(interval).await;
|
|
this.update(&mut cx, |this, cx| this.resume_cursor_blinking(epoch, cx))
|
|
})
|
|
.detach();
|
|
}
|
|
|
|
fn resume_cursor_blinking(&mut self, epoch: usize, cx: &mut ModelContext<Self>) {
|
|
if epoch == self.blink_epoch {
|
|
self.blinking_paused = false;
|
|
self.blink_cursors(epoch, cx);
|
|
}
|
|
}
|
|
|
|
fn blink_cursors(&mut self, epoch: usize, cx: &mut ModelContext<Self>) {
|
|
if EditorSettings::get_global(cx).cursor_blink {
|
|
if epoch == self.blink_epoch && self.enabled && !self.blinking_paused {
|
|
self.visible = !self.visible;
|
|
cx.notify();
|
|
|
|
let epoch = self.next_blink_epoch();
|
|
let interval = self.blink_interval;
|
|
cx.spawn(|this, mut cx| async move {
|
|
Timer::after(interval).await;
|
|
if let Some(this) = this.upgrade() {
|
|
this.update(&mut cx, |this, cx| this.blink_cursors(epoch, cx))
|
|
.ok();
|
|
}
|
|
})
|
|
.detach();
|
|
}
|
|
} else {
|
|
self.show_cursor(cx);
|
|
}
|
|
}
|
|
|
|
pub fn show_cursor(&mut self, cx: &mut ModelContext<'_, BlinkManager>) {
|
|
if !self.visible {
|
|
self.visible = true;
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
pub fn enable(&mut self, cx: &mut ModelContext<Self>) {
|
|
if self.enabled {
|
|
return;
|
|
}
|
|
|
|
self.enabled = true;
|
|
// Set cursors as invisible and start blinking: this causes cursors
|
|
// to be visible during the next render.
|
|
self.visible = false;
|
|
self.blink_cursors(self.blink_epoch, cx);
|
|
}
|
|
|
|
pub fn disable(&mut self, _cx: &mut ModelContext<Self>) {
|
|
self.visible = false;
|
|
self.enabled = false;
|
|
}
|
|
|
|
pub fn visible(&self) -> bool {
|
|
self.visible
|
|
}
|
|
}
|