mirror of
https://github.com/helix-editor/helix.git
synced 2025-04-05 11:57:43 +03:00
Autosave all when the terminal loses focus (#3178)
* Autosave all when the terminal loses focus * Correct comment on focus config Co-authored-by: Blaž Hrastnik <blaz@mxxn.io> * Need a block_try_flush_writes in all quit_all paths Co-authored-by: Blaž Hrastnik <blaz@mxxn.io>
This commit is contained in:
parent
4ff5feeb0c
commit
7e29ee6dae
5 changed files with 47 additions and 49 deletions
|
@ -49,6 +49,7 @@ on unix operating systems.
|
||||||
| `gutters` | Gutters to display: Available are `diagnostics` and `line-numbers` and `spacer`, note that `diagnostics` also includes other features like breakpoints, 1-width padding will be inserted if gutters is non-empty | `["diagnostics", "line-numbers"]` |
|
| `gutters` | Gutters to display: Available are `diagnostics` and `line-numbers` and `spacer`, note that `diagnostics` also includes other features like breakpoints, 1-width padding will be inserted if gutters is non-empty | `["diagnostics", "line-numbers"]` |
|
||||||
| `auto-completion` | Enable automatic pop up of auto-completion. | `true` |
|
| `auto-completion` | Enable automatic pop up of auto-completion. | `true` |
|
||||||
| `auto-format` | Enable automatic formatting on save. | `true` |
|
| `auto-format` | Enable automatic formatting on save. | `true` |
|
||||||
|
| `auto-save` | Enable automatic saving on focus moving away from Helix. Requires [focus event support](https://github.com/helix-editor/helix/wiki/Terminal-Support) from your terminal. | `false` |
|
||||||
| `idle-timeout` | Time in milliseconds since last keypress before idle timers trigger. Used for autocompletion, set to 0 for instant. | `400` |
|
| `idle-timeout` | Time in milliseconds since last keypress before idle timers trigger. Used for autocompletion, set to 0 for instant. | `400` |
|
||||||
| `completion-trigger-len` | The min-length of word under cursor to trigger autocompletion | `2` |
|
| `completion-trigger-len` | The min-length of word under cursor to trigger autocompletion | `2` |
|
||||||
| `auto-info` | Whether to display infoboxes | `true` |
|
| `auto-info` | Whether to display infoboxes | `true` |
|
||||||
|
|
|
@ -37,8 +37,8 @@ use anyhow::{Context, Error};
|
||||||
|
|
||||||
use crossterm::{
|
use crossterm::{
|
||||||
event::{
|
event::{
|
||||||
DisableBracketedPaste, DisableMouseCapture, EnableBracketedPaste, EnableMouseCapture,
|
DisableBracketedPaste, DisableFocusChange, DisableMouseCapture, EnableBracketedPaste,
|
||||||
Event as CrosstermEvent,
|
EnableFocusChange, EnableMouseCapture, Event as CrosstermEvent,
|
||||||
},
|
},
|
||||||
execute, terminal,
|
execute, terminal,
|
||||||
tty::IsTty,
|
tty::IsTty,
|
||||||
|
@ -102,6 +102,7 @@ fn restore_term() -> Result<(), Error> {
|
||||||
execute!(
|
execute!(
|
||||||
stdout,
|
stdout,
|
||||||
DisableBracketedPaste,
|
DisableBracketedPaste,
|
||||||
|
DisableFocusChange,
|
||||||
terminal::LeaveAlternateScreen
|
terminal::LeaveAlternateScreen
|
||||||
)?;
|
)?;
|
||||||
terminal::disable_raw_mode()?;
|
terminal::disable_raw_mode()?;
|
||||||
|
@ -925,7 +926,12 @@ impl Application {
|
||||||
async fn claim_term(&mut self) -> Result<(), Error> {
|
async fn claim_term(&mut self) -> Result<(), Error> {
|
||||||
terminal::enable_raw_mode()?;
|
terminal::enable_raw_mode()?;
|
||||||
let mut stdout = stdout();
|
let mut stdout = stdout();
|
||||||
execute!(stdout, terminal::EnterAlternateScreen, EnableBracketedPaste)?;
|
execute!(
|
||||||
|
stdout,
|
||||||
|
terminal::EnterAlternateScreen,
|
||||||
|
EnableBracketedPaste,
|
||||||
|
EnableFocusChange
|
||||||
|
)?;
|
||||||
execute!(stdout, terminal::Clear(terminal::ClearType::All))?;
|
execute!(stdout, terminal::Clear(terminal::ClearType::All))?;
|
||||||
if self.config.load().editor.mouse {
|
if self.config.load().editor.mouse {
|
||||||
execute!(stdout, EnableMouseCapture)?;
|
execute!(stdout, EnableMouseCapture)?;
|
||||||
|
|
|
@ -559,17 +559,11 @@ pub(super) fn buffers_remaining_impl(editor: &mut Editor) -> anyhow::Result<()>
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_all_impl(
|
pub fn write_all_impl(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
_args: &[Cow<str>],
|
|
||||||
event: PromptEvent,
|
|
||||||
quit: bool,
|
|
||||||
force: bool,
|
force: bool,
|
||||||
|
write_scratch: bool,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut errors: Vec<&'static str> = Vec::new();
|
let mut errors: Vec<&'static str> = Vec::new();
|
||||||
let auto_format = cx.editor.config().auto_format;
|
let auto_format = cx.editor.config().auto_format;
|
||||||
let jobs = &mut cx.jobs;
|
let jobs = &mut cx.jobs;
|
||||||
|
@ -580,12 +574,13 @@ fn write_all_impl(
|
||||||
.documents
|
.documents
|
||||||
.values()
|
.values()
|
||||||
.filter_map(|doc| {
|
.filter_map(|doc| {
|
||||||
if doc.path().is_none() {
|
if !doc.is_modified() {
|
||||||
errors.push("cannot write a buffer without a filename\n");
|
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
if doc.path().is_none() {
|
||||||
if !doc.is_modified() {
|
if write_scratch {
|
||||||
|
errors.push("cannot write a buffer without a filename\n");
|
||||||
|
}
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -611,20 +606,6 @@ fn write_all_impl(
|
||||||
cx.editor.save::<PathBuf>(id, None, force)?;
|
cx.editor.save::<PathBuf>(id, None, force)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
if quit {
|
|
||||||
cx.block_try_flush_writes()?;
|
|
||||||
|
|
||||||
if !force {
|
|
||||||
buffers_remaining_impl(cx.editor)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// close all views
|
|
||||||
let views: Vec<_> = cx.editor.tree.views().map(|(view, _)| view.id).collect();
|
|
||||||
for view_id in views {
|
|
||||||
cx.editor.close(view_id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !errors.is_empty() && !force {
|
if !errors.is_empty() && !force {
|
||||||
bail!("{:?}", errors);
|
bail!("{:?}", errors);
|
||||||
}
|
}
|
||||||
|
@ -634,49 +615,50 @@ fn write_all_impl(
|
||||||
|
|
||||||
fn write_all(
|
fn write_all(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
args: &[Cow<str>],
|
_args: &[Cow<str>],
|
||||||
event: PromptEvent,
|
event: PromptEvent,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
if event != PromptEvent::Validate {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
write_all_impl(cx, args, event, false, false)
|
write_all_impl(cx, false, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_all_quit(
|
fn write_all_quit(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
args: &[Cow<str>],
|
_args: &[Cow<str>],
|
||||||
event: PromptEvent,
|
event: PromptEvent,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
if event != PromptEvent::Validate {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
write_all_impl(cx, false, true)?;
|
||||||
write_all_impl(cx, args, event, true, false)
|
quit_all_impl(cx, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn force_write_all_quit(
|
fn force_write_all_quit(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
args: &[Cow<str>],
|
_args: &[Cow<str>],
|
||||||
event: PromptEvent,
|
event: PromptEvent,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
if event != PromptEvent::Validate {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
let _ = write_all_impl(cx, true, true);
|
||||||
write_all_impl(cx, args, event, true, true)
|
quit_all_impl(cx, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn quit_all_impl(editor: &mut Editor, force: bool) -> anyhow::Result<()> {
|
fn quit_all_impl(cx: &mut compositor::Context, force: bool) -> anyhow::Result<()> {
|
||||||
|
cx.block_try_flush_writes()?;
|
||||||
if !force {
|
if !force {
|
||||||
buffers_remaining_impl(editor)?;
|
buffers_remaining_impl(cx.editor)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
// close all views
|
// close all views
|
||||||
let views: Vec<_> = editor.tree.views().map(|(view, _)| view.id).collect();
|
let views: Vec<_> = cx.editor.tree.views().map(|(view, _)| view.id).collect();
|
||||||
for view_id in views {
|
for view_id in views {
|
||||||
editor.close(view_id);
|
cx.editor.close(view_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -691,8 +673,7 @@ fn quit_all(
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
cx.block_try_flush_writes()?;
|
quit_all_impl(cx, false)
|
||||||
quit_all_impl(cx.editor, false)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn force_quit_all(
|
fn force_quit_all(
|
||||||
|
@ -704,7 +685,7 @@ fn force_quit_all(
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
quit_all_impl(cx.editor, true)
|
quit_all_impl(cx, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn cquit(
|
fn cquit(
|
||||||
|
@ -722,8 +703,7 @@ fn cquit(
|
||||||
.unwrap_or(1);
|
.unwrap_or(1);
|
||||||
|
|
||||||
cx.editor.exit_code = exit_code;
|
cx.editor.exit_code = exit_code;
|
||||||
cx.block_try_flush_writes()?;
|
quit_all_impl(cx, false)
|
||||||
quit_all_impl(cx.editor, false)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn force_cquit(
|
fn force_cquit(
|
||||||
|
@ -741,7 +721,7 @@ fn force_cquit(
|
||||||
.unwrap_or(1);
|
.unwrap_or(1);
|
||||||
cx.editor.exit_code = exit_code;
|
cx.editor.exit_code = exit_code;
|
||||||
|
|
||||||
quit_all_impl(cx.editor, true)
|
quit_all_impl(cx, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn theme(
|
fn theme(
|
||||||
|
@ -2141,7 +2121,7 @@ pub static TYPABLE_COMMAND_MAP: Lazy<HashMap<&'static str, &'static TypableComma
|
||||||
.collect()
|
.collect()
|
||||||
});
|
});
|
||||||
|
|
||||||
pub fn command_mode(cx: &mut Context) {
|
pub(super) fn command_mode(cx: &mut Context) {
|
||||||
let mut prompt = Prompt::new(
|
let mut prompt = Prompt::new(
|
||||||
":".into(),
|
":".into(),
|
||||||
Some(':'),
|
Some(':'),
|
||||||
|
|
|
@ -1426,7 +1426,15 @@ impl Component for EditorView {
|
||||||
|
|
||||||
Event::Mouse(event) => self.handle_mouse_event(event, &mut cx),
|
Event::Mouse(event) => self.handle_mouse_event(event, &mut cx),
|
||||||
Event::IdleTimeout => self.handle_idle_timeout(&mut cx),
|
Event::IdleTimeout => self.handle_idle_timeout(&mut cx),
|
||||||
Event::FocusGained | Event::FocusLost => EventResult::Ignored(None),
|
Event::FocusGained => EventResult::Ignored(None),
|
||||||
|
Event::FocusLost => {
|
||||||
|
if context.editor.config().auto_save {
|
||||||
|
if let Err(e) = commands::typed::write_all_impl(context, false, false) {
|
||||||
|
context.editor.set_error(format!("{}", e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EventResult::Consumed(None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -140,6 +140,8 @@ pub struct Config {
|
||||||
pub auto_completion: bool,
|
pub auto_completion: bool,
|
||||||
/// Automatic formatting on save. Defaults to true.
|
/// Automatic formatting on save. Defaults to true.
|
||||||
pub auto_format: bool,
|
pub auto_format: bool,
|
||||||
|
/// Automatic save on focus lost. Defaults to false.
|
||||||
|
pub auto_save: bool,
|
||||||
/// Time in milliseconds since last keypress before idle timers trigger.
|
/// Time in milliseconds since last keypress before idle timers trigger.
|
||||||
/// Used for autocompletion, set to 0 for instant. Defaults to 400ms.
|
/// Used for autocompletion, set to 0 for instant. Defaults to 400ms.
|
||||||
#[serde(
|
#[serde(
|
||||||
|
@ -592,6 +594,7 @@ impl Default for Config {
|
||||||
auto_pairs: AutoPairConfig::default(),
|
auto_pairs: AutoPairConfig::default(),
|
||||||
auto_completion: true,
|
auto_completion: true,
|
||||||
auto_format: true,
|
auto_format: true,
|
||||||
|
auto_save: false,
|
||||||
idle_timeout: Duration::from_millis(400),
|
idle_timeout: Duration::from_millis(400),
|
||||||
completion_trigger_len: 2,
|
completion_trigger_len: 2,
|
||||||
auto_info: true,
|
auto_info: true,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue