mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 21:54:42 +00:00
introduce SingleFileChange
This commit is contained in:
parent
b112430ca7
commit
6f00bb1cb0
3 changed files with 64 additions and 35 deletions
|
@ -6,7 +6,7 @@
|
||||||
use ra_text_edit::TextEdit;
|
use ra_text_edit::TextEdit;
|
||||||
use relative_path::RelativePathBuf;
|
use relative_path::RelativePathBuf;
|
||||||
|
|
||||||
use crate::{FileId, FilePosition, SourceRootId};
|
use crate::{FileId, FilePosition, SourceRootId, TextUnit};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct SourceChange {
|
pub struct SourceChange {
|
||||||
|
@ -100,3 +100,20 @@ pub enum FileSystemEdit {
|
||||||
CreateFile { source_root: SourceRootId, path: RelativePathBuf },
|
CreateFile { source_root: SourceRootId, path: RelativePathBuf },
|
||||||
MoveFile { src: FileId, dst_source_root: SourceRootId, dst_path: RelativePathBuf },
|
MoveFile { src: FileId, dst_source_root: SourceRootId, dst_path: RelativePathBuf },
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) struct SingleFileChange {
|
||||||
|
pub label: String,
|
||||||
|
pub edit: TextEdit,
|
||||||
|
pub cursor_position: Option<TextUnit>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SingleFileChange {
|
||||||
|
pub(crate) fn into_source_change(self, file_id: FileId) -> SourceChange {
|
||||||
|
SourceChange {
|
||||||
|
label: self.label,
|
||||||
|
source_file_edits: vec![SourceFileEdit { file_id, edit: self.edit }],
|
||||||
|
file_system_edits: Vec::new(),
|
||||||
|
cursor_position: self.cursor_position.map(|offset| FilePosition { file_id, offset }),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -24,7 +24,7 @@ use ra_syntax::{
|
||||||
};
|
};
|
||||||
use ra_text_edit::{TextEdit, TextEditBuilder};
|
use ra_text_edit::{TextEdit, TextEditBuilder};
|
||||||
|
|
||||||
use crate::{db::RootDatabase, SourceChange, SourceFileEdit};
|
use crate::{db::RootDatabase, source_change::SingleFileChange, SourceChange, SourceFileEdit};
|
||||||
|
|
||||||
pub(crate) fn on_enter(db: &RootDatabase, position: FilePosition) -> Option<SourceChange> {
|
pub(crate) fn on_enter(db: &RootDatabase, position: FilePosition) -> Option<SourceChange> {
|
||||||
let parse = db.parse(position.file_id);
|
let parse = db.parse(position.file_id);
|
||||||
|
@ -88,32 +88,19 @@ pub(crate) fn on_char_typed(
|
||||||
) -> Option<SourceChange> {
|
) -> Option<SourceChange> {
|
||||||
let file = &db.parse(position.file_id).tree();
|
let file = &db.parse(position.file_id).tree();
|
||||||
assert_eq!(file.syntax().text().char_at(position.offset), Some(char_typed));
|
assert_eq!(file.syntax().text().char_at(position.offset), Some(char_typed));
|
||||||
match char_typed {
|
let single_file_change = match char_typed {
|
||||||
'=' => {
|
'=' => on_eq_typed(file, position.offset)?,
|
||||||
let edit = on_eq_typed(file, position.offset)?;
|
'.' => on_dot_typed(file, position.offset)?,
|
||||||
Some(SourceChange::source_file_edit(
|
_ => return None,
|
||||||
"add semicolon",
|
};
|
||||||
SourceFileEdit { edit, file_id: position.file_id },
|
|
||||||
))
|
Some(single_file_change.into_source_change(position.file_id))
|
||||||
}
|
|
||||||
'.' => {
|
|
||||||
let (edit, cursor_offset) = on_dot_typed(file, position.offset)?;
|
|
||||||
Some(
|
|
||||||
SourceChange::source_file_edit(
|
|
||||||
"reindent dot",
|
|
||||||
SourceFileEdit { edit, file_id: position.file_id },
|
|
||||||
)
|
|
||||||
.with_cursor(FilePosition { file_id: position.file_id, offset: cursor_offset }),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns an edit which should be applied after `=` was typed. Primarily,
|
/// Returns an edit which should be applied after `=` was typed. Primarily,
|
||||||
/// this works when adding `let =`.
|
/// this works when adding `let =`.
|
||||||
// FIXME: use a snippet completion instead of this hack here.
|
// FIXME: use a snippet completion instead of this hack here.
|
||||||
fn on_eq_typed(file: &SourceFile, offset: TextUnit) -> Option<TextEdit> {
|
fn on_eq_typed(file: &SourceFile, offset: TextUnit) -> Option<SingleFileChange> {
|
||||||
assert_eq!(file.syntax().text().char_at(offset), Some('='));
|
assert_eq!(file.syntax().text().char_at(offset), Some('='));
|
||||||
let let_stmt: ast::LetStmt = find_node_at_offset(file.syntax(), offset)?;
|
let let_stmt: ast::LetStmt = find_node_at_offset(file.syntax(), offset)?;
|
||||||
if let_stmt.has_semi() {
|
if let_stmt.has_semi() {
|
||||||
|
@ -131,13 +118,15 @@ fn on_eq_typed(file: &SourceFile, offset: TextUnit) -> Option<TextEdit> {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
let offset = let_stmt.syntax().text_range().end();
|
let offset = let_stmt.syntax().text_range().end();
|
||||||
let mut edit = TextEditBuilder::default();
|
Some(SingleFileChange {
|
||||||
edit.insert(offset, ";".to_string());
|
label: "add semicolon".to_string(),
|
||||||
Some(edit.finish())
|
edit: TextEdit::insert(offset, ";".to_string()),
|
||||||
|
cursor_position: None,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns an edit which should be applied when a dot ('.') is typed on a blank line, indenting the line appropriately.
|
/// Returns an edit which should be applied when a dot ('.') is typed on a blank line, indenting the line appropriately.
|
||||||
fn on_dot_typed(file: &SourceFile, offset: TextUnit) -> Option<(TextEdit, TextUnit)> {
|
fn on_dot_typed(file: &SourceFile, offset: TextUnit) -> Option<SingleFileChange> {
|
||||||
assert_eq!(file.syntax().text().char_at(offset), Some('.'));
|
assert_eq!(file.syntax().text().char_at(offset), Some('.'));
|
||||||
let whitespace =
|
let whitespace =
|
||||||
file.syntax().token_at_offset(offset).left_biased().and_then(ast::Whitespace::cast)?;
|
file.syntax().token_at_offset(offset).left_biased().and_then(ast::Whitespace::cast)?;
|
||||||
|
@ -157,12 +146,17 @@ fn on_dot_typed(file: &SourceFile, offset: TextUnit) -> Option<(TextEdit, TextUn
|
||||||
if current_indent_len == target_indent_len {
|
if current_indent_len == target_indent_len {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
let mut edit = TextEditBuilder::default();
|
|
||||||
edit.replace(TextRange::from_to(offset - current_indent_len, offset), target_indent);
|
|
||||||
|
|
||||||
let cursor_offset = offset + target_indent_len - current_indent_len + TextUnit::of_char('.');
|
Some(SingleFileChange {
|
||||||
|
label: "reindent dot".to_string(),
|
||||||
Some((edit.finish(), cursor_offset))
|
edit: TextEdit::replace(
|
||||||
|
TextRange::from_to(offset - current_indent_len, offset),
|
||||||
|
target_indent,
|
||||||
|
),
|
||||||
|
cursor_position: Some(
|
||||||
|
offset + target_indent_len - current_indent_len + TextUnit::of_char('.'),
|
||||||
|
),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -182,7 +176,7 @@ mod tests {
|
||||||
let before = edit.finish().apply(&before);
|
let before = edit.finish().apply(&before);
|
||||||
let parse = SourceFile::parse(&before);
|
let parse = SourceFile::parse(&before);
|
||||||
if let Some(result) = on_eq_typed(&parse.tree(), offset) {
|
if let Some(result) = on_eq_typed(&parse.tree(), offset) {
|
||||||
let actual = result.apply(&before);
|
let actual = result.edit.apply(&before);
|
||||||
assert_eq_text!(after, &actual);
|
assert_eq_text!(after, &actual);
|
||||||
} else {
|
} else {
|
||||||
assert_eq_text!(&before, after)
|
assert_eq_text!(&before, after)
|
||||||
|
@ -230,8 +224,8 @@ fn foo() {
|
||||||
let before = edit.finish().apply(&before);
|
let before = edit.finish().apply(&before);
|
||||||
let (analysis, file_id) = single_file(&before);
|
let (analysis, file_id) = single_file(&before);
|
||||||
let file = analysis.parse(file_id).unwrap();
|
let file = analysis.parse(file_id).unwrap();
|
||||||
if let Some((edit, _cursor_offset)) = on_dot_typed(&file, offset) {
|
if let Some(result) = on_dot_typed(&file, offset) {
|
||||||
let actual = edit.apply(&before);
|
let actual = result.edit.apply(&before);
|
||||||
assert_eq_text!(after, &actual);
|
assert_eq_text!(after, &actual);
|
||||||
} else {
|
} else {
|
||||||
assert_eq_text!(&before, after)
|
assert_eq_text!(&before, after)
|
||||||
|
|
|
@ -32,6 +32,24 @@ impl TextEditBuilder {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TextEdit {
|
impl TextEdit {
|
||||||
|
pub fn insert(offset: TextUnit, text: String) -> TextEdit {
|
||||||
|
let mut builder = TextEditBuilder::default();
|
||||||
|
builder.insert(offset, text);
|
||||||
|
builder.finish()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn delete(range: TextRange) -> TextEdit {
|
||||||
|
let mut builder = TextEditBuilder::default();
|
||||||
|
builder.delete(range);
|
||||||
|
builder.finish()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn replace(range: TextRange, replace_with: String) -> TextEdit {
|
||||||
|
let mut builder = TextEditBuilder::default();
|
||||||
|
builder.replace(range, replace_with);
|
||||||
|
builder.finish()
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn from_atoms(mut atoms: Vec<AtomTextEdit>) -> TextEdit {
|
pub(crate) fn from_atoms(mut atoms: Vec<AtomTextEdit>) -> TextEdit {
|
||||||
atoms.sort_by_key(|a| (a.delete.start(), a.delete.end()));
|
atoms.sort_by_key(|a| (a.delete.start(), a.delete.end()));
|
||||||
for (a1, a2) in atoms.iter().zip(atoms.iter().skip(1)) {
|
for (a1, a2) in atoms.iter().zip(atoms.iter().skip(1)) {
|
||||||
|
|
Loading…
Reference in a new issue