#[allow(unused)] use log::info; use sv_parser::{RefNode, SyntaxTree}; use tower_lsp::lsp_types::*; use crate::{definition::{get_ident, Scope}, server::LSPServer, sources::{LSPSupport, ParseIR, Source}}; pub fn document_highlight( server: &LSPServer, token: &str, file: &Source, pos: Position, uri: &Url ) -> Option> { let scope_tree = server.srcs.scope_tree.read().ok()?; // use the byte_idx of the definition if possible, otherwise use the cursor let byte_idx = match scope_tree .as_ref()? .get_definition(token, file.text.pos_to_byte(&pos), uri) { Some(def) => def.byte_idx, None => file.text.pos_to_byte(&pos), }; let syntax_tree = file.parse_ir.as_ref()?; match syntax_tree { ParseIR::SyntaxTree(syntax_tree) => { let references = all_identifiers(&syntax_tree, &token); Some( scope_tree .as_ref()? .document_highlights(&uri, &file.text, references, byte_idx), ) }, _ => { info!("error happen in [sv_document_highlight]"); None } } } /// return all identifiers in a syntax tree matching a given token fn all_identifiers(syntax_tree: &SyntaxTree, token: &str) -> Vec<(String, usize)> { let mut idents: Vec<(String, usize)> = Vec::new(); for node in syntax_tree { if let RefNode::Identifier(_) = node { let (ident, byte_idx) = get_ident(syntax_tree, node); if ident == token { idents.push((ident, byte_idx)); } } } idents }