1769: fix renaming of modules r=matklad a=matklad



Co-authored-by: Aleksey Kladov <aleksey.kladov@gmail.com>
This commit is contained in:
bors[bot] 2019-09-05 18:46:14 +00:00 committed by GitHub
commit 9856096a80
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 92 additions and 80 deletions

View file

@ -466,7 +466,7 @@ impl Analysis {
&self, &self,
position: FilePosition, position: FilePosition,
) -> Cancelable<Option<ReferenceSearchResult>> { ) -> Cancelable<Option<ReferenceSearchResult>> {
self.with_db(|db| references::find_all_refs(db, position)) self.with_db(|db| references::find_all_refs(db, position).map(|it| it.info))
} }
/// Returns a short text describing element at position. /// Returns a short text describing element at position.
@ -536,7 +536,7 @@ impl Analysis {
&self, &self,
position: FilePosition, position: FilePosition,
new_name: &str, new_name: &str,
) -> Cancelable<Option<SourceChange>> { ) -> Cancelable<Option<RangeInfo<SourceChange>>> {
self.with_db(|db| references::rename(db, position, new_name)) self.with_db(|db| references::rename(db, position, new_name))
} }

View file

@ -4,7 +4,7 @@ use ra_syntax::{algo::find_node_at_offset, ast, AstNode, SourceFile, SyntaxNode}
use relative_path::{RelativePath, RelativePathBuf}; use relative_path::{RelativePath, RelativePathBuf};
use crate::{ use crate::{
db::RootDatabase, FileId, FilePosition, FileRange, FileSystemEdit, NavigationTarget, db::RootDatabase, FileId, FilePosition, FileRange, FileSystemEdit, NavigationTarget, RangeInfo,
SourceChange, SourceFileEdit, TextRange, SourceChange, SourceFileEdit, TextRange,
}; };
@ -48,9 +48,9 @@ impl IntoIterator for ReferenceSearchResult {
pub(crate) fn find_all_refs( pub(crate) fn find_all_refs(
db: &RootDatabase, db: &RootDatabase,
position: FilePosition, position: FilePosition,
) -> Option<ReferenceSearchResult> { ) -> Option<RangeInfo<ReferenceSearchResult>> {
let parse = db.parse(position.file_id); let parse = db.parse(position.file_id);
let (binding, analyzer) = find_binding(db, &parse.tree(), position)?; let RangeInfo { range, info: (binding, analyzer) } = find_binding(db, &parse.tree(), position)?;
let declaration = NavigationTarget::from_bind_pat(position.file_id, &binding); let declaration = NavigationTarget::from_bind_pat(position.file_id, &binding);
let references = analyzer let references = analyzer
@ -59,24 +59,26 @@ pub(crate) fn find_all_refs(
.map(move |ref_desc| FileRange { file_id: position.file_id, range: ref_desc.range }) .map(move |ref_desc| FileRange { file_id: position.file_id, range: ref_desc.range })
.collect::<Vec<_>>(); .collect::<Vec<_>>();
return Some(ReferenceSearchResult { declaration, references }); return Some(RangeInfo::new(range, ReferenceSearchResult { declaration, references }));
fn find_binding<'a>( fn find_binding<'a>(
db: &RootDatabase, db: &RootDatabase,
source_file: &SourceFile, source_file: &SourceFile,
position: FilePosition, position: FilePosition,
) -> Option<(ast::BindPat, hir::SourceAnalyzer)> { ) -> Option<RangeInfo<(ast::BindPat, hir::SourceAnalyzer)>> {
let syntax = source_file.syntax(); let syntax = source_file.syntax();
if let Some(binding) = find_node_at_offset::<ast::BindPat>(syntax, position.offset) { if let Some(binding) = find_node_at_offset::<ast::BindPat>(syntax, position.offset) {
let range = binding.syntax().text_range();
let analyzer = hir::SourceAnalyzer::new(db, position.file_id, binding.syntax(), None); let analyzer = hir::SourceAnalyzer::new(db, position.file_id, binding.syntax(), None);
return Some((binding, analyzer)); return Some(RangeInfo::new(range, (binding, analyzer)));
}; };
let name_ref = find_node_at_offset::<ast::NameRef>(syntax, position.offset)?; let name_ref = find_node_at_offset::<ast::NameRef>(syntax, position.offset)?;
let range = name_ref.syntax().text_range();
let analyzer = hir::SourceAnalyzer::new(db, position.file_id, name_ref.syntax(), None); let analyzer = hir::SourceAnalyzer::new(db, position.file_id, name_ref.syntax(), None);
let resolved = analyzer.resolve_local_name(&name_ref)?; let resolved = analyzer.resolve_local_name(&name_ref)?;
if let Either::A(ptr) = resolved.ptr() { if let Either::A(ptr) = resolved.ptr() {
if let ast::Pat::BindPat(binding) = ptr.to_node(source_file.syntax()) { if let ast::Pat::BindPat(binding) = ptr.to_node(source_file.syntax()) {
return Some((binding, analyzer)); return Some(RangeInfo::new(range, (binding, analyzer)));
} }
} }
None None
@ -87,12 +89,14 @@ pub(crate) fn rename(
db: &RootDatabase, db: &RootDatabase,
position: FilePosition, position: FilePosition,
new_name: &str, new_name: &str,
) -> Option<SourceChange> { ) -> Option<RangeInfo<SourceChange>> {
let parse = db.parse(position.file_id); let parse = db.parse(position.file_id);
if let Some((ast_name, ast_module)) = if let Some((ast_name, ast_module)) =
find_name_and_module_at_offset(parse.tree().syntax(), position) find_name_and_module_at_offset(parse.tree().syntax(), position)
{ {
let range = ast_name.syntax().text_range();
rename_mod(db, &ast_name, &ast_module, position, new_name) rename_mod(db, &ast_name, &ast_module, position, new_name)
.map(|info| RangeInfo::new(range, info))
} else { } else {
rename_reference(db, position, new_name) rename_reference(db, position, new_name)
} }
@ -107,7 +111,7 @@ fn find_name_and_module_at_offset(
Some((ast_name, ast_module)) Some((ast_name, ast_module))
} }
fn source_edit_from_fileid_range( fn source_edit_from_file_id_range(
file_id: FileId, file_id: FileId,
range: TextRange, range: TextRange,
new_name: &str, new_name: &str,
@ -179,19 +183,19 @@ fn rename_reference(
db: &RootDatabase, db: &RootDatabase,
position: FilePosition, position: FilePosition,
new_name: &str, new_name: &str,
) -> Option<SourceChange> { ) -> Option<RangeInfo<SourceChange>> {
let refs = find_all_refs(db, position)?; let RangeInfo { range, info: refs } = find_all_refs(db, position)?;
let edit = refs let edit = refs
.into_iter() .into_iter()
.map(|range| source_edit_from_fileid_range(range.file_id, range.range, new_name)) .map(|range| source_edit_from_file_id_range(range.file_id, range.range, new_name))
.collect::<Vec<_>>(); .collect::<Vec<_>>();
if edit.is_empty() { if edit.is_empty() {
return None; return None;
} }
Some(SourceChange::source_file_edits("rename", edit)) Some(RangeInfo::new(range, SourceChange::source_file_edits("rename", edit)))
} }
#[cfg(test)] #[cfg(test)]
@ -342,8 +346,11 @@ mod tests {
let new_name = "foo2"; let new_name = "foo2";
let source_change = analysis.rename(position, new_name).unwrap(); let source_change = analysis.rename(position, new_name).unwrap();
assert_debug_snapshot!(&source_change, assert_debug_snapshot!(&source_change,
@r#"Some( @r###"
SourceChange { Some(
RangeInfo {
range: [4; 7),
info: SourceChange {
label: "rename", label: "rename",
source_file_edits: [ source_file_edits: [
SourceFileEdit { SourceFileEdit {
@ -373,7 +380,9 @@ mod tests {
], ],
cursor_position: None, cursor_position: None,
}, },
)"#); },
)
"###);
} }
#[test] #[test]
@ -389,8 +398,11 @@ mod tests {
let new_name = "foo2"; let new_name = "foo2";
let source_change = analysis.rename(position, new_name).unwrap(); let source_change = analysis.rename(position, new_name).unwrap();
assert_debug_snapshot!(&source_change, assert_debug_snapshot!(&source_change,
@r###"Some( @r###"
SourceChange { Some(
RangeInfo {
range: [4; 7),
info: SourceChange {
label: "rename", label: "rename",
source_file_edits: [ source_file_edits: [
SourceFileEdit { SourceFileEdit {
@ -420,7 +432,9 @@ mod tests {
], ],
cursor_position: None, cursor_position: None,
}, },
)"### },
)
"###
); );
} }
@ -430,7 +444,7 @@ mod tests {
let mut text_edit_builder = ra_text_edit::TextEditBuilder::default(); let mut text_edit_builder = ra_text_edit::TextEditBuilder::default();
let mut file_id: Option<FileId> = None; let mut file_id: Option<FileId> = None;
if let Some(change) = source_change { if let Some(change) = source_change {
for edit in change.source_file_edits { for edit in change.info.source_file_edits {
file_id = Some(edit.file_id); file_id = Some(edit.file_id);
for atom in edit.edit.as_atoms() { for atom in edit.edit.as_atoms() {
text_edit_builder.replace(atom.delete, atom.insert.clone()); text_edit_builder.replace(atom.delete, atom.insert.clone());

View file

@ -460,18 +460,16 @@ pub fn handle_prepare_rename(
// We support renaming references like handle_rename does. // We support renaming references like handle_rename does.
// In the future we may want to reject the renaming of things like keywords here too. // In the future we may want to reject the renaming of things like keywords here too.
let refs = match world.analysis().find_all_refs(position)? { let optional_change = world.analysis().rename(position, "dummy")?;
let range = match optional_change {
None => return Ok(None), None => return Ok(None),
Some(refs) => refs, Some(it) => it.range,
}; };
// Refs should always have a declaration
let r = refs.declaration();
let file_id = params.text_document.try_conv_with(&world)?; let file_id = params.text_document.try_conv_with(&world)?;
let line_index = world.analysis().file_line_index(file_id)?; let line_index = world.analysis().file_line_index(file_id)?;
let loc = to_location(r.file_id(), r.range(), &world, &line_index)?; let range = range.conv_with(&line_index);
Ok(Some(PrepareRenameResponse::Range(range)))
Ok(Some(PrepareRenameResponse::Range(loc.range)))
} }
pub fn handle_rename(world: WorldSnapshot, params: RenameParams) -> Result<Option<WorkspaceEdit>> { pub fn handle_rename(world: WorldSnapshot, params: RenameParams) -> Result<Option<WorkspaceEdit>> {
@ -488,7 +486,7 @@ pub fn handle_rename(world: WorldSnapshot, params: RenameParams) -> Result<Optio
let optional_change = world.analysis().rename(position, &*params.new_name)?; let optional_change = world.analysis().rename(position, &*params.new_name)?;
let change = match optional_change { let change = match optional_change {
None => return Ok(None), None => return Ok(None),
Some(it) => it, Some(it) => it.info,
}; };
let source_change_req = change.try_conv_with(&world)?; let source_change_req = change.try_conv_with(&world)?;