diff options
Diffstat (limited to 'crates/ra_analysis/src/completion')
-rw-r--r-- | crates/ra_analysis/src/completion/mod.rs | 54 | ||||
-rw-r--r-- | crates/ra_analysis/src/completion/reference_completion.rs | 106 |
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 | ||
3 | use ra_editor::find_node_at_offset; | 3 | use ra_editor::find_node_at_offset; |
4 | use ra_syntax::{ | 4 | use 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 | }; |
10 | use ra_db::SyntaxDatabase; | ||
11 | use rustc_hash::{FxHashMap}; | 11 | use rustc_hash::{FxHashMap}; |
12 | 12 | ||
13 | use crate::{ | 13 | use 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 | }; |
9 | use hir::{ | ||
10 | self, | ||
11 | FnScopes, | ||
12 | Def, | ||
13 | Path, | ||
14 | }; | ||
9 | 15 | ||
10 | use crate::{ | 16 | use 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 | ||
20 | pub(super) fn completions( | 22 | pub(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 | ||
123 | fn 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 | |||
149 | fn complete_fn(name_ref: ast::NameRef, scopes: &FnScopes, acc: &mut Vec<CompletionItem>) { | 128 | fn 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 | |||
171 | fn complete_path( | 150 | fn 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 | ||
193 | fn 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 | |||
206 | fn complete_mod_item_snippets(acc: &mut Vec<CompletionItem>) { | 178 | fn 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(), |