Make punctuation highlighting configurable, disable it by default

This commit is contained in:
Lukas Wirth 2022-08-22 13:15:42 +02:00
parent 6627b473e2
commit 16315edaee
4 changed files with 56 additions and 13 deletions

View file

@ -296,7 +296,7 @@ impl Highlight {
Highlight { tag, mods: HlMods::default() } Highlight { tag, mods: HlMods::default() }
} }
pub fn is_empty(&self) -> bool { pub fn is_empty(&self) -> bool {
self.tag == HlTag::None && self.mods == HlMods::default() self.tag == HlTag::None && self.mods.is_empty()
} }
} }
@ -330,6 +330,10 @@ impl ops::BitOr<HlMod> for Highlight {
} }
impl HlMods { impl HlMods {
pub fn is_empty(&self) -> bool {
self.0 == 0
}
pub fn contains(self, m: HlMod) -> bool { pub fn contains(self, m: HlMod) -> bool {
self.0 & m.mask() == m.mask() self.0 & m.mask() == m.mask()
} }

View file

@ -391,6 +391,16 @@ config_data! {
/// By disabling semantic tokens for strings, other grammars can be used to highlight /// By disabling semantic tokens for strings, other grammars can be used to highlight
/// their contents. /// their contents.
semanticHighlighting_strings_enable: bool = "true", semanticHighlighting_strings_enable: bool = "true",
/// Use semantic tokens for punctuations.
///
/// When disabled, rust-analyzer will emit semantic tokens only for punctuation tokens when
/// they are tagged with modifiers.
semanticHighlighting_punctuation_enable: bool = "false",
/// Use specialized semantic tokens for punctuations.
///
/// When enabled, rust-analyzer will emit special token types for punctuation tokens instead
/// of the generic `punctuation` token type.
semanticHighlighting_punctuation_specialize: bool = "false",
/// Show full signature of the callable. Only shows parameters if disabled. /// Show full signature of the callable. Only shows parameters if disabled.
signatureInfo_detail: SignatureDetail = "\"full\"", signatureInfo_detail: SignatureDetail = "\"full\"",
@ -523,6 +533,13 @@ impl HoverActionsConfig {
} }
} }
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct HighlightingConfig {
pub strings: bool,
pub punctuation: bool,
pub specialize_punctuation: bool,
}
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct FilesConfig { pub struct FilesConfig {
pub watcher: FilesWatcher, pub watcher: FilesWatcher,
@ -1171,8 +1188,12 @@ impl Config {
} }
} }
pub fn highlighting_strings(&self) -> bool { pub fn highlighting_config(&self) -> HighlightingConfig {
self.data.semanticHighlighting_strings_enable HighlightingConfig {
strings: self.data.semanticHighlighting_strings_enable,
punctuation: self.data.semanticHighlighting_punctuation_enable,
specialize_punctuation: self.data.semanticHighlighting_punctuation_specialize,
}
} }
pub fn hover(&self) -> HoverConfig { pub fn hover(&self) -> HoverConfig {

View file

@ -1505,9 +1505,9 @@ pub(crate) fn handle_semantic_tokens_full(
let line_index = snap.file_line_index(file_id)?; let line_index = snap.file_line_index(file_id)?;
let highlights = snap.analysis.highlight(file_id)?; let highlights = snap.analysis.highlight(file_id)?;
let highlight_strings = snap.config.highlighting_strings(); let highlighting_config = snap.config.highlighting_config();
let semantic_tokens = let semantic_tokens =
to_proto::semantic_tokens(&text, &line_index, highlights, highlight_strings); to_proto::semantic_tokens(&text, &line_index, highlights, highlighting_config);
// Unconditionally cache the tokens // Unconditionally cache the tokens
snap.semantic_tokens_cache.lock().insert(params.text_document.uri, semantic_tokens.clone()); snap.semantic_tokens_cache.lock().insert(params.text_document.uri, semantic_tokens.clone());
@ -1526,7 +1526,7 @@ pub(crate) fn handle_semantic_tokens_full_delta(
let line_index = snap.file_line_index(file_id)?; let line_index = snap.file_line_index(file_id)?;
let highlights = snap.analysis.highlight(file_id)?; let highlights = snap.analysis.highlight(file_id)?;
let highlight_strings = snap.config.highlighting_strings(); let highlight_strings = snap.config.highlighting_config();
let semantic_tokens = let semantic_tokens =
to_proto::semantic_tokens(&text, &line_index, highlights, highlight_strings); to_proto::semantic_tokens(&text, &line_index, highlights, highlight_strings);
@ -1557,7 +1557,7 @@ pub(crate) fn handle_semantic_tokens_range(
let line_index = snap.file_line_index(frange.file_id)?; let line_index = snap.file_line_index(frange.file_id)?;
let highlights = snap.analysis.highlight_range(frange)?; let highlights = snap.analysis.highlight_range(frange)?;
let highlight_strings = snap.config.highlighting_strings(); let highlight_strings = snap.config.highlighting_config();
let semantic_tokens = let semantic_tokens =
to_proto::semantic_tokens(&text, &line_index, highlights, highlight_strings); to_proto::semantic_tokens(&text, &line_index, highlights, highlight_strings);
Ok(Some(semantic_tokens.into())) Ok(Some(semantic_tokens.into()))

View file

@ -18,7 +18,7 @@ use vfs::AbsPath;
use crate::{ use crate::{
cargo_target_spec::CargoTargetSpec, cargo_target_spec::CargoTargetSpec,
config::{CallInfoConfig, Config}, config::{CallInfoConfig, Config, HighlightingConfig},
global_state::GlobalStateSnapshot, global_state::GlobalStateSnapshot,
line_index::{LineEndings, LineIndex, OffsetEncoding}, line_index::{LineEndings, LineIndex, OffsetEncoding},
lsp_ext, lsp_ext,
@ -517,19 +517,37 @@ pub(crate) fn semantic_tokens(
text: &str, text: &str,
line_index: &LineIndex, line_index: &LineIndex,
highlights: Vec<HlRange>, highlights: Vec<HlRange>,
highlight_strings: bool, config: HighlightingConfig,
) -> lsp_types::SemanticTokens { ) -> lsp_types::SemanticTokens {
let id = TOKEN_RESULT_COUNTER.fetch_add(1, Ordering::SeqCst).to_string(); let id = TOKEN_RESULT_COUNTER.fetch_add(1, Ordering::SeqCst).to_string();
let mut builder = semantic_tokens::SemanticTokensBuilder::new(id); let mut builder = semantic_tokens::SemanticTokensBuilder::new(id);
for highlight_range in highlights { for mut highlight_range in highlights {
if highlight_range.highlight.is_empty() { if highlight_range.highlight.is_empty() {
continue; continue;
} }
let (ty, mods) = semantic_token_type_and_modifiers(highlight_range.highlight);
if !highlight_strings && ty == lsp_types::SemanticTokenType::STRING { // apply config filtering
continue; match &mut highlight_range.highlight.tag {
HlTag::StringLiteral if !config.strings => continue,
// If punctuation is disabled, make the macro bang part of the macro call again.
tag @ HlTag::Punctuation(HlPunct::MacroBang)
if !config.punctuation || !config.specialize_punctuation =>
{
*tag = HlTag::Symbol(SymbolKind::Macro);
}
HlTag::Punctuation(_)
if !config.punctuation && highlight_range.highlight.mods.is_empty() =>
{
continue
}
tag @ HlTag::Punctuation(_) if !config.specialize_punctuation => {
*tag = HlTag::Punctuation(HlPunct::Other);
}
_ => (),
} }
let (ty, mods) = semantic_token_type_and_modifiers(highlight_range.highlight);
let token_index = semantic_tokens::type_index(ty); let token_index = semantic_tokens::type_index(ty);
let modifier_bitset = mods.0; let modifier_bitset = mods.0;