Honor Autosave
setting in Editor
This commit is contained in:
parent
d589017a80
commit
885172f4dd
2 changed files with 75 additions and 19 deletions
|
@ -17,6 +17,7 @@ use collections::{BTreeMap, Bound, HashMap, HashSet, VecDeque};
|
||||||
pub use display_map::DisplayPoint;
|
pub use display_map::DisplayPoint;
|
||||||
use display_map::*;
|
use display_map::*;
|
||||||
pub use element::*;
|
pub use element::*;
|
||||||
|
use futures::{channel::oneshot, FutureExt};
|
||||||
use fuzzy::{StringMatch, StringMatchCandidate};
|
use fuzzy::{StringMatch, StringMatchCandidate};
|
||||||
use gpui::{
|
use gpui::{
|
||||||
actions,
|
actions,
|
||||||
|
@ -28,8 +29,8 @@ use gpui::{
|
||||||
impl_actions, impl_internal_actions,
|
impl_actions, impl_internal_actions,
|
||||||
platform::CursorStyle,
|
platform::CursorStyle,
|
||||||
text_layout, AppContext, AsyncAppContext, ClipboardItem, Element, ElementBox, Entity,
|
text_layout, AppContext, AsyncAppContext, ClipboardItem, Element, ElementBox, Entity,
|
||||||
ModelHandle, MutableAppContext, RenderContext, Task, View, ViewContext, ViewHandle,
|
ModelHandle, MutableAppContext, RenderContext, Subscription, Task, View, ViewContext,
|
||||||
WeakViewHandle,
|
ViewHandle, WeakViewHandle,
|
||||||
};
|
};
|
||||||
use hover_popover::{hide_hover, HoverState};
|
use hover_popover::{hide_hover, HoverState};
|
||||||
pub use language::{char_kind, CharKind};
|
pub use language::{char_kind, CharKind};
|
||||||
|
@ -48,7 +49,7 @@ use ordered_float::OrderedFloat;
|
||||||
use project::{LocationLink, Project, ProjectPath, ProjectTransaction};
|
use project::{LocationLink, Project, ProjectPath, ProjectTransaction};
|
||||||
use selections_collection::{resolve_multiple, MutableSelectionsCollection, SelectionsCollection};
|
use selections_collection::{resolve_multiple, MutableSelectionsCollection, SelectionsCollection};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use settings::Settings;
|
use settings::{Autosave, Settings};
|
||||||
use smallvec::SmallVec;
|
use smallvec::SmallVec;
|
||||||
use smol::Timer;
|
use smol::Timer;
|
||||||
use snippet::Snippet;
|
use snippet::Snippet;
|
||||||
|
@ -436,6 +437,9 @@ pub struct Editor {
|
||||||
leader_replica_id: Option<u16>,
|
leader_replica_id: Option<u16>,
|
||||||
hover_state: HoverState,
|
hover_state: HoverState,
|
||||||
link_go_to_definition_state: LinkGoToDefinitionState,
|
link_go_to_definition_state: LinkGoToDefinitionState,
|
||||||
|
pending_autosave: Option<Task<Option<()>>>,
|
||||||
|
cancel_pending_autosave: Option<oneshot::Sender<()>>,
|
||||||
|
_subscriptions: Vec<Subscription>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct EditorSnapshot {
|
pub struct EditorSnapshot {
|
||||||
|
@ -973,17 +977,13 @@ impl Editor {
|
||||||
cx,
|
cx,
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
cx.observe(&buffer, Self::on_buffer_changed).detach();
|
|
||||||
cx.subscribe(&buffer, Self::on_buffer_event).detach();
|
|
||||||
cx.observe(&display_map, Self::on_display_map_changed)
|
|
||||||
.detach();
|
|
||||||
|
|
||||||
let selections = SelectionsCollection::new(display_map.clone(), buffer.clone());
|
let selections = SelectionsCollection::new(display_map.clone(), buffer.clone());
|
||||||
|
|
||||||
let mut this = Self {
|
let mut this = Self {
|
||||||
handle: cx.weak_handle(),
|
handle: cx.weak_handle(),
|
||||||
buffer,
|
buffer: buffer.clone(),
|
||||||
display_map,
|
display_map: display_map.clone(),
|
||||||
selections,
|
selections,
|
||||||
columnar_selection_tail: None,
|
columnar_selection_tail: None,
|
||||||
add_selections_state: None,
|
add_selections_state: None,
|
||||||
|
@ -1026,6 +1026,14 @@ impl Editor {
|
||||||
leader_replica_id: None,
|
leader_replica_id: None,
|
||||||
hover_state: Default::default(),
|
hover_state: Default::default(),
|
||||||
link_go_to_definition_state: Default::default(),
|
link_go_to_definition_state: Default::default(),
|
||||||
|
pending_autosave: Default::default(),
|
||||||
|
cancel_pending_autosave: Default::default(),
|
||||||
|
_subscriptions: vec![
|
||||||
|
cx.observe(&buffer, Self::on_buffer_changed),
|
||||||
|
cx.subscribe(&buffer, Self::on_buffer_event),
|
||||||
|
cx.observe(&display_map, Self::on_display_map_changed),
|
||||||
|
cx.observe_window_activation(Self::on_window_activation_changed),
|
||||||
|
],
|
||||||
};
|
};
|
||||||
this.end_selection(cx);
|
this.end_selection(cx);
|
||||||
|
|
||||||
|
@ -2148,7 +2156,10 @@ impl Editor {
|
||||||
.iter()
|
.iter()
|
||||||
.zip(autoclose_pair.ranges.iter().map(|r| r.to_offset(&buffer)))
|
.zip(autoclose_pair.ranges.iter().map(|r| r.to_offset(&buffer)))
|
||||||
{
|
{
|
||||||
if selection.is_empty() && autoclose_range.is_empty() && selection.start == autoclose_range.start {
|
if selection.is_empty()
|
||||||
|
&& autoclose_range.is_empty()
|
||||||
|
&& selection.start == autoclose_range.start
|
||||||
|
{
|
||||||
new_selections.push(Selection {
|
new_selections.push(Selection {
|
||||||
id: selection.id,
|
id: selection.id,
|
||||||
start: selection.start - autoclose_pair.pair.start.len(),
|
start: selection.start - autoclose_pair.pair.start.len(),
|
||||||
|
@ -5570,6 +5581,33 @@ impl Editor {
|
||||||
self.refresh_active_diagnostics(cx);
|
self.refresh_active_diagnostics(cx);
|
||||||
self.refresh_code_actions(cx);
|
self.refresh_code_actions(cx);
|
||||||
cx.emit(Event::BufferEdited);
|
cx.emit(Event::BufferEdited);
|
||||||
|
if let Autosave::AfterDelay { milliseconds } = cx.global::<Settings>().autosave {
|
||||||
|
let pending_autosave =
|
||||||
|
self.pending_autosave.take().unwrap_or(Task::ready(None));
|
||||||
|
if let Some(cancel_pending_autosave) = self.cancel_pending_autosave.take() {
|
||||||
|
let _ = cancel_pending_autosave.send(());
|
||||||
|
}
|
||||||
|
|
||||||
|
let (cancel_tx, mut cancel_rx) = oneshot::channel();
|
||||||
|
self.cancel_pending_autosave = Some(cancel_tx);
|
||||||
|
self.pending_autosave = Some(cx.spawn_weak(|this, mut cx| async move {
|
||||||
|
let mut timer = futures::future::join(
|
||||||
|
cx.background().timer(Duration::from_millis(milliseconds)),
|
||||||
|
pending_autosave,
|
||||||
|
)
|
||||||
|
.fuse();
|
||||||
|
futures::select! {
|
||||||
|
_ = timer => {}
|
||||||
|
_ = cancel_rx => return None,
|
||||||
|
}
|
||||||
|
|
||||||
|
this.upgrade(&cx)?
|
||||||
|
.update(&mut cx, |this, cx| this.autosave(cx))
|
||||||
|
.await
|
||||||
|
.log_err();
|
||||||
|
None
|
||||||
|
}));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
language::Event::Reparsed => cx.emit(Event::Reparsed),
|
language::Event::Reparsed => cx.emit(Event::Reparsed),
|
||||||
language::Event::DirtyChanged => cx.emit(Event::DirtyChanged),
|
language::Event::DirtyChanged => cx.emit(Event::DirtyChanged),
|
||||||
|
@ -5588,6 +5626,22 @@ impl Editor {
|
||||||
cx.notify();
|
cx.notify();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn on_window_activation_changed(&mut self, active: bool, cx: &mut ViewContext<Self>) {
|
||||||
|
if !active && cx.global::<Settings>().autosave == Autosave::OnWindowChange {
|
||||||
|
self.autosave(cx).detach_and_log_err(cx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn autosave(&mut self, cx: &mut ViewContext<Self>) -> Task<Result<()>> {
|
||||||
|
if let Some(project) = self.project.clone() {
|
||||||
|
if self.buffer.read(cx).is_dirty(cx) && !self.buffer.read(cx).has_conflict(cx) {
|
||||||
|
return workspace::Item::save(self, project, cx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Task::ready(Ok(()))
|
||||||
|
}
|
||||||
|
|
||||||
pub fn set_searchable(&mut self, searchable: bool) {
|
pub fn set_searchable(&mut self, searchable: bool) {
|
||||||
self.searchable = searchable;
|
self.searchable = searchable;
|
||||||
}
|
}
|
||||||
|
@ -5805,6 +5859,10 @@ impl View for Editor {
|
||||||
hide_hover(self, cx);
|
hide_hover(self, cx);
|
||||||
cx.emit(Event::Blurred);
|
cx.emit(Event::Blurred);
|
||||||
cx.notify();
|
cx.notify();
|
||||||
|
|
||||||
|
if cx.global::<Settings>().autosave == Autosave::OnFocusChange {
|
||||||
|
self.autosave(cx).detach_and_log_err(cx);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn keymap_context(&self, _: &AppContext) -> gpui::keymap::Context {
|
fn keymap_context(&self, _: &AppContext) -> gpui::keymap::Context {
|
||||||
|
|
|
@ -25,6 +25,7 @@ pub struct Settings {
|
||||||
pub default_buffer_font_size: f32,
|
pub default_buffer_font_size: f32,
|
||||||
pub hover_popover_enabled: bool,
|
pub hover_popover_enabled: bool,
|
||||||
pub vim_mode: bool,
|
pub vim_mode: bool,
|
||||||
|
pub autosave: Autosave,
|
||||||
pub language_settings: LanguageSettings,
|
pub language_settings: LanguageSettings,
|
||||||
pub language_defaults: HashMap<Arc<str>, LanguageSettings>,
|
pub language_defaults: HashMap<Arc<str>, LanguageSettings>,
|
||||||
pub language_overrides: HashMap<Arc<str>, LanguageSettings>,
|
pub language_overrides: HashMap<Arc<str>, LanguageSettings>,
|
||||||
|
@ -39,7 +40,6 @@ pub struct LanguageSettings {
|
||||||
pub preferred_line_length: Option<u32>,
|
pub preferred_line_length: Option<u32>,
|
||||||
pub format_on_save: Option<bool>,
|
pub format_on_save: Option<bool>,
|
||||||
pub enable_language_server: Option<bool>,
|
pub enable_language_server: Option<bool>,
|
||||||
pub autosave: Option<Autosave>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Deserialize, PartialEq, Eq, JsonSchema)]
|
#[derive(Copy, Clone, Debug, Deserialize, PartialEq, Eq, JsonSchema)]
|
||||||
|
@ -54,7 +54,7 @@ pub enum SoftWrap {
|
||||||
#[serde(rename_all = "snake_case")]
|
#[serde(rename_all = "snake_case")]
|
||||||
pub enum Autosave {
|
pub enum Autosave {
|
||||||
Off,
|
Off,
|
||||||
AfterDelay { milliseconds: usize },
|
AfterDelay { milliseconds: u64 },
|
||||||
OnFocusChange,
|
OnFocusChange,
|
||||||
OnWindowChange,
|
OnWindowChange,
|
||||||
}
|
}
|
||||||
|
@ -74,6 +74,8 @@ pub struct SettingsFileContent {
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub format_on_save: Option<bool>,
|
pub format_on_save: Option<bool>,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
|
pub autosave: Option<Autosave>,
|
||||||
|
#[serde(default)]
|
||||||
pub enable_language_server: Option<bool>,
|
pub enable_language_server: Option<bool>,
|
||||||
#[serde(flatten)]
|
#[serde(flatten)]
|
||||||
pub editor: LanguageSettings,
|
pub editor: LanguageSettings,
|
||||||
|
@ -95,6 +97,7 @@ impl Settings {
|
||||||
default_buffer_font_size: 15.,
|
default_buffer_font_size: 15.,
|
||||||
hover_popover_enabled: true,
|
hover_popover_enabled: true,
|
||||||
vim_mode: false,
|
vim_mode: false,
|
||||||
|
autosave: Autosave::Off,
|
||||||
language_settings: Default::default(),
|
language_settings: Default::default(),
|
||||||
language_defaults: Default::default(),
|
language_defaults: Default::default(),
|
||||||
language_overrides: Default::default(),
|
language_overrides: Default::default(),
|
||||||
|
@ -138,11 +141,6 @@ impl Settings {
|
||||||
.unwrap_or(true)
|
.unwrap_or(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn autosave(&self, language: Option<&str>) -> Autosave {
|
|
||||||
self.language_setting(language, |settings| settings.autosave)
|
|
||||||
.unwrap_or(Autosave::Off)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn enable_language_server(&self, language: Option<&str>) -> bool {
|
pub fn enable_language_server(&self, language: Option<&str>) -> bool {
|
||||||
self.language_setting(language, |settings| settings.enable_language_server)
|
self.language_setting(language, |settings| settings.enable_language_server)
|
||||||
.unwrap_or(true)
|
.unwrap_or(true)
|
||||||
|
@ -172,6 +170,7 @@ impl Settings {
|
||||||
default_buffer_font_size: 14.,
|
default_buffer_font_size: 14.,
|
||||||
hover_popover_enabled: true,
|
hover_popover_enabled: true,
|
||||||
vim_mode: false,
|
vim_mode: false,
|
||||||
|
autosave: Autosave::Off,
|
||||||
language_settings: Default::default(),
|
language_settings: Default::default(),
|
||||||
language_defaults: Default::default(),
|
language_defaults: Default::default(),
|
||||||
language_overrides: Default::default(),
|
language_overrides: Default::default(),
|
||||||
|
@ -213,6 +212,7 @@ impl Settings {
|
||||||
merge(&mut self.default_buffer_font_size, data.buffer_font_size);
|
merge(&mut self.default_buffer_font_size, data.buffer_font_size);
|
||||||
merge(&mut self.hover_popover_enabled, data.hover_popover_enabled);
|
merge(&mut self.hover_popover_enabled, data.hover_popover_enabled);
|
||||||
merge(&mut self.vim_mode, data.vim_mode);
|
merge(&mut self.vim_mode, data.vim_mode);
|
||||||
|
merge(&mut self.autosave, data.autosave);
|
||||||
merge_option(
|
merge_option(
|
||||||
&mut self.language_settings.format_on_save,
|
&mut self.language_settings.format_on_save,
|
||||||
data.format_on_save,
|
data.format_on_save,
|
||||||
|
@ -227,7 +227,6 @@ impl Settings {
|
||||||
&mut self.language_settings.preferred_line_length,
|
&mut self.language_settings.preferred_line_length,
|
||||||
data.editor.preferred_line_length,
|
data.editor.preferred_line_length,
|
||||||
);
|
);
|
||||||
merge_option(&mut self.language_settings.autosave, data.editor.autosave);
|
|
||||||
|
|
||||||
for (language_name, settings) in data.language_overrides.clone().into_iter() {
|
for (language_name, settings) in data.language_overrides.clone().into_iter() {
|
||||||
let target = self
|
let target = self
|
||||||
|
@ -246,7 +245,6 @@ impl Settings {
|
||||||
&mut target.preferred_line_length,
|
&mut target.preferred_line_length,
|
||||||
settings.preferred_line_length,
|
settings.preferred_line_length,
|
||||||
);
|
);
|
||||||
merge_option(&mut target.autosave, settings.autosave);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue