1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-18 03:44:05 -05:00

fix(lsp): display module types only dependencies on hover (#12683)

Fixes: #12675
This commit is contained in:
Kitson Kelly 2021-11-08 11:50:48 +11:00 committed by GitHub
parent 91f8bdda2c
commit 182de1452b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 115 additions and 10 deletions

View file

@ -263,6 +263,13 @@ impl Document {
self.maybe_lsp_version.is_some() self.maybe_lsp_version.is_some()
} }
fn maybe_types_dependency(&self) -> deno_graph::Resolved {
let module_result = self.maybe_module.as_ref()?;
let module = module_result.as_ref().ok()?;
let (_, maybe_dep) = module.maybe_types_dependency.as_ref()?;
maybe_dep.clone()
}
fn media_type(&self) -> MediaType { fn media_type(&self) -> MediaType {
if let Some(Ok(module)) = &self.maybe_module { if let Some(Ok(module)) = &self.maybe_module {
module.media_type module.media_type
@ -597,6 +604,16 @@ impl Inner {
}) })
} }
fn get_maybe_types_for_dependency(
&mut self,
dependency: &deno_graph::Dependency,
) -> deno_graph::Resolved {
let code_dep = dependency.maybe_code.as_ref()?;
let (specifier, _) = code_dep.as_ref().ok()?;
let doc = self.get(specifier)?;
doc.maybe_types_dependency()
}
fn get_navigation_tree( fn get_navigation_tree(
&mut self, &mut self,
specifier: &ModuleSpecifier, specifier: &ModuleSpecifier,
@ -952,6 +969,16 @@ impl Documents {
self.0.lock().get_maybe_dependency(specifier, position) self.0.lock().get_maybe_dependency(specifier, position)
} }
/// For a given dependency, try to resolve the maybe_types_dependency for the
/// dependency. This covers modules that assert their own types, like via the
/// triple-slash reference, or the `X-TypeScript-Types` header.
pub fn get_maybe_types_for_dependency(
&self,
dependency: &deno_graph::Dependency,
) -> deno_graph::Resolved {
self.0.lock().get_maybe_types_for_dependency(dependency)
}
/// Get a reference to the navigation tree stored for a given specifier, if /// Get a reference to the navigation tree stored for a given specifier, if
/// any. /// any.
pub fn get_navigation_tree( pub fn get_navigation_tree(

View file

@ -1087,21 +1087,34 @@ impl Inner {
&specifier, &specifier,
&params.text_document_position_params.position, &params.text_document_position_params.position,
) { ) {
let value = match (&dep.maybe_code, &dep.maybe_type) { let maybe_types_dependency =
(Some(code_dep), Some(type_dep)) => format!( self.documents.get_maybe_types_for_dependency(&dep);
let value = match (&dep.maybe_code, &dep.maybe_type, &maybe_types_dependency) {
(Some(code_dep), Some(type_dep), None) => format!(
"**Resolved Dependency**\n\n**Code**: {}\n\n**Types**: {}\n", "**Resolved Dependency**\n\n**Code**: {}\n\n**Types**: {}\n",
to_hover_text(code_dep), to_hover_text(code_dep),
to_hover_text(type_dep) to_hover_text(type_dep)
), ),
(Some(code_dep), None) => format!( (Some(code_dep), Some(type_dep), Some(types_dep)) => format!(
"**Resolved Dependency**\n\n**Code**: {}\n**Types**: {}\n**Import Types**: {}\n",
to_hover_text(code_dep),
to_hover_text(types_dep),
to_hover_text(type_dep)
),
(Some(code_dep), None, None) => format!(
"**Resolved Dependency**\n\n**Code**: {}\n", "**Resolved Dependency**\n\n**Code**: {}\n",
to_hover_text(code_dep) to_hover_text(code_dep)
), ),
(None, Some(type_dep)) => format!( (Some(code_dep), None, Some(types_dep)) => format!(
"**Resolved Dependency**\n\n**Code**: {}\n\n**Types**: {}\n",
to_hover_text(code_dep),
to_hover_text(types_dep)
),
(None, Some(type_dep), _) => format!(
"**Resolved Dependency**\n\n**Types**: {}\n", "**Resolved Dependency**\n\n**Types**: {}\n",
to_hover_text(type_dep) to_hover_text(type_dep)
), ),
(None, None) => unreachable!("{}", json!(params)), (None, None, _) => unreachable!("{}", json!(params)),
}; };
Some(Hover { Some(Hover {
contents: HoverContents::Markup(MarkupContent { contents: HoverContents::Markup(MarkupContent {
@ -2661,11 +2674,6 @@ impl Inner {
.performance .performance
.mark("virtual_text_document", Some(&params)); .mark("virtual_text_document", Some(&params));
let specifier = self.url_map.normalize_url(&params.text_document.uri); let specifier = self.url_map.normalize_url(&params.text_document.uri);
info!(
"virtual_text_document\n{}\nspecifier: {}",
json!(params),
specifier
);
let contents = if specifier.as_str() == "deno:/status.md" { let contents = if specifier.as_str() == "deno:/status.md" {
let mut contents = String::new(); let mut contents = String::new();
let mut documents_specifiers = self.documents.specifiers(false, false); let mut documents_specifiers = self.documents.specifiers(false, false);

View file

@ -1066,6 +1066,76 @@ fn lsp_hover_dependency() {
); );
} }
#[test]
fn lsp_hover_typescript_types() {
let _g = http_server();
let mut client = init("initialize_params.json");
did_open(
&mut client,
json!({
"textDocument": {
"uri": "file:///a/file.ts",
"languageId": "typescript",
"version": 1,
"text": "import * as a from \"http://127.0.0.1:4545/xTypeScriptTypes.js\";\n\nconsole.log(a.foo);\n",
}
}),
);
let (maybe_res, maybe_err) = client
.write_request::<_, _, Value>(
"deno/cache",
json!({
"referrer": {
"uri": "file:///a/file.ts",
},
"uris": [
{
"uri": "http://127.0.0.1:4545/xTypeScriptTypes.js",
}
],
}),
)
.unwrap();
assert!(maybe_err.is_none());
assert!(maybe_res.is_some());
let (maybe_res, maybe_err) = client
.write_request::<_, _, Value>(
"textDocument/hover",
json!({
"textDocument": {
"uri": "file:///a/file.ts"
},
"position": {
"line": 0,
"character": 24
}
}),
)
.unwrap();
assert!(maybe_res.is_some());
assert!(maybe_err.is_none());
assert_eq!(
json!(maybe_res.unwrap()),
json!({
"contents": {
"kind": "markdown",
"value": "**Resolved Dependency**\n\n**Code**: http&#8203;://127.0.0.1:4545/xTypeScriptTypes.js\n\n**Types**: http&#8203;://127.0.0.1:4545/xTypeScriptTypes.d.ts\n"
},
"range": {
"start": {
"line": 0,
"character": 19
},
"end": {
"line": 0,
"character": 62
}
}
})
);
shutdown(&mut client);
}
#[test] #[test]
fn lsp_call_hierarchy() { fn lsp_call_hierarchy() {
let mut client = init("initialize_params.json"); let mut client = init("initialize_params.json");