aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_analysis/src/completion
diff options
context:
space:
mode:
Diffstat (limited to 'crates/ra_analysis/src/completion')
-rw-r--r--crates/ra_analysis/src/completion/mod.rs54
-rw-r--r--crates/ra_analysis/src/completion/reference_completion.rs106
2 files changed, 63 insertions, 97 deletions
diff --git a/crates/ra_analysis/src/completion/mod.rs b/crates/ra_analysis/src/completion/mod.rs
index 2e082705e..e5ba92acd 100644
--- a/crates/ra_analysis/src/completion/mod.rs
+++ b/crates/ra_analysis/src/completion/mod.rs
@@ -2,18 +2,16 @@ mod reference_completion;
2 2
3use ra_editor::find_node_at_offset; 3use ra_editor::find_node_at_offset;
4use ra_syntax::{ 4use ra_syntax::{
5 algo::find_leaf_at_offset,
6 algo::visit::{visitor_ctx, VisitorCtx}, 5 algo::visit::{visitor_ctx, VisitorCtx},
7 ast, 6 ast,
8 AstNode, AtomEdit, 7 AstNode, AtomEdit,
9 SyntaxNodeRef, 8 SyntaxNodeRef,
10}; 9};
10use ra_db::SyntaxDatabase;
11use rustc_hash::{FxHashMap}; 11use rustc_hash::{FxHashMap};
12 12
13use crate::{ 13use crate::{
14 db::{self, SyntaxDatabase}, 14 db,
15 descriptors::{DescriptorDatabase, module::ModuleSource},
16 input::{FilesDatabase},
17 Cancelable, FilePosition 15 Cancelable, FilePosition
18}; 16};
19 17
@@ -31,39 +29,21 @@ pub(crate) fn completions(
31 db: &db::RootDatabase, 29 db: &db::RootDatabase,
32 position: FilePosition, 30 position: FilePosition,
33) -> Cancelable<Option<Vec<CompletionItem>>> { 31) -> Cancelable<Option<Vec<CompletionItem>>> {
34 let original_file = db.file_syntax(position.file_id); 32 let original_file = db.source_file(position.file_id);
35 // Insert a fake ident to get a valid parse tree 33 // Insert a fake ident to get a valid parse tree
36 let file = { 34 let file = {
37 let edit = AtomEdit::insert(position.offset, "intellijRulezz".to_string()); 35 let edit = AtomEdit::insert(position.offset, "intellijRulezz".to_string());
38 original_file.reparse(&edit) 36 original_file.reparse(&edit)
39 }; 37 };
40 38
41 let leaf = match find_leaf_at_offset(original_file.syntax(), position.offset).left_biased() { 39 let module = ctry!(hir::Module::guess_from_position(db, position)?);
42 None => return Ok(None),
43 Some(it) => it,
44 };
45 let source_root_id = db.file_source_root(position.file_id);
46 let module_tree = db.module_tree(source_root_id)?;
47 let module_source = ModuleSource::for_node(position.file_id, leaf);
48 let module_id = match module_tree.any_module_for_source(module_source) {
49 None => return Ok(None),
50 Some(it) => it,
51 };
52 40
53 let mut res = Vec::new(); 41 let mut res = Vec::new();
54 let mut has_completions = false; 42 let mut has_completions = false;
55 // First, let's try to complete a reference to some declaration. 43 // First, let's try to complete a reference to some declaration.
56 if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(file.syntax(), position.offset) { 44 if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(file.syntax(), position.offset) {
57 has_completions = true; 45 has_completions = true;
58 reference_completion::completions( 46 reference_completion::completions(&mut res, db, &module, &file, name_ref)?;
59 &mut res,
60 db,
61 source_root_id,
62 &module_tree,
63 module_id,
64 &file,
65 name_ref,
66 )?;
67 // special case, `trait T { fn foo(i_am_a_name_ref) {} }` 47 // special case, `trait T { fn foo(i_am_a_name_ref) {} }`
68 if is_node::<ast::Param>(name_ref.syntax()) { 48 if is_node::<ast::Param>(name_ref.syntax()) {
69 param_completions(name_ref.syntax(), &mut res); 49 param_completions(name_ref.syntax(), &mut res);
@@ -219,9 +199,9 @@ mod tests {
219 <|> 199 <|>
220 } 200 }
221 ", 201 ",
222 r#"[CompletionItem { label: "Foo", lookup: None, snippet: None }, 202 r#"[CompletionItem { label: "quux", lookup: None, snippet: None },
223 CompletionItem { label: "Baz", lookup: None, snippet: None }, 203 CompletionItem { label: "Foo", lookup: None, snippet: None },
224 CompletionItem { label: "quux", lookup: None, snippet: None }]"#, 204 CompletionItem { label: "Baz", lookup: None, snippet: None }]"#,
225 ); 205 );
226 } 206 }
227 207
@@ -236,6 +216,20 @@ mod tests {
236 } 216 }
237 217
238 #[test] 218 #[test]
219 fn test_completion_self_path() {
220 check_scope_completion(
221 r"
222 use self::m::<|>;
223
224 mod m {
225 struct Bar;
226 }
227 ",
228 r#"[CompletionItem { label: "Bar", lookup: None, snippet: None }]"#,
229 );
230 }
231
232 #[test]
239 fn test_completion_mod_scope_nested() { 233 fn test_completion_mod_scope_nested() {
240 check_scope_completion( 234 check_scope_completion(
241 r" 235 r"
@@ -245,8 +239,8 @@ mod tests {
245 fn quux() { <|> } 239 fn quux() { <|> }
246 } 240 }
247 ", 241 ",
248 r#"[CompletionItem { label: "Bar", lookup: None, snippet: None }, 242 r#"[CompletionItem { label: "quux", lookup: None, snippet: None },
249 CompletionItem { label: "quux", lookup: None, snippet: None }]"#, 243 CompletionItem { label: "Bar", lookup: None, snippet: None }]"#,
250 ); 244 );
251 } 245 }
252 246
diff --git a/crates/ra_analysis/src/completion/reference_completion.rs b/crates/ra_analysis/src/completion/reference_completion.rs
index 6c5fd0be6..e1a2d5241 100644
--- a/crates/ra_analysis/src/completion/reference_completion.rs
+++ b/crates/ra_analysis/src/completion/reference_completion.rs
@@ -6,23 +6,23 @@ use ra_syntax::{
6 ast::{self, LoopBodyOwner}, 6 ast::{self, LoopBodyOwner},
7 SyntaxKind::*, 7 SyntaxKind::*,
8}; 8};
9use hir::{
10 self,
11 FnScopes,
12 Def,
13 Path,
14};
9 15
10use crate::{ 16use crate::{
11 db::RootDatabase, 17 db::RootDatabase,
12 input::{SourceRootId},
13 completion::CompletionItem, 18 completion::CompletionItem,
14 descriptors::module::{ModuleId, ModuleTree},
15 descriptors::function::FnScopes,
16 descriptors::DescriptorDatabase,
17 Cancelable 19 Cancelable
18}; 20};
19 21
20pub(super) fn completions( 22pub(super) fn completions(
21 acc: &mut Vec<CompletionItem>, 23 acc: &mut Vec<CompletionItem>,
22 db: &RootDatabase, 24 db: &RootDatabase,
23 source_root_id: SourceRootId, 25 module: &hir::Module,
24 module_tree: &ModuleTree,
25 module_id: ModuleId,
26 file: &SourceFileNode, 26 file: &SourceFileNode,
27 name_ref: ast::NameRef, 27 name_ref: ast::NameRef,
28) -> Cancelable<()> { 28) -> Cancelable<()> {
@@ -40,25 +40,28 @@ pub(super) fn completions(
40 complete_expr_snippets(acc); 40 complete_expr_snippets(acc);
41 } 41 }
42 42
43 let module_scope = db.module_scope(source_root_id, module_id)?; 43 let module_scope = module.scope(db)?;
44 acc.extend( 44 acc.extend(
45 module_scope 45 module_scope
46 .entries() 46 .entries()
47 .iter() 47 .filter(|(_name, res)| {
48 .filter(|entry| {
49 // Don't expose this item 48 // Don't expose this item
50 !entry.ptr().range().is_subrange(&name_ref.syntax().range()) 49 match res.import {
50 None => true,
51 Some(import) => {
52 let range = import.range(db, module.source().file_id());
53 !range.is_subrange(&name_ref.syntax().range())
54 }
55 }
51 }) 56 })
52 .map(|entry| CompletionItem { 57 .map(|(name, _res)| CompletionItem {
53 label: entry.name().to_string(), 58 label: name.to_string(),
54 lookup: None, 59 lookup: None,
55 snippet: None, 60 snippet: None,
56 }), 61 }),
57 ); 62 );
58 } 63 }
59 NameRefKind::CratePath(path) => { 64 NameRefKind::Path(path) => complete_path(acc, db, module, path)?,
60 complete_path(acc, db, source_root_id, module_tree, module_id, path)?
61 }
62 NameRefKind::BareIdentInMod => { 65 NameRefKind::BareIdentInMod => {
63 let name_range = name_ref.syntax().range(); 66 let name_range = name_ref.syntax().range();
64 let top_node = name_ref 67 let top_node = name_ref
@@ -82,8 +85,8 @@ enum NameRefKind<'a> {
82 LocalRef { 85 LocalRef {
83 enclosing_fn: Option<ast::FnDef<'a>>, 86 enclosing_fn: Option<ast::FnDef<'a>>,
84 }, 87 },
85 /// NameRef is the last segment in crate:: path 88 /// NameRef is the last segment in some path
86 CratePath(Vec<ast::NameRef<'a>>), 89 Path(Path),
87 /// NameRef is bare identifier at the module's root. 90 /// NameRef is bare identifier at the module's root.
88 /// Used for keyword completion 91 /// Used for keyword completion
89 BareIdentInMod, 92 BareIdentInMod,
@@ -105,8 +108,10 @@ fn classify_name_ref(name_ref: ast::NameRef) -> Option<NameRefKind> {
105 let parent = name_ref.syntax().parent()?; 108 let parent = name_ref.syntax().parent()?;
106 if let Some(segment) = ast::PathSegment::cast(parent) { 109 if let Some(segment) = ast::PathSegment::cast(parent) {
107 let path = segment.parent_path(); 110 let path = segment.parent_path();
108 if let Some(crate_path) = crate_path(path) { 111 if let Some(path) = Path::from_ast(path) {
109 return Some(NameRefKind::CratePath(crate_path)); 112 if !path.is_ident() {
113 return Some(NameRefKind::Path(path));
114 }
110 } 115 }
111 if path.qualifier().is_none() { 116 if path.qualifier().is_none() {
112 let enclosing_fn = name_ref 117 let enclosing_fn = name_ref
@@ -120,32 +125,6 @@ fn classify_name_ref(name_ref: ast::NameRef) -> Option<NameRefKind> {
120 None 125 None
121} 126}
122 127
123fn crate_path(mut path: ast::Path) -> Option<Vec<ast::NameRef>> {
124 let mut res = Vec::new();
125 loop {
126 let segment = path.segment()?;
127 match segment.kind()? {
128 ast::PathSegmentKind::Name(name) => res.push(name),
129 ast::PathSegmentKind::CrateKw => break,
130 ast::PathSegmentKind::SelfKw | ast::PathSegmentKind::SuperKw => return None,
131 }
132 path = qualifier(path)?;
133 }
134 res.reverse();
135 return Some(res);
136
137 fn qualifier(path: ast::Path) -> Option<ast::Path> {
138 if let Some(q) = path.qualifier() {
139 return Some(q);
140 }
141 // TODO: this bottom up traversal is not too precise.
142 // Should we handle do a top-down analysiss, recording results?
143 let use_tree_list = path.syntax().ancestors().find_map(ast::UseTreeList::cast)?;
144 let use_tree = use_tree_list.parent_use_tree();
145 use_tree.path()
146 }
147}
148
149fn complete_fn(name_ref: ast::NameRef, scopes: &FnScopes, acc: &mut Vec<CompletionItem>) { 128fn complete_fn(name_ref: ast::NameRef, scopes: &FnScopes, acc: &mut Vec<CompletionItem>) {
150 let mut shadowed = FxHashSet::default(); 129 let mut shadowed = FxHashSet::default();
151 acc.extend( 130 acc.extend(
@@ -171,18 +150,24 @@ fn complete_fn(name_ref: ast::NameRef, scopes: &FnScopes, acc: &mut Vec<Completi
171fn complete_path( 150fn complete_path(
172 acc: &mut Vec<CompletionItem>, 151 acc: &mut Vec<CompletionItem>,
173 db: &RootDatabase, 152 db: &RootDatabase,
174 source_root_id: SourceRootId, 153 module: &hir::Module,
175 module_tree: &ModuleTree, 154 mut path: Path,
176 module_id: ModuleId,
177 crate_path: Vec<ast::NameRef>,
178) -> Cancelable<()> { 155) -> Cancelable<()> {
179 let target_module_id = match find_target_module(module_tree, module_id, crate_path) { 156 if path.segments.is_empty() {
157 return Ok(());
158 }
159 path.segments.pop();
160 let def_id = match module.resolve_path(db, path)? {
180 None => return Ok(()), 161 None => return Ok(()),
181 Some(it) => it, 162 Some(it) => it,
182 }; 163 };
183 let module_scope = db.module_scope(source_root_id, target_module_id)?; 164 let target_module = match def_id.resolve(db)? {
184 let completions = module_scope.entries().iter().map(|entry| CompletionItem { 165 Def::Module(it) => it,
185 label: entry.name().to_string(), 166 Def::Item => return Ok(()),
167 };
168 let module_scope = target_module.scope(db)?;
169 let completions = module_scope.entries().map(|(name, _res)| CompletionItem {
170 label: name.to_string(),
186 lookup: None, 171 lookup: None,
187 snippet: None, 172 snippet: None,
188 }); 173 });
@@ -190,19 +175,6 @@ fn complete_path(
190 Ok(()) 175 Ok(())
191} 176}
192 177
193fn find_target_module(
194 module_tree: &ModuleTree,
195 module_id: ModuleId,
196 mut crate_path: Vec<ast::NameRef>,
197) -> Option<ModuleId> {
198 crate_path.pop();
199 let mut target_module = module_id.root(&module_tree);
200 for name in crate_path {
201 target_module = target_module.child(module_tree, name.text().as_str())?;
202 }
203 Some(target_module)
204}
205
206fn complete_mod_item_snippets(acc: &mut Vec<CompletionItem>) { 178fn complete_mod_item_snippets(acc: &mut Vec<CompletionItem>) {
207 acc.push(CompletionItem { 179 acc.push(CompletionItem {
208 label: "tfn".to_string(), 180 label: "tfn".to_string(),