//! Syntax highlighting for format macro strings. use ide_db::{ defs::Definition, syntax_helpers::format_string::{is_format_string, lex_format_specifiers, FormatSpecifier}, SymbolKind, }; use syntax::{ast, TextRange}; use crate::{ syntax_highlighting::{highlight::highlight_def, highlights::Highlights}, HlRange, HlTag, }; pub(super) fn highlight_format_string( stack: &mut Highlights, sema: &hir::Semantics<'_, ide_db::RootDatabase>, krate: hir::Crate, string: &ast::String, expanded_string: &ast::String, range: TextRange, ) { if !is_format_string(expanded_string) { return; } // FIXME: Replace this with the HIR info we have now. lex_format_specifiers(string, &mut |piece_range, kind| { if let Some(highlight) = highlight_format_specifier(kind) { stack.add(HlRange { range: piece_range + range.start(), highlight: highlight.into(), binding_hash: None, }); } }); if let Some(parts) = sema.as_format_args_parts(string) { parts.into_iter().for_each(|(range, res)| { if let Some(res) = res { stack.add(HlRange { range, highlight: highlight_def(sema, krate, Definition::from(res)), binding_hash: None, }) } }) } } fn highlight_format_specifier(kind: FormatSpecifier) -> Option { Some(match kind { FormatSpecifier::Open | FormatSpecifier::Close | FormatSpecifier::Colon | FormatSpecifier::Fill | FormatSpecifier::Align | FormatSpecifier::Sign | FormatSpecifier::NumberSign | FormatSpecifier::DollarSign | FormatSpecifier::Dot | FormatSpecifier::Asterisk | FormatSpecifier::QuestionMark => HlTag::FormatSpecifier, FormatSpecifier::Integer | FormatSpecifier::Zero => HlTag::NumericLiteral, FormatSpecifier::Identifier => HlTag::Symbol(SymbolKind::Local), FormatSpecifier::Escape => HlTag::EscapeSequence, }) }