# typed: strict # frozen_string_literal: true module RubyLsp module ResponseBuilders class SemanticHighlighting < ResponseBuilder class UndefinedTokenType < StandardError; end TOKEN_TYPES = T.let( { namespace: 0, type: 1, class: 2, enum: 3, interface: 4, struct: 5, typeParameter: 6, parameter: 7, variable: 8, property: 9, enumMember: 10, event: 11, function: 12, method: 13, macro: 14, keyword: 15, modifier: 16, comment: 17, string: 18, number: 19, regexp: 20, operator: 21, decorator: 22, }.freeze, T::Hash[Symbol, Integer], ) TOKEN_MODIFIERS = T.let( { declaration: 0, definition: 1, readonly: 2, static: 3, deprecated: 4, abstract: 5, async: 6, modification: 7, documentation: 8, default_library: 9, }.freeze, T::Hash[Symbol, Integer], ) extend T::Sig ResponseType = type_member { { fixed: Interface::SemanticTokens } } sig { void } def initialize super @stack = T.let([], T::Array[SemanticToken]) end sig { params(location: Prism::Location, type: Symbol, modifiers: T::Array[Symbol]).void } def add_token(location, type, modifiers = []) length = location.end_offset - location.start_offset modifiers_indices = modifiers.filter_map { |modifier| TOKEN_MODIFIERS[modifier] } @stack.push( SemanticToken.new( location: location, length: length, type: T.must(TOKEN_TYPES[type]), modifier: modifiers_indices, ), ) end sig { returns(T.nilable(SemanticToken)) } def last @stack.last end sig { override.returns(Interface::SemanticTokens) } def response SemanticTokenEncoder.new.encode(@stack) end class SemanticToken extend T::Sig sig { returns(Prism::Location) } attr_reader :location sig { returns(Integer) } attr_reader :length sig { returns(Integer) } attr_reader :type sig { returns(T::Array[Integer]) } attr_reader :modifier sig { params(location: Prism::Location, length: Integer, type: Integer, modifier: T::Array[Integer]).void } def initialize(location:, length:, type:, modifier:) @location = location @length = length @type = type @modifier = modifier end sig { params(type_symbol: Symbol).void } def replace_type(type_symbol) type_int = TOKEN_TYPES[type_symbol] raise UndefinedTokenType, "Undefined token type: #{type_symbol}" unless type_int @type = type_int end sig { params(modifier_symbols: T::Array[Symbol]).void } def replace_modifier(modifier_symbols) @modifier = modifier_symbols.filter_map do |modifier_symbol| modifier_index = TOKEN_MODIFIERS[modifier_symbol] raise UndefinedTokenType, "Undefined token modifier: #{modifier_symbol}" unless modifier_index modifier_index end end end class SemanticTokenEncoder extend T::Sig sig { void } def initialize @current_row = T.let(0, Integer) @current_column = T.let(0, Integer) end sig do params( tokens: T::Array[SemanticToken], ).returns(Interface::SemanticTokens) end def encode(tokens) sorted_tokens = tokens.sort_by.with_index do |token, index| # Enumerable#sort_by is not deterministic when the compared values are equal. # When that happens, we need to use the index as a tie breaker to ensure # that the order of the tokens is always the same. [token.location.start_line, token.location.start_column, index] end delta = sorted_tokens.flat_map do |token| compute_delta(token) end Interface::SemanticTokens.new(data: delta) end # The delta array is computed according to the LSP specification: # > The protocol for the token format relative uses relative # > positions, because most tokens remain stable relative to # > each other when edits are made in a file. This simplifies # > the computation of a delta if a server supports it. So each # > token is represented using 5 integers. # For more information on how each number is calculated, read: # https://microsoft.github.io/language-server-protocol/specifications/specification-current/#textDocument_semanticTokens sig { params(token: SemanticToken).returns(T::Array[Integer]) } def compute_delta(token) row = token.location.start_line - 1 column = token.location.start_column begin delta_line = row - @current_row delta_column = column delta_column -= @current_column if delta_line == 0 [delta_line, delta_column, token.length, token.type, encode_modifiers(token.modifier)] ensure @current_row = row @current_column = column end end # Encode an array of modifiers to positions onto a bit flag # For example, [:default_library] will be encoded as # 0b1000000000, as :default_library is the 10th bit according # to the token modifiers index map. sig { params(modifiers: T::Array[Integer]).returns(Integer) } def encode_modifiers(modifiers) modifiers.inject(0) do |encoded_modifiers, modifier| encoded_modifiers | (1 << modifier) end end end end end end