rust-analyzer/crates/ide_completion/src/snippet.rs

182 lines
6.3 KiB
Rust
Raw Normal View History

2021-10-04 20:13:12 +00:00
//! User (postfix)-snippet definitions.
//!
2021-10-11 19:49:39 +00:00
//! Actual logic is implemented in [`crate::completions::postfix`] and [`crate::completions::snippet`] respectively.
use std::ops::Deref;
// Feature: User Snippet Completions
//
// rust-analyzer allows the user to define custom (postfix)-snippets that may depend on items to be accessible for the current scope to be applicable.
//
// A custom snippet can be defined by adding it to the `rust-analyzer.completion.snippets` object respectively.
//
// [source,json]
// ----
// {
// "rust-analyzer.completion.snippets": {
// "thread spawn": {
// "prefix": ["spawn", "tspawn"],
// "body": [
// "thread::spawn(move || {",
// "\t$0",
// ")};",
// ],
// "description": "Insert a thread::spawn call",
// "requires": "std::thread",
// "scope": "expr",
// }
// }
// }
// ----
//
// In the example above:
//
// * `"thread spawn"` is the name of the snippet.
//
// * `prefix` defines one or more trigger words that will trigger the snippets completion.
// Using `postfix` will instead create a postfix snippet.
//
// * `body` is one or more lines of content joined via newlines for the final output.
//
// * `description` is an optional description of the snippet, if unset the snippet name will be used.
//
// * `requires` is an optional list of item paths that have to be resolvable in the current crate where the completion is rendered.
// On failure of resolution the snippet won't be applicable, otherwise the snippet will insert an import for the items on insertion if
// the items aren't yet in scope.
//
// * `scope` is an optional filter for when the snippet should be applicable. Possible values are:
// ** for Snippet-Scopes: `expr`, `item` (default: `item`)
// ** for Postfix-Snippet-Scopes: `expr`, `type` (default: `expr`)
//
// The `body` field also has access to placeholders as visible in the example as `$0`.
// These placeholders take the form of `$number` or `${number:placeholder_text}` which can be traversed as tabstop in ascending order starting from 1,
// with `$0` being a special case that always comes last.
//
// There is also a special placeholder, `${receiver}`, which will be replaced by the receiver expression for postfix snippets, or nothing in case of normal snippets.
// It does not act as a tabstop.
2021-10-04 17:22:41 +00:00
use ide_db::helpers::{import_assets::LocatedImport, insert_use::ImportScope};
use itertools::Itertools;
use syntax::{ast, AstNode, GreenNode, SyntaxNode};
2021-10-04 17:22:41 +00:00
use crate::{context::CompletionContext, ImportEdit};
2021-10-11 19:49:39 +00:00
/// A snippet scope describing where a snippet may apply to.
/// These may differ slightly in meaning depending on the snippet trigger.
2021-10-04 17:22:41 +00:00
#[derive(Clone, Debug, PartialEq, Eq)]
pub enum SnippetScope {
Item,
Expr,
Type,
2021-10-04 17:22:41 +00:00
}
2021-10-11 19:49:39 +00:00
/// A user supplied snippet.
2021-10-04 17:22:41 +00:00
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct Snippet {
2021-10-11 19:49:39 +00:00
pub postfix_triggers: Box<[Box<str>]>,
pub prefix_triggers: Box<[Box<str>]>,
2021-10-04 17:22:41 +00:00
pub scope: SnippetScope,
2021-10-11 19:49:39 +00:00
pub description: Option<Box<str>>,
snippet: String,
// These are `ast::Path`'s but due to SyntaxNodes not being Send we store these
// and reconstruct them on demand instead. This is cheaper than reparsing them
// from strings
requires: Box<[GreenNode]>,
2021-10-04 17:22:41 +00:00
}
2021-10-04 17:22:41 +00:00
impl Snippet {
pub fn new(
prefix_triggers: &[String],
postfix_triggers: &[String],
2021-10-04 17:22:41 +00:00
snippet: &[String],
description: &str,
2021-10-04 17:22:41 +00:00
requires: &[String],
scope: SnippetScope,
2021-10-04 17:22:41 +00:00
) -> Option<Self> {
2021-10-11 19:49:39 +00:00
if prefix_triggers.is_empty() && postfix_triggers.is_empty() {
return None;
}
let (requires, snippet, description) = validate_snippet(snippet, description, requires)?;
2021-10-04 17:22:41 +00:00
Some(Snippet {
// Box::into doesn't work as that has a Copy bound 😒
2021-10-11 19:49:39 +00:00
postfix_triggers: postfix_triggers.iter().map(Deref::deref).map(Into::into).collect(),
prefix_triggers: prefix_triggers.iter().map(Deref::deref).map(Into::into).collect(),
scope,
2021-10-04 17:22:41 +00:00
snippet,
description,
requires,
2021-10-04 17:22:41 +00:00
})
}
2021-10-11 19:49:39 +00:00
/// Returns [`None`] if the required items do not resolve.
2021-10-04 17:22:41 +00:00
pub(crate) fn imports(
&self,
ctx: &CompletionContext,
import_scope: &ImportScope,
) -> Option<Vec<ImportEdit>> {
2021-10-04 17:22:41 +00:00
import_edits(ctx, import_scope, &self.requires)
}
pub fn snippet(&self) -> String {
self.snippet.replace("${receiver}", "")
2021-10-04 17:22:41 +00:00
}
pub fn postfix_snippet(&self, receiver: &str) -> String {
self.snippet.replace("${receiver}", receiver)
2021-10-04 17:22:41 +00:00
}
}
fn import_edits(
ctx: &CompletionContext,
import_scope: &ImportScope,
requires: &[GreenNode],
) -> Option<Vec<ImportEdit>> {
let resolve = |import: &GreenNode| {
let path = ast::Path::cast(SyntaxNode::new_root(import.clone()))?;
let item = match ctx.scope.speculative_resolve(&path)? {
hir::PathResolution::Macro(mac) => mac.into(),
hir::PathResolution::Def(def) => def.into(),
_ => return None,
};
let path = ctx.scope.module()?.find_use_path_prefixed(
ctx.db,
item,
ctx.config.insert_use.prefix_kind,
)?;
Some((path.len() > 1).then(|| ImportEdit {
import: LocatedImport::new(path.clone(), item, item, None),
scope: import_scope.clone(),
}))
2021-10-04 17:22:41 +00:00
};
let mut res = Vec::with_capacity(requires.len());
for import in requires {
match resolve(import) {
Some(first) => res.extend(first),
None => return None,
2021-10-04 17:22:41 +00:00
}
}
Some(res)
}
fn validate_snippet(
snippet: &[String],
description: &str,
requires: &[String],
) -> Option<(Box<[GreenNode]>, String, Option<Box<str>>)> {
let mut imports = Vec::with_capacity(requires.len());
for path in requires.iter() {
let path = ast::Path::parse(path).ok()?;
let valid_use_path = path.segments().all(|seg| {
matches!(seg.kind(), Some(ast::PathSegmentKind::Name(_)))
|| seg.generic_arg_list().is_none()
});
if !valid_use_path {
return None;
}
let green = path.syntax().green().into_owned();
imports.push(green);
}
let snippet = snippet.iter().join("\n");
2021-10-11 19:49:39 +00:00
let description = if description.is_empty() { None } else { Some(description.into()) };
Some((imports.into_boxed_slice(), snippet, description))
2021-10-04 17:22:41 +00:00
}