901: Add basic support for showing fn signature when hovering r=matklad a=vipentti

This adds basic support for displaying function signature when hovering over a usage of a function. 

Additionally refactored `hover` to return `HoverResult` to ease with testing and in general to be more robust.

Co-authored-by: Ville Penttinen <villem.penttinen@gmail.com>
This commit is contained in:
bors[bot] 2019-02-27 11:50:38 +00:00
commit 1927eb088a
4 changed files with 179 additions and 21 deletions

View file

@ -110,7 +110,7 @@ pub(crate) fn reference_definition(
Approximate(navs) Approximate(navs)
} }
fn name_definition( pub(crate) fn name_definition(
db: &RootDatabase, db: &RootDatabase,
file_id: FileId, file_id: FileId,
name: &ast::Name, name: &ast::Name,

View file

@ -1,14 +1,74 @@
use ra_db::SourceDatabase; use ra_db::SourceDatabase;
use ra_syntax::{ use ra_syntax::{
AstNode, SyntaxNode, TreeArc, ast, AstNode, SyntaxNode, TreeArc, ast::{self, NameOwner, VisibilityOwner},
algo::{find_covering_node, find_node_at_offset, find_leaf_at_offset, visit::{visitor, Visitor}}, algo::{find_covering_node, find_node_at_offset, find_leaf_at_offset, visit::{visitor, Visitor}},
}; };
use crate::{db::RootDatabase, RangeInfo, FilePosition, FileRange, NavigationTarget}; use crate::{db::RootDatabase, RangeInfo, FilePosition, FileRange, NavigationTarget};
pub(crate) fn hover(db: &RootDatabase, position: FilePosition) -> Option<RangeInfo<String>> { /// Contains the results when hovering over an item
#[derive(Debug, Clone)]
pub struct HoverResult {
results: Vec<String>,
exact: bool,
}
impl HoverResult {
pub fn new() -> HoverResult {
HoverResult {
results: Vec::new(),
// We assume exact by default
exact: true,
}
}
pub fn extend(&mut self, item: Option<String>) {
self.results.extend(item);
}
pub fn is_exact(&self) -> bool {
self.exact
}
pub fn is_empty(&self) -> bool {
self.results.is_empty()
}
pub fn len(&self) -> usize {
self.results.len()
}
pub fn first(&self) -> Option<&str> {
self.results.first().map(String::as_str)
}
pub fn results(&self) -> &[String] {
&self.results
}
/// Returns the results converted into markup
/// for displaying in a UI
pub fn to_markup(&self) -> String {
let mut markup = if !self.exact {
let mut msg = String::from("Failed to exactly resolve the symbol. This is probably because rust_analyzer does not yet support glob imports or traits.");
if !self.results.is_empty() {
msg.push_str(" \nThese items were found instead:");
}
msg.push_str("\n\n---\n");
msg
} else {
String::new()
};
markup.push_str(&self.results.join("\n\n---\n"));
markup
}
}
pub(crate) fn hover(db: &RootDatabase, position: FilePosition) -> Option<RangeInfo<HoverResult>> {
let file = db.parse(position.file_id); let file = db.parse(position.file_id);
let mut res = Vec::new(); let mut res = HoverResult::new();
let mut range = None; let mut range = None;
if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(file.syntax(), position.offset) { if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(file.syntax(), position.offset) {
@ -17,11 +77,9 @@ pub(crate) fn hover(db: &RootDatabase, position: FilePosition) -> Option<RangeIn
match ref_result { match ref_result {
Exact(nav) => res.extend(doc_text_for(db, nav)), Exact(nav) => res.extend(doc_text_for(db, nav)),
Approximate(navs) => { Approximate(navs) => {
let mut msg = String::from("Failed to exactly resolve the symbol. This is probably because rust_analyzer does not yet support glob imports or traits."); // We are no longer exact
if !navs.is_empty() { res.exact = false;
msg.push_str(" \nThese items were found instead:");
}
res.push(msg);
for nav in navs { for nav in navs {
res.extend(doc_text_for(db, nav)) res.extend(doc_text_for(db, nav))
} }
@ -30,7 +88,20 @@ pub(crate) fn hover(db: &RootDatabase, position: FilePosition) -> Option<RangeIn
if !res.is_empty() { if !res.is_empty() {
range = Some(name_ref.syntax().range()) range = Some(name_ref.syntax().range())
} }
} else if let Some(name) = find_node_at_offset::<ast::Name>(file.syntax(), position.offset) {
let navs = crate::goto_definition::name_definition(db, position.file_id, name);
if let Some(navs) = navs {
for nav in navs {
res.extend(doc_text_for(db, nav))
}
}
if !res.is_empty() && range.is_none() {
range = Some(name.syntax().range());
}
} }
if range.is_none() { if range.is_none() {
let node = find_leaf_at_offset(file.syntax(), position.offset).find_map(|leaf| { let node = find_leaf_at_offset(file.syntax(), position.offset).find_map(|leaf| {
leaf.ancestors().find(|n| ast::Expr::cast(*n).is_some() || ast::Pat::cast(*n).is_some()) leaf.ancestors().find(|n| ast::Expr::cast(*n).is_some() || ast::Pat::cast(*n).is_some())
@ -38,13 +109,13 @@ pub(crate) fn hover(db: &RootDatabase, position: FilePosition) -> Option<RangeIn
let frange = FileRange { file_id: position.file_id, range: node.range() }; let frange = FileRange { file_id: position.file_id, range: node.range() };
res.extend(type_of(db, frange).map(Into::into)); res.extend(type_of(db, frange).map(Into::into));
range = Some(node.range()); range = Some(node.range());
}; }
let range = range?; let range = range?;
if res.is_empty() { if res.is_empty() {
return None; return None;
} }
let res = RangeInfo::new(range, res.join("\n\n---\n")); let res = RangeInfo::new(range, res);
Some(res) Some(res)
} }
@ -120,7 +191,7 @@ impl NavigationTarget {
fn visit_node<T>(node: &T, label: &str) -> Option<String> fn visit_node<T>(node: &T, label: &str) -> Option<String>
where where
T: ast::NameOwner + ast::VisibilityOwner, T: NameOwner + VisibilityOwner,
{ {
let mut string = let mut string =
node.visibility().map(|v| format!("{} ", v.syntax().text())).unwrap_or_default(); node.visibility().map(|v| format!("{} ", v.syntax().text())).unwrap_or_default();
@ -130,7 +201,7 @@ impl NavigationTarget {
} }
visitor() visitor()
.visit(|node: &ast::FnDef| visit_node(node, "fn ")) .visit(crate::completion::function_label)
.visit(|node: &ast::StructDef| visit_node(node, "struct ")) .visit(|node: &ast::StructDef| visit_node(node, "struct "))
.visit(|node: &ast::EnumDef| visit_node(node, "enum ")) .visit(|node: &ast::EnumDef| visit_node(node, "enum "))
.visit(|node: &ast::TraitDef| visit_node(node, "trait ")) .visit(|node: &ast::TraitDef| visit_node(node, "trait "))
@ -145,7 +216,24 @@ impl NavigationTarget {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use ra_syntax::TextRange; use ra_syntax::TextRange;
use crate::mock_analysis::{single_file_with_position, single_file_with_range}; use crate::mock_analysis::{single_file_with_position, single_file_with_range, analysis_and_position};
fn trim_markup(s: &str) -> &str {
s.trim_start_matches("```rust\n").trim_end_matches("\n```")
}
fn check_hover_result(fixture: &str, expected: &[&str]) {
let (analysis, position) = analysis_and_position(fixture);
let hover = analysis.hover(position).unwrap().unwrap();
for (markup, expected) in
hover.info.results().iter().zip(expected.iter().chain(std::iter::repeat(&"<missing>")))
{
assert_eq!(trim_markup(&markup), *expected);
}
assert_eq!(hover.info.len(), expected.len());
}
#[test] #[test]
fn hover_shows_type_of_an_expression() { fn hover_shows_type_of_an_expression() {
@ -160,7 +248,76 @@ mod tests {
); );
let hover = analysis.hover(position).unwrap().unwrap(); let hover = analysis.hover(position).unwrap().unwrap();
assert_eq!(hover.range, TextRange::from_to(95.into(), 100.into())); assert_eq!(hover.range, TextRange::from_to(95.into(), 100.into()));
assert_eq!(hover.info, "u32"); assert_eq!(hover.info.first(), Some("u32"));
}
#[test]
fn hover_shows_fn_signature() {
// Single file with result
check_hover_result(
r#"
//- /main.rs
pub fn foo() -> u32 { 1 }
fn main() {
let foo_test = fo<|>o();
}
"#,
&["pub fn foo() -> u32"],
);
// Multiple results
check_hover_result(
r#"
//- /a.rs
pub fn foo() -> u32 { 1 }
//- /b.rs
pub fn foo() -> &str { "" }
//- /c.rs
pub fn foo(a: u32, b: u32) {}
//- /main.rs
mod a;
mod b;
mod c;
fn main() {
let foo_test = fo<|>o();
}
"#,
&["pub fn foo() -> &str", "pub fn foo() -> u32", "pub fn foo(a: u32, b: u32)"],
);
}
#[test]
fn hover_shows_fn_signature_with_type_params() {
check_hover_result(
r#"
//- /main.rs
pub fn foo<'a, T: AsRef<str>>(b: &'a T) -> &'a str { }
fn main() {
let foo_test = fo<|>o();
}
"#,
&["pub fn foo<'a, T: AsRef<str>>(b: &'a T) -> &'a str"],
);
}
#[test]
fn hover_shows_fn_signature_on_fn_name() {
check_hover_result(
r#"
//- /main.rs
pub fn foo<|>(a: u32, b: u32) -> u32 {}
fn main() {
}
"#,
&["pub fn foo(a: u32, b: u32) -> u32"],
);
} }
#[test] #[test]
@ -177,21 +334,21 @@ mod tests {
); );
let hover = analysis.hover(position).unwrap().unwrap(); let hover = analysis.hover(position).unwrap().unwrap();
// not the nicest way to show it currently // not the nicest way to show it currently
assert_eq!(hover.info, "Some<i32>(T) -> Option<T>"); assert_eq!(hover.info.first(), Some("Some<i32>(T) -> Option<T>"));
} }
#[test] #[test]
fn hover_for_local_variable() { fn hover_for_local_variable() {
let (analysis, position) = single_file_with_position("fn func(foo: i32) { fo<|>o; }"); let (analysis, position) = single_file_with_position("fn func(foo: i32) { fo<|>o; }");
let hover = analysis.hover(position).unwrap().unwrap(); let hover = analysis.hover(position).unwrap().unwrap();
assert_eq!(hover.info, "i32"); assert_eq!(hover.info.first(), Some("i32"));
} }
#[test] #[test]
fn hover_for_local_variable_pat() { fn hover_for_local_variable_pat() {
let (analysis, position) = single_file_with_position("fn func(fo<|>o: i32) {}"); let (analysis, position) = single_file_with_position("fn func(fo<|>o: i32) {}");
let hover = analysis.hover(position).unwrap().unwrap(); let hover = analysis.hover(position).unwrap().unwrap();
assert_eq!(hover.info, "i32"); assert_eq!(hover.info.first(), Some("i32"));
} }
#[test] #[test]
@ -258,6 +415,6 @@ mod tests {
", ",
); );
let hover = analysis.hover(position).unwrap().unwrap(); let hover = analysis.hover(position).unwrap().unwrap();
assert_eq!(hover.info, "Thing"); assert_eq!(hover.info.first(), Some("Thing"));
} }
} }

View file

@ -58,6 +58,7 @@ pub use crate::{
navigation_target::NavigationTarget, navigation_target::NavigationTarget,
references::ReferenceSearchResult, references::ReferenceSearchResult,
assists::{Assist, AssistId}, assists::{Assist, AssistId},
hover::{HoverResult},
}; };
pub use ra_ide_api_light::{ pub use ra_ide_api_light::{
Fold, FoldKind, HighlightedRange, Severity, StructureNode, LocalEdit, Fold, FoldKind, HighlightedRange, Severity, StructureNode, LocalEdit,
@ -328,7 +329,7 @@ impl Analysis {
} }
/// Returns a short text describing element at position. /// Returns a short text describing element at position.
pub fn hover(&self, position: FilePosition) -> Cancelable<Option<RangeInfo<String>>> { pub fn hover(&self, position: FilePosition) -> Cancelable<Option<RangeInfo<HoverResult>>> {
self.with_db(|db| hover::hover(db, position)) self.with_db(|db| hover::hover(db, position))
} }

View file

@ -441,7 +441,7 @@ pub fn handle_hover(
let res = Hover { let res = Hover {
contents: HoverContents::Markup(MarkupContent { contents: HoverContents::Markup(MarkupContent {
kind: MarkupKind::Markdown, kind: MarkupKind::Markdown,
value: info.info, value: info.info.to_markup(),
}), }),
range: Some(range), range: Some(range),
}; };