Make formatting happen asynchronously.
This commit is contained in:
parent
3007478567
commit
c9be480bf8
4 changed files with 128 additions and 40 deletions
|
@ -182,6 +182,22 @@ pub mod util {
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// The result of asking the language server to format the document. This can be turned into a
|
||||||
|
/// `Transaction`, but the advantage of not doing that straight away is that this one is
|
||||||
|
/// `Send` and `Sync`.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct LspFormatting {
|
||||||
|
pub doc: Rope,
|
||||||
|
pub edits: Vec<lsp::TextEdit>,
|
||||||
|
pub offset_encoding: OffsetEncoding,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<LspFormatting> for Transaction {
|
||||||
|
fn from(fmt: LspFormatting) -> Transaction {
|
||||||
|
generate_transaction_from_edits(&fmt.doc, fmt.edits, fmt.offset_encoding)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Clone)]
|
#[derive(Debug, PartialEq, Clone)]
|
||||||
|
|
|
@ -35,7 +35,7 @@ use crate::{
|
||||||
ui::{self, Completion, Picker, Popup, Prompt, PromptEvent},
|
ui::{self, Completion, Picker, Popup, Prompt, PromptEvent},
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::application::{LspCallbackWrapper, LspCallbacks};
|
use crate::application::{LspCallback, LspCallbackWrapper, LspCallbacks};
|
||||||
use futures_util::FutureExt;
|
use futures_util::FutureExt;
|
||||||
use std::{fmt, future::Future, path::Display, str::FromStr};
|
use std::{fmt, future::Future, path::Display, str::FromStr};
|
||||||
|
|
||||||
|
@ -1145,11 +1145,12 @@ mod cmd {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_impl<P: AsRef<Path>>(
|
fn write_impl<P: AsRef<Path>>(
|
||||||
view: &View,
|
cx: &mut compositor::Context,
|
||||||
doc: &mut Document,
|
|
||||||
path: Option<P>,
|
path: Option<P>,
|
||||||
) -> Result<tokio::task::JoinHandle<Result<(), anyhow::Error>>, anyhow::Error> {
|
) -> Result<tokio::task::JoinHandle<Result<(), anyhow::Error>>, anyhow::Error> {
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
|
let callbacks = &mut cx.callbacks;
|
||||||
|
let (view, doc) = current!(cx.editor);
|
||||||
|
|
||||||
if let Some(path) = path {
|
if let Some(path) = path {
|
||||||
if let Err(err) = doc.set_path(path.as_ref()) {
|
if let Err(err) = doc.set_path(path.as_ref()) {
|
||||||
|
@ -1163,15 +1164,21 @@ mod cmd {
|
||||||
.language_config()
|
.language_config()
|
||||||
.map(|config| config.auto_format)
|
.map(|config| config.auto_format)
|
||||||
.unwrap_or_default();
|
.unwrap_or_default();
|
||||||
if autofmt {
|
let fmt = if autofmt {
|
||||||
doc.format(view.id); // TODO: merge into save
|
doc.format().map(|fmt| {
|
||||||
}
|
let shared = fmt.shared();
|
||||||
Ok(tokio::spawn(doc.save()))
|
let callback = make_format_callback(doc.id(), doc.version(), true, shared.clone());
|
||||||
|
callbacks.push(callback);
|
||||||
|
shared
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
Ok(tokio::spawn(doc.format_and_save(fmt)))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
fn write(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
||||||
let (view, doc) = current!(cx.editor);
|
if let Err(e) = write_impl(cx, args.first()) {
|
||||||
if let Err(e) = write_impl(view, doc, args.first()) {
|
|
||||||
cx.editor.set_error(e.to_string());
|
cx.editor.set_error(e.to_string());
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -1181,9 +1188,12 @@ mod cmd {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
fn format(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
||||||
let (view, doc) = current!(cx.editor);
|
let (_, doc) = current!(cx.editor);
|
||||||
|
|
||||||
doc.format(view.id)
|
if let Some(format) = doc.format() {
|
||||||
|
let callback = make_format_callback(doc.id(), doc.version(), false, format);
|
||||||
|
cx.callbacks.push(callback);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn set_indent_style(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
fn set_indent_style(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
||||||
|
@ -1288,8 +1298,7 @@ mod cmd {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_quit(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
fn write_quit(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
||||||
let (view, doc) = current!(cx.editor);
|
match write_impl(cx, args.first()) {
|
||||||
match write_impl(view, doc, args.first()) {
|
|
||||||
Ok(handle) => {
|
Ok(handle) => {
|
||||||
if let Err(e) = helix_lsp::block_on(handle) {
|
if let Err(e) = helix_lsp::block_on(handle) {
|
||||||
cx.editor.set_error(e.to_string());
|
cx.editor.set_error(e.to_string());
|
||||||
|
@ -1305,7 +1314,7 @@ mod cmd {
|
||||||
|
|
||||||
fn force_write_quit(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
fn force_write_quit(cx: &mut compositor::Context, args: &[&str], event: PromptEvent) {
|
||||||
let (view, doc) = current!(cx.editor);
|
let (view, doc) = current!(cx.editor);
|
||||||
match write_impl(view, doc, args.first()) {
|
match write_impl(cx, args.first()) {
|
||||||
Ok(handle) => {
|
Ok(handle) => {
|
||||||
if let Err(e) = helix_lsp::block_on(handle) {
|
if let Err(e) = helix_lsp::block_on(handle) {
|
||||||
cx.editor.set_error(e.to_string());
|
cx.editor.set_error(e.to_string());
|
||||||
|
@ -1902,6 +1911,35 @@ fn append_to_line(cx: &mut Context) {
|
||||||
doc.set_selection(view.id, selection);
|
doc.set_selection(view.id, selection);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Creates an LspCallback that waits for formatting changes to be computed. When they're done,
|
||||||
|
// it applies them, but only if the doc hasn't changed.
|
||||||
|
fn make_format_callback(
|
||||||
|
doc_id: DocumentId,
|
||||||
|
doc_version: i32,
|
||||||
|
set_unmodified: bool,
|
||||||
|
format: impl Future<Output = helix_lsp::util::LspFormatting> + Send + 'static,
|
||||||
|
) -> LspCallback {
|
||||||
|
Box::pin(async move {
|
||||||
|
let format = format.await;
|
||||||
|
let call: LspCallbackWrapper =
|
||||||
|
Box::new(move |editor: &mut Editor, compositor: &mut Compositor| {
|
||||||
|
let view_id = view!(editor).id;
|
||||||
|
if let Some(doc) = editor.document_mut(doc_id) {
|
||||||
|
if doc.version() == doc_version {
|
||||||
|
doc.apply(&Transaction::from(format), view_id);
|
||||||
|
doc.append_changes_to_history(view_id);
|
||||||
|
if set_unmodified {
|
||||||
|
doc.reset_modified();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log::info!("discarded formatting changes because the document changed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Ok(call)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
enum Open {
|
enum Open {
|
||||||
Below,
|
Below,
|
||||||
Above,
|
Above,
|
||||||
|
|
|
@ -16,6 +16,7 @@ use helix_core::{
|
||||||
ChangeSet, Diagnostic, LineEnding, Rope, RopeBuilder, Selection, State, Syntax, Transaction,
|
ChangeSet, Diagnostic, LineEnding, Rope, RopeBuilder, Selection, State, Syntax, Transaction,
|
||||||
DEFAULT_LINE_ENDING,
|
DEFAULT_LINE_ENDING,
|
||||||
};
|
};
|
||||||
|
use helix_lsp::util::LspFormatting;
|
||||||
|
|
||||||
use crate::{DocumentId, Theme, ViewId};
|
use crate::{DocumentId, Theme, ViewId};
|
||||||
|
|
||||||
|
@ -472,39 +473,55 @@ impl Document {
|
||||||
Ok(doc)
|
Ok(doc)
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: remove view_id dependency here
|
pub fn format(&self) -> Option<impl Future<Output = LspFormatting> + 'static> {
|
||||||
pub fn format(&mut self, view_id: ViewId) {
|
if let Some(language_server) = self.language_server.clone() {
|
||||||
if let Some(language_server) = self.language_server() {
|
let text = self.text.clone();
|
||||||
// TODO: await, no blocking
|
let id = self.identifier();
|
||||||
let transaction = helix_lsp::block_on(language_server.text_document_formatting(
|
let fut = async move {
|
||||||
self.identifier(),
|
let edits = language_server
|
||||||
lsp::FormattingOptions::default(),
|
.text_document_formatting(id, lsp::FormattingOptions::default(), None)
|
||||||
None,
|
.await
|
||||||
))
|
.unwrap_or_else(|e| {
|
||||||
.map(|edits| {
|
log::warn!("LSP formatting failed: {}", e);
|
||||||
helix_lsp::util::generate_transaction_from_edits(
|
Default::default()
|
||||||
self.text(),
|
});
|
||||||
|
LspFormatting {
|
||||||
|
doc: text,
|
||||||
edits,
|
edits,
|
||||||
language_server.offset_encoding(),
|
offset_encoding: language_server.offset_encoding(),
|
||||||
)
|
}
|
||||||
});
|
};
|
||||||
|
Some(fut)
|
||||||
if let Ok(transaction) = transaction {
|
} else {
|
||||||
self.apply(&transaction, view_id);
|
None
|
||||||
self.append_changes_to_history(view_id);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn save(&mut self) -> impl Future<Output = Result<(), anyhow::Error>> {
|
||||||
|
self.save_impl::<futures_util::future::Ready<_>>(None)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn format_and_save(
|
||||||
|
&mut self,
|
||||||
|
formatting: Option<impl Future<Output = LspFormatting>>,
|
||||||
|
) -> impl Future<Output = anyhow::Result<()>> {
|
||||||
|
self.save_impl(formatting)
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: do we need some way of ensuring two save operations on the same doc can't run at once?
|
// TODO: do we need some way of ensuring two save operations on the same doc can't run at once?
|
||||||
// or is that handled by the OS/async layer
|
// or is that handled by the OS/async layer
|
||||||
/// The `Document`'s text is encoded according to its encoding and written to the file located
|
/// The `Document`'s text is encoded according to its encoding and written to the file located
|
||||||
/// at its `path()`.
|
/// at its `path()`.
|
||||||
pub fn save(&mut self) -> impl Future<Output = Result<(), anyhow::Error>> {
|
///
|
||||||
|
/// If `formatting` is present, it supplies some changes that we apply to the text before saving.
|
||||||
|
fn save_impl<F: Future<Output = LspFormatting>>(
|
||||||
|
&mut self,
|
||||||
|
formatting: Option<F>,
|
||||||
|
) -> impl Future<Output = Result<(), anyhow::Error>> {
|
||||||
// we clone and move text + path into the future so that we asynchronously save the current
|
// we clone and move text + path into the future so that we asynchronously save the current
|
||||||
// state without blocking any further edits.
|
// state without blocking any further edits.
|
||||||
|
|
||||||
let text = self.text().clone();
|
let mut text = self.text().clone();
|
||||||
let path = self.path.clone().expect("Can't save with no path set!"); // TODO: handle no path
|
let path = self.path.clone().expect("Can't save with no path set!"); // TODO: handle no path
|
||||||
let identifier = self.identifier();
|
let identifier = self.identifier();
|
||||||
|
|
||||||
|
@ -512,10 +529,7 @@ impl Document {
|
||||||
|
|
||||||
let language_server = self.language_server.clone();
|
let language_server = self.language_server.clone();
|
||||||
|
|
||||||
// reset the modified flag
|
self.reset_modified();
|
||||||
let history = self.history.take();
|
|
||||||
self.last_saved_revision = history.current_revision();
|
|
||||||
self.history.set(history);
|
|
||||||
|
|
||||||
let encoding = self.encoding;
|
let encoding = self.encoding;
|
||||||
|
|
||||||
|
@ -531,6 +545,15 @@ impl Document {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(fmt) = formatting {
|
||||||
|
let success = Transaction::from(fmt.await).changes().apply(&mut text);
|
||||||
|
if !success {
|
||||||
|
// This shouldn't happen, because the transaction changes were generated
|
||||||
|
// from the same text we're saving.
|
||||||
|
log::error!("failed to apply format changes before saving");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let mut file = File::create(path).await?;
|
let mut file = File::create(path).await?;
|
||||||
to_writer(&mut file, encoding, &text).await?;
|
to_writer(&mut file, encoding, &text).await?;
|
||||||
|
|
||||||
|
@ -877,6 +900,13 @@ impl Document {
|
||||||
current_revision != self.last_saved_revision || !self.changes.is_empty()
|
current_revision != self.last_saved_revision || !self.changes.is_empty()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn reset_modified(&mut self) {
|
||||||
|
let history = self.history.take();
|
||||||
|
let current_revision = history.current_revision();
|
||||||
|
self.history.set(history);
|
||||||
|
self.last_saved_revision = current_revision;
|
||||||
|
}
|
||||||
|
|
||||||
pub fn mode(&self) -> Mode {
|
pub fn mode(&self) -> Mode {
|
||||||
self.mode
|
self.mode
|
||||||
}
|
}
|
||||||
|
|
|
@ -270,6 +270,10 @@ impl Editor {
|
||||||
self.documents.get(id)
|
self.documents.get(id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn document_mut(&mut self, id: DocumentId) -> Option<&mut Document> {
|
||||||
|
self.documents.get_mut(id)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn documents(&self) -> impl Iterator<Item = &Document> {
|
pub fn documents(&self) -> impl Iterator<Item = &Document> {
|
||||||
self.documents.iter().map(|(_id, doc)| doc)
|
self.documents.iter().map(|(_id, doc)| doc)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue