mirror of
https://github.com/latex-lsp/texlab.git
synced 2025-12-23 09:19:21 +00:00
Refactor "rename" feature
This commit is contained in:
parent
f435287cce
commit
2c8ba68410
9 changed files with 387 additions and 260 deletions
|
|
@ -2,7 +2,13 @@ mod command;
|
|||
mod entry;
|
||||
mod label;
|
||||
|
||||
use lsp_types::{Range, RenameParams, TextDocumentPositionParams, WorkspaceEdit};
|
||||
use std::sync::Arc;
|
||||
|
||||
use lsp_types::{Range, RenameParams, TextDocumentPositionParams, TextEdit, Url, WorkspaceEdit};
|
||||
use rowan::TextRange;
|
||||
use rustc_hash::FxHashMap;
|
||||
|
||||
use crate::LineIndexExt;
|
||||
|
||||
use self::{
|
||||
command::{prepare_command_rename, rename_command},
|
||||
|
|
@ -14,14 +20,49 @@ use super::{cursor::CursorContext, FeatureRequest};
|
|||
|
||||
pub fn prepare_rename_all(request: FeatureRequest<TextDocumentPositionParams>) -> Option<Range> {
|
||||
let context = CursorContext::new(request);
|
||||
prepare_entry_rename(&context)
|
||||
let range = prepare_entry_rename(&context)
|
||||
.or_else(|| prepare_label_rename(&context))
|
||||
.or_else(|| prepare_command_rename(&context))
|
||||
.or_else(|| prepare_command_rename(&context))?;
|
||||
|
||||
let line_index = &context.request.main_document().line_index;
|
||||
Some(line_index.line_col_lsp_range(range))
|
||||
}
|
||||
|
||||
pub fn rename_all(request: FeatureRequest<RenameParams>) -> Option<WorkspaceEdit> {
|
||||
let context = CursorContext::new(request);
|
||||
rename_entry(&context)
|
||||
let result = rename_entry(&context)
|
||||
.or_else(|| rename_label(&context))
|
||||
.or_else(|| rename_command(&context))
|
||||
.or_else(|| rename_command(&context))?;
|
||||
|
||||
let changes = result
|
||||
.changes
|
||||
.into_iter()
|
||||
.map(|(uri, old_edits)| {
|
||||
let document = &context.request.workspace.documents_by_uri[&uri];
|
||||
let new_edits = old_edits
|
||||
.into_iter()
|
||||
.map(|Indel { delete, insert }| {
|
||||
TextEdit::new(document.line_index.line_col_lsp_range(delete), insert)
|
||||
})
|
||||
.collect();
|
||||
|
||||
(uri.as_ref().clone(), new_edits)
|
||||
})
|
||||
.collect();
|
||||
|
||||
Some(WorkspaceEdit::new(changes))
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq, Clone)]
|
||||
struct Indel {
|
||||
delete: TextRange,
|
||||
insert: String,
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq, Clone)]
|
||||
struct RenameResult {
|
||||
changes: FxHashMap<Arc<Url>, Vec<Indel>>,
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests;
|
||||
|
|
|
|||
|
|
@ -1,88 +1,45 @@
|
|||
use std::collections::HashMap;
|
||||
use std::sync::Arc;
|
||||
|
||||
use lsp_types::{Range, RenameParams, TextEdit, WorkspaceEdit};
|
||||
use lsp_types::RenameParams;
|
||||
use rowan::{TextRange, TextSize};
|
||||
use rustc_hash::FxHashMap;
|
||||
|
||||
use crate::{
|
||||
features::cursor::{CursorContext, HasPosition},
|
||||
syntax::latex,
|
||||
LineIndexExt,
|
||||
};
|
||||
|
||||
pub fn prepare_command_rename<P: HasPosition>(context: &CursorContext<P>) -> Option<Range> {
|
||||
Some(
|
||||
context
|
||||
.request
|
||||
.main_document()
|
||||
.line_index
|
||||
.line_col_lsp_range(context.cursor.command_range(context.offset)?),
|
||||
)
|
||||
use super::{Indel, RenameResult};
|
||||
|
||||
pub(super) fn prepare_command_rename<P: HasPosition>(
|
||||
context: &CursorContext<P>,
|
||||
) -> Option<TextRange> {
|
||||
context.cursor.command_range(context.offset)
|
||||
}
|
||||
|
||||
pub fn rename_command(context: &CursorContext<RenameParams>) -> Option<WorkspaceEdit> {
|
||||
pub(super) fn rename_command(context: &CursorContext<RenameParams>) -> Option<RenameResult> {
|
||||
prepare_command_rename(context)?;
|
||||
let name = context.cursor.as_latex()?.text();
|
||||
let mut changes = HashMap::new();
|
||||
let mut changes = FxHashMap::default();
|
||||
for document in context.request.workspace.documents_by_uri.values() {
|
||||
if let Some(data) = document.data.as_latex() {
|
||||
let edits = latex::SyntaxNode::new_root(data.green.clone())
|
||||
let root = latex::SyntaxNode::new_root(data.green.clone());
|
||||
let edits = root
|
||||
.descendants_with_tokens()
|
||||
.filter_map(|element| element.into_token())
|
||||
.filter(|token| token.kind().is_command_name() && token.text() == name)
|
||||
.map(|token| {
|
||||
let range = token.text_range();
|
||||
let range = document.line_index.line_col_lsp_range(TextRange::new(
|
||||
range.start() + TextSize::from(1),
|
||||
range.end(),
|
||||
));
|
||||
TextEdit::new(range, context.request.params.new_name.clone())
|
||||
Indel {
|
||||
delete: TextRange::new(range.start() + TextSize::from(1), range.end()),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
changes.insert(document.uri.as_ref().clone(), edits);
|
||||
changes.insert(Arc::clone(&document.uri), edits);
|
||||
}
|
||||
}
|
||||
|
||||
Some(WorkspaceEdit::new(changes))
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::{features::testing::FeatureTester, RangeExt};
|
||||
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_command() {
|
||||
let tester = FeatureTester::builder()
|
||||
.files(vec![
|
||||
("foo.tex", r#"\baz\include{bar.tex}"#),
|
||||
("bar.tex", r#"\baz"#),
|
||||
])
|
||||
.main("foo.tex")
|
||||
.line(0)
|
||||
.character(2)
|
||||
.new_name("qux")
|
||||
.build();
|
||||
|
||||
let uri1 = tester.uri("foo.tex");
|
||||
let uri2 = tester.uri("bar.tex");
|
||||
let req = tester.rename();
|
||||
|
||||
let context = CursorContext::new(req);
|
||||
let actual_edit = rename_command(&context).unwrap();
|
||||
|
||||
let mut expected_changes = HashMap::new();
|
||||
expected_changes.insert(
|
||||
uri1.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 1, 0, 4), "qux".into())],
|
||||
);
|
||||
expected_changes.insert(
|
||||
uri2.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 1, 0, 4), "qux".into())],
|
||||
);
|
||||
let expected_edit = WorkspaceEdit::new(expected_changes);
|
||||
|
||||
assert_eq!(actual_edit, expected_edit);
|
||||
}
|
||||
Some(RenameResult { changes })
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,7 +1,8 @@
|
|||
use std::collections::HashMap;
|
||||
use std::sync::Arc;
|
||||
|
||||
use lsp_types::{Range, RenameParams, TextEdit, WorkspaceEdit};
|
||||
use rowan::ast::AstNode;
|
||||
use lsp_types::RenameParams;
|
||||
use rowan::{ast::AstNode, TextRange};
|
||||
use rustc_hash::FxHashMap;
|
||||
|
||||
use crate::{
|
||||
features::cursor::{CursorContext, HasPosition},
|
||||
|
|
@ -9,139 +10,63 @@ use crate::{
|
|||
bibtex::{self, HasName},
|
||||
latex,
|
||||
},
|
||||
DocumentData, LineIndexExt,
|
||||
DocumentData,
|
||||
};
|
||||
|
||||
pub fn prepare_entry_rename<P: HasPosition>(context: &CursorContext<P>) -> Option<Range> {
|
||||
use super::{Indel, RenameResult};
|
||||
|
||||
pub(super) fn prepare_entry_rename<P: HasPosition>(
|
||||
context: &CursorContext<P>,
|
||||
) -> Option<TextRange> {
|
||||
let (_, range) = context
|
||||
.find_citation_key_word()
|
||||
.or_else(|| context.find_entry_key())?;
|
||||
|
||||
Some(
|
||||
context
|
||||
.request
|
||||
.main_document()
|
||||
.line_index
|
||||
.line_col_lsp_range(range),
|
||||
)
|
||||
Some(range)
|
||||
}
|
||||
|
||||
pub fn rename_entry(context: &CursorContext<RenameParams>) -> Option<WorkspaceEdit> {
|
||||
pub(super) fn rename_entry(context: &CursorContext<RenameParams>) -> Option<RenameResult> {
|
||||
prepare_entry_rename(context)?;
|
||||
let (key_text, _) = context
|
||||
.find_citation_key_word()
|
||||
.or_else(|| context.find_entry_key())?;
|
||||
|
||||
let mut changes = HashMap::new();
|
||||
let mut changes = FxHashMap::default();
|
||||
for document in context.request.workspace.documents_by_uri.values() {
|
||||
let uri = Arc::clone(&document.uri);
|
||||
match &document.data {
|
||||
DocumentData::Latex(data) => {
|
||||
let edits: Vec<_> = latex::SyntaxNode::new_root(data.green.clone())
|
||||
let root = latex::SyntaxNode::new_root(data.green.clone());
|
||||
let edits: Vec<_> = root
|
||||
.descendants()
|
||||
.filter_map(latex::Citation::cast)
|
||||
.filter_map(|citation| citation.key_list())
|
||||
.flat_map(|keys| keys.keys())
|
||||
.filter(|key| key.to_string() == key_text)
|
||||
.map(|key| {
|
||||
document
|
||||
.line_index
|
||||
.line_col_lsp_range(latex::small_range(&key))
|
||||
.map(|key| Indel {
|
||||
delete: latex::small_range(&key),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
})
|
||||
.map(|range| TextEdit::new(range, context.request.params.new_name.clone()))
|
||||
.collect();
|
||||
changes.insert(document.uri.as_ref().clone(), edits);
|
||||
changes.insert(uri, edits);
|
||||
}
|
||||
DocumentData::Bibtex(data) => {
|
||||
let edits: Vec<_> = bibtex::SyntaxNode::new_root(data.green.clone())
|
||||
let root = bibtex::SyntaxNode::new_root(data.green.clone());
|
||||
let edits: Vec<_> = root
|
||||
.descendants()
|
||||
.filter_map(bibtex::Entry::cast)
|
||||
.filter_map(|entry| entry.name_token())
|
||||
.filter(|key| key.text() == key_text)
|
||||
.map(|key| document.line_index.line_col_lsp_range(key.text_range()))
|
||||
.map(|range| TextEdit::new(range, context.request.params.new_name.clone()))
|
||||
.map(|key| Indel {
|
||||
delete: key.text_range(),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
})
|
||||
.collect();
|
||||
changes.insert(document.uri.as_ref().clone(), edits);
|
||||
changes.insert(uri, edits);
|
||||
}
|
||||
DocumentData::BuildLog(_) => {}
|
||||
}
|
||||
}
|
||||
|
||||
Some(WorkspaceEdit::new(changes))
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use lsp_types::TextEdit;
|
||||
|
||||
use crate::{features::testing::FeatureTester, RangeExt};
|
||||
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_entry() {
|
||||
let tester = FeatureTester::builder()
|
||||
.files(vec![
|
||||
("main.bib", r#"@article{foo, bar = baz}"#),
|
||||
("main.tex", "\\addbibresource{main.bib}\n\\cite{foo}"),
|
||||
])
|
||||
.main("main.bib")
|
||||
.line(0)
|
||||
.character(9)
|
||||
.new_name("qux")
|
||||
.build();
|
||||
|
||||
let uri1 = tester.uri("main.bib");
|
||||
let uri2 = tester.uri("main.tex");
|
||||
let request = tester.rename();
|
||||
|
||||
let context = CursorContext::new(request);
|
||||
let actual_edit = rename_entry(&context).unwrap();
|
||||
|
||||
let mut expected_changes = HashMap::new();
|
||||
expected_changes.insert(
|
||||
uri1.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 9, 0, 12), "qux".into())],
|
||||
);
|
||||
expected_changes.insert(
|
||||
uri2.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(1, 6, 1, 9), "qux".into())],
|
||||
);
|
||||
let expected_edit = WorkspaceEdit::new(expected_changes);
|
||||
|
||||
assert_eq!(actual_edit, expected_edit);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_citation() {
|
||||
let tester = FeatureTester::builder()
|
||||
.files(vec![
|
||||
("main.bib", r#"@article{foo, bar = baz}"#),
|
||||
("main.tex", "\\addbibresource{main.bib}\n\\cite{foo}"),
|
||||
])
|
||||
.main("main.tex")
|
||||
.line(1)
|
||||
.character(6)
|
||||
.new_name("qux")
|
||||
.build();
|
||||
|
||||
let uri1 = tester.uri("main.bib");
|
||||
let uri2 = tester.uri("main.tex");
|
||||
let request = tester.rename();
|
||||
|
||||
let context = CursorContext::new(request);
|
||||
let actual_edit = rename_entry(&context).unwrap();
|
||||
|
||||
let mut expected_changes = HashMap::new();
|
||||
expected_changes.insert(
|
||||
uri1.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 9, 0, 12), "qux".into())],
|
||||
);
|
||||
expected_changes.insert(
|
||||
uri2.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(1, 6, 1, 9), "qux".into())],
|
||||
);
|
||||
let expected_edit = WorkspaceEdit::new(expected_changes);
|
||||
|
||||
assert_eq!(actual_edit, expected_edit);
|
||||
}
|
||||
Some(RenameResult { changes })
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,31 +1,28 @@
|
|||
use std::collections::HashMap;
|
||||
use std::sync::Arc;
|
||||
|
||||
use lsp_types::{Range, RenameParams, TextEdit, WorkspaceEdit};
|
||||
use rowan::ast::AstNode;
|
||||
use lsp_types::RenameParams;
|
||||
use rowan::{ast::AstNode, TextRange};
|
||||
use rustc_hash::FxHashMap;
|
||||
|
||||
use crate::{
|
||||
features::cursor::{CursorContext, HasPosition},
|
||||
syntax::latex,
|
||||
LineIndexExt,
|
||||
};
|
||||
|
||||
pub fn prepare_label_rename<P: HasPosition>(context: &CursorContext<P>) -> Option<Range> {
|
||||
let (_, range) = context.find_label_name_key()?;
|
||||
use super::{Indel, RenameResult};
|
||||
|
||||
Some(
|
||||
context
|
||||
.request
|
||||
.main_document()
|
||||
.line_index
|
||||
.line_col_lsp_range(range),
|
||||
)
|
||||
pub(super) fn prepare_label_rename<P: HasPosition>(
|
||||
context: &CursorContext<P>,
|
||||
) -> Option<TextRange> {
|
||||
let (_, range) = context.find_label_name_key()?;
|
||||
Some(range)
|
||||
}
|
||||
|
||||
pub fn rename_label(context: &CursorContext<RenameParams>) -> Option<WorkspaceEdit> {
|
||||
pub(super) fn rename_label(context: &CursorContext<RenameParams>) -> Option<RenameResult> {
|
||||
prepare_label_rename(context)?;
|
||||
let (name_text, _) = context.find_label_name_key()?;
|
||||
|
||||
let mut changes = HashMap::new();
|
||||
let mut changes = FxHashMap::default();
|
||||
for document in context.request.workspace.documents_by_uri.values() {
|
||||
if let Some(data) = document.data.as_latex() {
|
||||
let mut edits = Vec::new();
|
||||
|
|
@ -34,16 +31,12 @@ pub fn rename_label(context: &CursorContext<RenameParams>) -> Option<WorkspaceEd
|
|||
.and_then(|label| label.name())
|
||||
.and_then(|name| name.key())
|
||||
.filter(|name| name.to_string() == name_text)
|
||||
.map(|name| {
|
||||
document
|
||||
.line_index
|
||||
.line_col_lsp_range(latex::small_range(&name))
|
||||
})
|
||||
.map(|name| latex::small_range(&name))
|
||||
{
|
||||
edits.push(TextEdit::new(
|
||||
range,
|
||||
context.request.params.new_name.clone(),
|
||||
));
|
||||
edits.push(Indel {
|
||||
delete: range,
|
||||
insert: context.request.params.new_name.clone(),
|
||||
});
|
||||
}
|
||||
|
||||
latex::LabelReference::cast(node.clone())
|
||||
|
|
@ -51,16 +44,11 @@ pub fn rename_label(context: &CursorContext<RenameParams>) -> Option<WorkspaceEd
|
|||
.into_iter()
|
||||
.flat_map(|label| label.keys())
|
||||
.filter(|name| name.to_string() == name_text)
|
||||
.map(|name| {
|
||||
document
|
||||
.line_index
|
||||
.line_col_lsp_range(latex::small_range(&name))
|
||||
})
|
||||
.for_each(|range| {
|
||||
edits.push(TextEdit::new(
|
||||
range,
|
||||
context.request.params.new_name.clone(),
|
||||
));
|
||||
.for_each(|name| {
|
||||
edits.push(Indel {
|
||||
delete: latex::small_range(&name),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
});
|
||||
});
|
||||
|
||||
if let Some(label) = latex::LabelReferenceRange::cast(node.clone()) {
|
||||
|
|
@ -69,12 +57,10 @@ pub fn rename_label(context: &CursorContext<RenameParams>) -> Option<WorkspaceEd
|
|||
.and_then(|name| name.key())
|
||||
.filter(|name| name.to_string() == name_text)
|
||||
{
|
||||
edits.push(TextEdit::new(
|
||||
document
|
||||
.line_index
|
||||
.line_col_lsp_range(latex::small_range(&name1)),
|
||||
context.request.params.new_name.clone(),
|
||||
));
|
||||
edits.push(Indel {
|
||||
delete: latex::small_range(&name1),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
});
|
||||
}
|
||||
|
||||
if let Some(name2) = label
|
||||
|
|
@ -82,61 +68,17 @@ pub fn rename_label(context: &CursorContext<RenameParams>) -> Option<WorkspaceEd
|
|||
.and_then(|name| name.key())
|
||||
.filter(|name| name.to_string() == name_text)
|
||||
{
|
||||
edits.push(TextEdit::new(
|
||||
document
|
||||
.line_index
|
||||
.line_col_lsp_range(latex::small_range(&name2)),
|
||||
context.request.params.new_name.clone(),
|
||||
));
|
||||
edits.push(Indel {
|
||||
delete: latex::small_range(&name2),
|
||||
insert: context.request.params.new_name.clone(),
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
changes.insert(document.uri.as_ref().clone(), edits);
|
||||
changes.insert(Arc::clone(&document.uri), edits);
|
||||
}
|
||||
}
|
||||
|
||||
Some(WorkspaceEdit::new(changes))
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::{features::testing::FeatureTester, RangeExt};
|
||||
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_label() {
|
||||
let tester = FeatureTester::builder()
|
||||
.files(vec![
|
||||
("foo.tex", r#"\label{foo}\include{bar}"#),
|
||||
("bar.tex", r#"\ref{foo}"#),
|
||||
("baz.tex", r#"\ref{foo}"#),
|
||||
])
|
||||
.main("foo.tex")
|
||||
.line(0)
|
||||
.character(7)
|
||||
.new_name("bar")
|
||||
.build();
|
||||
|
||||
let uri1 = tester.uri("foo.tex");
|
||||
let uri2 = tester.uri("bar.tex");
|
||||
let request = tester.rename();
|
||||
|
||||
let context = CursorContext::new(request);
|
||||
let actual_edit = rename_label(&context).unwrap();
|
||||
|
||||
let mut expected_changes = HashMap::new();
|
||||
expected_changes.insert(
|
||||
uri1.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 7, 0, 10), "bar".into())],
|
||||
);
|
||||
expected_changes.insert(
|
||||
uri2.as_ref().clone(),
|
||||
vec![TextEdit::new(Range::new_simple(0, 5, 0, 8), "bar".into())],
|
||||
);
|
||||
let expected_edit = WorkspaceEdit::new(expected_changes);
|
||||
|
||||
assert_eq!(actual_edit, expected_edit);
|
||||
}
|
||||
Some(RenameResult { changes })
|
||||
}
|
||||
|
|
|
|||
|
|
@ -0,0 +1,39 @@
|
|||
---
|
||||
source: src/features/rename/tests.rs
|
||||
assertion_line: 92
|
||||
expression: edit
|
||||
---
|
||||
{
|
||||
"changes": {
|
||||
"[tmp]/main.bib": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 9
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 12
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
],
|
||||
"[tmp]/main.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 1,
|
||||
"character": 6
|
||||
},
|
||||
"end": {
|
||||
"line": 1,
|
||||
"character": 9
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,39 @@
|
|||
---
|
||||
source: src/features/rename/tests.rs
|
||||
assertion_line: 23
|
||||
expression: edit
|
||||
---
|
||||
{
|
||||
"changes": {
|
||||
"[tmp]/bar.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 1
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 4
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
],
|
||||
"[tmp]/foo.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 1
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 4
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,39 @@
|
|||
---
|
||||
source: src/features/rename/tests.rs
|
||||
assertion_line: 51
|
||||
expression: edit
|
||||
---
|
||||
{
|
||||
"changes": {
|
||||
"[tmp]/main.bib": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 9
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 12
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
],
|
||||
"[tmp]/main.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 1,
|
||||
"character": 6
|
||||
},
|
||||
"end": {
|
||||
"line": 1,
|
||||
"character": 9
|
||||
}
|
||||
},
|
||||
"newText": "qux"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,39 @@
|
|||
---
|
||||
source: src/features/rename/tests.rs
|
||||
assertion_line: 99
|
||||
expression: edit
|
||||
---
|
||||
{
|
||||
"changes": {
|
||||
"[tmp]/bar.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 5
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 8
|
||||
}
|
||||
},
|
||||
"newText": "bar"
|
||||
}
|
||||
],
|
||||
"[tmp]/foo.tex": [
|
||||
{
|
||||
"range": {
|
||||
"start": {
|
||||
"line": 0,
|
||||
"character": 7
|
||||
},
|
||||
"end": {
|
||||
"line": 0,
|
||||
"character": 10
|
||||
}
|
||||
},
|
||||
"newText": "bar"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
106
src/features/rename/tests.rs
Normal file
106
src/features/rename/tests.rs
Normal file
|
|
@ -0,0 +1,106 @@
|
|||
use insta::assert_json_snapshot;
|
||||
|
||||
use crate::features::{redactions::redact_uri, testing::FeatureTester};
|
||||
|
||||
use super::rename_all;
|
||||
|
||||
#[test]
|
||||
fn test_command() {
|
||||
let edit = rename_all(
|
||||
FeatureTester::builder()
|
||||
.files(vec![
|
||||
("foo.tex", r#"\baz\include{bar.tex}"#),
|
||||
("bar.tex", r#"\baz"#),
|
||||
])
|
||||
.main("foo.tex")
|
||||
.line(0)
|
||||
.character(2)
|
||||
.new_name("qux")
|
||||
.build()
|
||||
.rename(),
|
||||
);
|
||||
|
||||
assert_json_snapshot!(
|
||||
edit,
|
||||
{
|
||||
".changes.$key" => insta::dynamic_redaction(redact_uri),
|
||||
".changes" => insta::sorted_redaction(),
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_entry() {
|
||||
let edit = rename_all(
|
||||
FeatureTester::builder()
|
||||
.files(vec![
|
||||
("main.bib", r#"@article{foo, bar = baz}"#),
|
||||
("main.tex", "\\addbibresource{main.bib}\n\\cite{foo}"),
|
||||
])
|
||||
.main("main.bib")
|
||||
.line(0)
|
||||
.character(9)
|
||||
.new_name("qux")
|
||||
.build()
|
||||
.rename(),
|
||||
);
|
||||
|
||||
assert_json_snapshot!(
|
||||
edit,
|
||||
{
|
||||
".changes.$key" => insta::dynamic_redaction(redact_uri),
|
||||
".changes" => insta::sorted_redaction(),
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_citation() {
|
||||
let edit = rename_all(
|
||||
FeatureTester::builder()
|
||||
.files(vec![
|
||||
("main.bib", r#"@article{foo, bar = baz}"#),
|
||||
("main.tex", "\\addbibresource{main.bib}\n\\cite{foo}"),
|
||||
])
|
||||
.main("main.tex")
|
||||
.line(1)
|
||||
.character(6)
|
||||
.new_name("qux")
|
||||
.build()
|
||||
.rename(),
|
||||
);
|
||||
|
||||
assert_json_snapshot!(
|
||||
edit,
|
||||
{
|
||||
".changes.$key" => insta::dynamic_redaction(redact_uri),
|
||||
".changes" => insta::sorted_redaction(),
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_label() {
|
||||
let edit = rename_all(
|
||||
FeatureTester::builder()
|
||||
.files(vec![
|
||||
("foo.tex", r#"\label{foo}\include{bar}"#),
|
||||
("bar.tex", r#"\ref{foo}"#),
|
||||
("baz.tex", r#"\ref{foo}"#),
|
||||
])
|
||||
.main("foo.tex")
|
||||
.line(0)
|
||||
.character(7)
|
||||
.new_name("bar")
|
||||
.build()
|
||||
.rename(),
|
||||
);
|
||||
|
||||
assert_json_snapshot!(
|
||||
edit,
|
||||
{
|
||||
".changes.$key" => insta::dynamic_redaction(redact_uri),
|
||||
".changes" => insta::sorted_redaction(),
|
||||
}
|
||||
);
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue