Skip to content

Commit

Permalink
Delay auto-save until exiting insert mode
Browse files Browse the repository at this point in the history
Saving while in insert mode causes issues with the modification
indicator and this is very easy to reproduce with the current state of
the auto-save hook. We can tweak the hook slightly to await the mode
switch out of insert mode to perform the save.

The debounce is preserved: if you save and then immediately exit insert
mode the debounce will be respected. If the debounce lapses while you
are in insert mode, the save occurs as you switch out of insert mode
immediately.
  • Loading branch information
the-mikedavis committed Jun 27, 2024
1 parent b4811f7 commit 8d8cbb9
Show file tree
Hide file tree
Showing 2 changed files with 61 additions and 10 deletions.
63 changes: 54 additions & 9 deletions helix-term/src/handlers/auto_save.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,36 +4,68 @@ use anyhow::Ok;
use arc_swap::access::Access;

use helix_event::{register_hook, send_blocking};
use helix_view::{events::DocumentDidChange, handlers::Handlers, Editor};
use helix_view::{
document::Mode,
events::DocumentDidChange,
handlers::{AutoSaveEvent, Handlers},
Editor,
};
use tokio::time::Instant;

use crate::{
commands, compositor,
events::OnModeSwitch,
job::{self, Jobs},
};

#[derive(Debug)]
pub(super) struct AutoSaveHandler;
pub(super) struct AutoSaveHandler {
save_pending: bool,
}

impl AutoSaveHandler {
pub fn new() -> AutoSaveHandler {
AutoSaveHandler
AutoSaveHandler {
save_pending: false,
}
}
}

impl helix_event::AsyncHook for AutoSaveHandler {
type Event = u64;
type Event = AutoSaveEvent;

fn handle_event(
&mut self,
timeout: Self::Event,
_: Option<tokio::time::Instant>,
event: Self::Event,
existing_debounce: Option<tokio::time::Instant>,
) -> Option<Instant> {
Some(Instant::now() + Duration::from_millis(timeout))
match event {
Self::Event::DocumentChanged { debounce } => {
self.save_pending = true;
Some(Instant::now() + Duration::from_millis(debounce))
}
Self::Event::LeftInsertMode => {
if existing_debounce.is_some() {
// If the change happened more recently than the debounce, let the
// debounce run down before saving.
existing_debounce
} else {
// Otherwise if there is a save pending, save immediately.
if self.save_pending {
self.finish_debounce();
}
None
}
}
}
}

fn finish_debounce(&mut self) {
job::dispatch_blocking(move |editor, _| request_auto_save(editor))
job::dispatch_blocking(move |editor, _| {
if editor.mode() != Mode::Insert {
request_auto_save(editor)
}
})
}
}

Expand All @@ -54,7 +86,20 @@ pub(super) fn register_hooks(handlers: &Handlers) {
register_hook!(move |event: &mut DocumentDidChange<'_>| {
let config = event.doc.config.load();
if config.auto_save.after_delay.enable {
send_blocking(&tx, config.auto_save.after_delay.timeout);
send_blocking(
&tx,
AutoSaveEvent::DocumentChanged {
debounce: config.auto_save.after_delay.timeout,
},
);
}
Ok(())
});

let tx = handlers.auto_save.clone();
register_hook!(move |event: &mut OnModeSwitch<'_, '_>| {
if event.cx.editor.config().auto_save.after_delay.enable && event.old_mode == Mode::Insert {
send_blocking(&tx, AutoSaveEvent::LeftInsertMode);
}
Ok(())
});
Expand Down
8 changes: 7 additions & 1 deletion helix-view/src/handlers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,17 @@ use crate::{DocumentId, Editor, ViewId};
pub mod dap;
pub mod lsp;

#[derive(Debug)]
pub enum AutoSaveEvent {
DocumentChanged { debounce: u64 },
LeftInsertMode,
}

pub struct Handlers {
// only public because most of the actual implementation is in helix-term right now :/
pub completions: Sender<lsp::CompletionEvent>,
pub signature_hints: Sender<lsp::SignatureHelpEvent>,
pub auto_save: Sender<u64>,
pub auto_save: Sender<AutoSaveEvent>,
}

impl Handlers {
Expand Down

0 comments on commit 8d8cbb9

Please sign in to comment.