1use hir::{ItemInNs, ModuleDef};
3use ide_db::imports::{
4 import_assets::{ImportAssets, LocatedImport},
5 insert_use::ImportScope,
6};
7use itertools::Itertools;
8use syntax::{AstNode, SyntaxNode, ast};
9
10use crate::{
11 Completions,
12 config::AutoImportExclusionType,
13 context::{
14 CompletionContext, DotAccess, PathCompletionCtx, PathKind, PatternContext, Qualified,
15 TypeLocation,
16 },
17 render::{RenderContext, render_resolution_with_import, render_resolution_with_import_pat},
18};
19
20pub(crate) fn import_on_the_fly_path(
112 acc: &mut Completions,
113 ctx: &CompletionContext<'_>,
114 path_ctx: &PathCompletionCtx<'_>,
115) -> Option<()> {
116 if !ctx.config.enable_imports_on_the_fly {
117 return None;
118 }
119 let qualified = match path_ctx {
120 PathCompletionCtx {
121 kind:
122 PathKind::Expr { .. }
123 | PathKind::Type { .. }
124 | PathKind::Attr { .. }
125 | PathKind::Derive { .. }
126 | PathKind::Item { .. }
127 | PathKind::Pat { .. },
128 qualified,
129 ..
130 } => qualified,
131 _ => return None,
132 };
133 let potential_import_name = import_name(ctx);
134 let qualifier = match qualified {
135 Qualified::With { path, .. } => Some(path.clone()),
136 Qualified::TypeAnchor { .. } => return None,
137 Qualified::No | Qualified::Absolute => None,
138 };
139 let import_assets = import_assets_for_path(
140 ctx,
141 Some(&path_ctx.path),
142 &potential_import_name,
143 qualifier.clone(),
144 )?;
145
146 import_on_the_fly(
147 acc,
148 ctx,
149 path_ctx,
150 import_assets,
151 qualifier.map(|it| it.syntax().clone()).or_else(|| ctx.original_token.parent())?,
152 potential_import_name,
153 )
154}
155
156pub(crate) fn import_on_the_fly_pat(
157 acc: &mut Completions,
158 ctx: &CompletionContext<'_>,
159 pattern_ctx: &PatternContext,
160) -> Option<()> {
161 if !ctx.config.enable_imports_on_the_fly {
162 return None;
163 }
164 if let PatternContext { record_pat: Some(_), .. } = pattern_ctx {
165 return None;
166 }
167
168 let potential_import_name = import_name(ctx);
169 let import_assets = import_assets_for_path(ctx, None, &potential_import_name, None)?;
170
171 import_on_the_fly_pat_(
172 acc,
173 ctx,
174 pattern_ctx,
175 import_assets,
176 ctx.original_token.parent()?,
177 potential_import_name,
178 )
179}
180
181pub(crate) fn import_on_the_fly_dot(
182 acc: &mut Completions,
183 ctx: &CompletionContext<'_>,
184 dot_access: &DotAccess<'_>,
185) -> Option<()> {
186 if !ctx.config.enable_imports_on_the_fly {
187 return None;
188 }
189 let receiver = dot_access.receiver.as_ref()?;
190 let ty = dot_access.receiver_ty.as_ref()?;
191 let potential_import_name = import_name(ctx);
192 let import_assets = ImportAssets::for_fuzzy_method_call(
193 ctx.module,
194 ty.original.clone(),
195 potential_import_name.clone(),
196 receiver.syntax().clone(),
197 )?;
198
199 import_on_the_fly_method(
200 acc,
201 ctx,
202 dot_access,
203 import_assets,
204 receiver.syntax().clone(),
205 potential_import_name,
206 )
207}
208
209fn import_on_the_fly(
210 acc: &mut Completions,
211 ctx: &CompletionContext<'_>,
212 path_ctx @ PathCompletionCtx { kind, .. }: &PathCompletionCtx<'_>,
213 import_assets: ImportAssets<'_>,
214 position: SyntaxNode,
215 potential_import_name: String,
216) -> Option<()> {
217 let _p = tracing::info_span!("import_on_the_fly", ?potential_import_name).entered();
218
219 ImportScope::find_insert_use_container(&position, &ctx.sema)?;
220
221 let ns_filter = |import: &LocatedImport| {
222 match (kind, import.original_item) {
223 (PathKind::Vis { .. } | PathKind::Use, _) => false,
225 (_, ItemInNs::Types(hir::ModuleDef::Module(_))) => true,
227 (
229 PathKind::Expr { .. }
230 | PathKind::Type { .. }
231 | PathKind::Item { .. }
232 | PathKind::Pat { .. },
233 ItemInNs::Macros(mac),
234 ) => mac.is_fn_like(ctx.db),
235 (PathKind::Item { .. }, ..) => false,
236
237 (PathKind::Expr { .. }, ItemInNs::Types(_) | ItemInNs::Values(_)) => true,
238
239 (PathKind::Pat { .. }, ItemInNs::Types(_)) => true,
240 (PathKind::Pat { .. }, ItemInNs::Values(def)) => {
241 matches!(def, hir::ModuleDef::Const(_))
242 }
243
244 (PathKind::Type { location }, ItemInNs::Types(ty)) => {
245 if matches!(location, TypeLocation::TypeBound) {
246 matches!(ty, ModuleDef::Trait(_))
247 } else if matches!(location, TypeLocation::ImplTrait) {
248 matches!(ty, ModuleDef::Trait(_) | ModuleDef::Module(_))
249 } else {
250 true
251 }
252 }
253 (PathKind::Type { .. }, ItemInNs::Values(_)) => false,
254
255 (PathKind::Attr { .. }, ItemInNs::Macros(mac)) => mac.is_attr(ctx.db),
256 (PathKind::Attr { .. }, _) => false,
257
258 (PathKind::Derive { existing_derives }, ItemInNs::Macros(mac)) => {
259 mac.is_derive(ctx.db) && !existing_derives.contains(&mac)
260 }
261 (PathKind::Derive { .. }, _) => false,
262 }
263 };
264 let user_input_lowercased = potential_import_name.to_lowercase();
265
266 let import_cfg = ctx.config.import_path_config();
267
268 import_assets
269 .search_for_imports(&ctx.sema, import_cfg, ctx.config.insert_use.prefix_kind)
270 .filter(ns_filter)
271 .filter(|import| {
272 let original_item = &import.original_item;
273 !ctx.is_item_hidden(&import.item_to_import)
274 && !ctx.is_item_hidden(original_item)
275 && ctx.check_stability(original_item.attrs(ctx.db).as_ref())
276 })
277 .filter(|import| filter_excluded_flyimport(ctx, import))
278 .sorted_by(|a, b| {
279 let key = |import_path| {
280 (
281 compute_fuzzy_completion_order_key(import_path, &user_input_lowercased),
282 import_path,
283 )
284 };
285 key(&a.import_path).cmp(&key(&b.import_path))
286 })
287 .filter_map(|import| {
288 render_resolution_with_import(RenderContext::new(ctx), path_ctx, import)
289 })
290 .map(|builder| builder.build(ctx.db))
291 .for_each(|item| acc.add(item));
292 Some(())
293}
294
295fn import_on_the_fly_pat_(
296 acc: &mut Completions,
297 ctx: &CompletionContext<'_>,
298 pattern_ctx: &PatternContext,
299 import_assets: ImportAssets<'_>,
300 position: SyntaxNode,
301 potential_import_name: String,
302) -> Option<()> {
303 let _p = tracing::info_span!("import_on_the_fly_pat_", ?potential_import_name).entered();
304
305 ImportScope::find_insert_use_container(&position, &ctx.sema)?;
306
307 let ns_filter = |import: &LocatedImport| match import.original_item {
308 ItemInNs::Macros(mac) => mac.is_fn_like(ctx.db),
309 ItemInNs::Types(_) => true,
310 ItemInNs::Values(def) => matches!(def, hir::ModuleDef::Const(_)),
311 };
312 let user_input_lowercased = potential_import_name.to_lowercase();
313 let cfg = ctx.config.import_path_config();
314
315 import_assets
316 .search_for_imports(&ctx.sema, cfg, ctx.config.insert_use.prefix_kind)
317 .filter(ns_filter)
318 .filter(|import| {
319 let original_item = &import.original_item;
320 !ctx.is_item_hidden(&import.item_to_import)
321 && !ctx.is_item_hidden(original_item)
322 && ctx.check_stability(original_item.attrs(ctx.db).as_ref())
323 })
324 .sorted_by(|a, b| {
325 let key = |import_path| {
326 (
327 compute_fuzzy_completion_order_key(import_path, &user_input_lowercased),
328 import_path,
329 )
330 };
331 key(&a.import_path).cmp(&key(&b.import_path))
332 })
333 .filter_map(|import| {
334 render_resolution_with_import_pat(RenderContext::new(ctx), pattern_ctx, import)
335 })
336 .map(|builder| builder.build(ctx.db))
337 .for_each(|item| acc.add(item));
338 Some(())
339}
340
341fn import_on_the_fly_method(
342 acc: &mut Completions,
343 ctx: &CompletionContext<'_>,
344 dot_access: &DotAccess<'_>,
345 import_assets: ImportAssets<'_>,
346 position: SyntaxNode,
347 potential_import_name: String,
348) -> Option<()> {
349 let _p = tracing::info_span!("import_on_the_fly_method", ?potential_import_name).entered();
350
351 ImportScope::find_insert_use_container(&position, &ctx.sema)?;
352
353 let user_input_lowercased = potential_import_name.to_lowercase();
354
355 let cfg = ctx.config.import_path_config();
356
357 import_assets
358 .search_for_imports(&ctx.sema, cfg, ctx.config.insert_use.prefix_kind)
359 .filter(|import| {
360 !ctx.is_item_hidden(&import.item_to_import)
361 && !ctx.is_item_hidden(&import.original_item)
362 })
363 .filter(|import| filter_excluded_flyimport(ctx, import))
364 .sorted_by(|a, b| {
365 let key = |import_path| {
366 (
367 compute_fuzzy_completion_order_key(import_path, &user_input_lowercased),
368 import_path,
369 )
370 };
371 key(&a.import_path).cmp(&key(&b.import_path))
372 })
373 .for_each(|import| {
374 if let ItemInNs::Values(hir::ModuleDef::Function(f)) = import.original_item {
375 acc.add_method_with_import(ctx, dot_access, f, import);
376 }
377 });
378 Some(())
379}
380
381fn filter_excluded_flyimport(ctx: &CompletionContext<'_>, import: &LocatedImport) -> bool {
382 let def = import.item_to_import.into_module_def();
383 let is_exclude_flyimport = ctx.exclude_flyimport.get(&def).copied();
384
385 if matches!(is_exclude_flyimport, Some(AutoImportExclusionType::Always))
386 || !import.complete_in_flyimport.0
387 {
388 return false;
389 }
390 let method_imported = import.item_to_import != import.original_item;
391 if method_imported
392 && (is_exclude_flyimport.is_some()
393 || ctx.exclude_flyimport.contains_key(&import.original_item.into_module_def()))
394 {
395 return false;
399 }
400 true
401}
402
403fn import_name(ctx: &CompletionContext<'_>) -> String {
404 let token_kind = ctx.token.kind();
405
406 if token_kind.is_any_identifier() { ctx.token.to_string() } else { String::new() }
407}
408
409fn import_assets_for_path<'db>(
410 ctx: &CompletionContext<'db>,
411 path: Option<&ast::Path>,
412 potential_import_name: &str,
413 qualifier: Option<ast::Path>,
414) -> Option<ImportAssets<'db>> {
415 let _p =
416 tracing::info_span!("import_assets_for_path", ?potential_import_name, ?qualifier).entered();
417
418 let fuzzy_name_length = potential_import_name.len();
419 let mut assets_for_path = ImportAssets::for_fuzzy_path(
420 ctx.module,
421 path,
422 qualifier,
423 potential_import_name.to_owned(),
424 &ctx.sema,
425 ctx.token.parent()?,
426 )?;
427 if fuzzy_name_length == 0 {
428 assets_for_path.path_fuzzy_name_to_exact();
430 } else if fuzzy_name_length < 3 {
431 cov_mark::hit!(flyimport_prefix_on_short_path);
432 assets_for_path.path_fuzzy_name_to_prefix();
433 }
434 Some(assets_for_path)
435}
436
437fn compute_fuzzy_completion_order_key(
438 proposed_mod_path: &hir::ModPath,
439 user_input_lowercased: &str,
440) -> usize {
441 cov_mark::hit!(certain_fuzzy_order_test);
442 let import_name = match proposed_mod_path.segments().last() {
443 Some(name) => name.as_str().to_ascii_lowercase(),
445 None => return usize::MAX,
446 };
447 match import_name.match_indices(user_input_lowercased).next() {
448 Some((first_matching_index, _)) => first_matching_index,
449 None => usize::MAX,
450 }
451}