mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-20 21:32:07 +00:00 
			
		
		
		
	 9cd27d1e15
			
		
	
	
		9cd27d1e15
		
	
	
	
	
		
			
			This adds a new semantic token type, PreprocessorMacro. Calls to preprocessor macros will now be highlighted when semantic highlighting is enabled in Hack Studio.
		
			
				
	
	
		
			166 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			166 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2022, Itamar S. <itamar8910@gmail.com>
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "SemanticSyntaxHighlighter.h"
 | |
| #include "Lexer.h"
 | |
| #include <LibDiff/Generator.h>
 | |
| #include <LibGUI/AutocompleteProvider.h>
 | |
| #include <LibGfx/Palette.h>
 | |
| 
 | |
| namespace Cpp {
 | |
| 
 | |
| void SemanticSyntaxHighlighter::rehighlight(Palette const& palette)
 | |
| {
 | |
|     Vector<GUI::AutocompleteProvider::TokenInfo> new_tokens_info;
 | |
|     auto text = m_client->get_text();
 | |
|     {
 | |
|         Threading::MutexLocker locker(m_lock);
 | |
|         Cpp::Lexer lexer(text);
 | |
|         lexer.set_ignore_whitespace(true);
 | |
|         auto current_tokens = lexer.lex();
 | |
| 
 | |
|         StringBuilder current_tokens_as_lines;
 | |
|         StringBuilder previous_tokens_as_lines;
 | |
| 
 | |
|         for (auto& token : current_tokens)
 | |
|             current_tokens_as_lines.appendff("{}\n", token.type_as_string());
 | |
| 
 | |
|         for (Cpp::Token const& token : m_saved_tokens)
 | |
|             previous_tokens_as_lines.appendff("{}\n", token.type_as_string());
 | |
| 
 | |
|         auto previous = previous_tokens_as_lines.build();
 | |
|         auto current = current_tokens_as_lines.build();
 | |
| 
 | |
|         // FIXME: Computing the diff on the entire document's tokens is quite inefficient.
 | |
|         //        An improvement over this could be only including the tokens that are in edited text ranges in the diff.
 | |
|         auto diff_hunks = Diff::from_text(previous.view(), current.view());
 | |
|         for (auto& token : current_tokens) {
 | |
|             new_tokens_info.append(GUI::AutocompleteProvider::TokenInfo { GUI::AutocompleteProvider::TokenInfo::SemanticType::Unknown,
 | |
|                 token.start().line, token.start().column, token.end().line, token.end().column });
 | |
|         }
 | |
|         size_t previous_token_index = 0;
 | |
|         size_t current_token_index = 0;
 | |
|         for (auto const& hunk : diff_hunks) {
 | |
|             for (; previous_token_index < hunk.original_start_line; ++previous_token_index) {
 | |
|                 new_tokens_info[current_token_index].type = m_tokens_info[previous_token_index].type;
 | |
|                 ++current_token_index;
 | |
|             }
 | |
|             for (size_t i = 0; i < hunk.added_lines.size(); ++i) {
 | |
|                 ++current_token_index;
 | |
|             }
 | |
|             for (size_t i = 0; i < hunk.removed_lines.size(); ++i) {
 | |
|                 ++previous_token_index;
 | |
|             }
 | |
|         }
 | |
|         while (current_token_index < new_tokens_info.size() && previous_token_index < m_tokens_info.size()) {
 | |
|             new_tokens_info[current_token_index].type = m_tokens_info[previous_token_index].type;
 | |
| 
 | |
|             ++previous_token_index;
 | |
|             ++current_token_index;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     update_spans(new_tokens_info, palette);
 | |
| }
 | |
| 
 | |
| static Syntax::TextStyle style_for_token_type(Gfx::Palette const& palette, GUI::AutocompleteProvider::TokenInfo::SemanticType type)
 | |
| {
 | |
|     switch (type) {
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Unknown:
 | |
|         return { palette.base_text(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Keyword:
 | |
|         return { palette.syntax_keyword(), true };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Type:
 | |
|         return { palette.syntax_type(), true };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Identifier:
 | |
|         return { palette.syntax_identifier(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::String:
 | |
|         return { palette.syntax_string(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Number:
 | |
|         return { palette.syntax_number(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::IncludePath:
 | |
|         return { palette.syntax_preprocessor_value(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::PreprocessorStatement:
 | |
|         return { palette.syntax_preprocessor_statement(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Comment:
 | |
|         return { palette.syntax_comment(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Function:
 | |
|         return { palette.syntax_function(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Variable:
 | |
|         return { palette.syntax_variable(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::CustomType:
 | |
|         return { palette.syntax_custom_type(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Namespace:
 | |
|         return { palette.syntax_namespace(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Member:
 | |
|         return { palette.syntax_member(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::Parameter:
 | |
|         return { palette.syntax_parameter(), false };
 | |
|     case GUI::AutocompleteProvider::TokenInfo::SemanticType::PreprocessorMacro:
 | |
|         return { palette.syntax_preprocessor_value(), false };
 | |
|     default:
 | |
|         VERIFY_NOT_REACHED();
 | |
|         return { palette.base_text(), false };
 | |
|     }
 | |
| }
 | |
| void SemanticSyntaxHighlighter::update_spans(Vector<GUI::AutocompleteProvider::TokenInfo> const& tokens_info, Gfx::Palette const& pallete)
 | |
| {
 | |
|     Vector<GUI::TextDocumentSpan> spans;
 | |
|     for (auto& token : tokens_info) {
 | |
|         // FIXME: The +1 for the token end column is a quick hack due to not wanting to modify the lexer (which is also used by the parser). Maybe there's a better way to do this.
 | |
|         GUI::TextDocumentSpan span;
 | |
|         span.range.set_start({ token.start_line, token.start_column });
 | |
|         span.range.set_end({ token.end_line, token.end_column + 1 });
 | |
|         auto style = style_for_token_type(pallete, token.type);
 | |
|         span.attributes.color = style.color;
 | |
|         span.attributes.bold = style.bold;
 | |
|         span.is_skippable = token.type == GUI::AutocompleteProvider::TokenInfo::SemanticType::Whitespace;
 | |
|         span.data = static_cast<u64>(token.type);
 | |
|         spans.append(span);
 | |
|     }
 | |
|     m_client->do_set_spans(move(spans));
 | |
| 
 | |
|     m_has_brace_buddies = false;
 | |
|     highlight_matching_token_pair();
 | |
| 
 | |
|     m_client->do_update();
 | |
| }
 | |
| 
 | |
| void SemanticSyntaxHighlighter::update_tokens_info(Vector<GUI::AutocompleteProvider::TokenInfo> tokens_info)
 | |
| {
 | |
|     {
 | |
|         Threading::MutexLocker locker(m_lock);
 | |
|         m_tokens_info = move(tokens_info);
 | |
| 
 | |
|         m_saved_tokens_text = m_client->get_text();
 | |
|         Cpp::Lexer lexer(m_saved_tokens_text);
 | |
|         lexer.set_ignore_whitespace(true);
 | |
|         m_saved_tokens = lexer.lex();
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool SemanticSyntaxHighlighter::is_identifier(u64 token_type) const
 | |
| {
 | |
|     using GUI::AutocompleteProvider;
 | |
|     auto type = static_cast<AutocompleteProvider::TokenInfo::SemanticType>(token_type);
 | |
| 
 | |
|     return type == AutocompleteProvider::TokenInfo::SemanticType::Identifier
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::Function
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::Variable
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::CustomType
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::Namespace
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::Member
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::Parameter
 | |
|         || type == AutocompleteProvider::TokenInfo::SemanticType::PreprocessorMacro;
 | |
| }
 | |
| 
 | |
| bool SemanticSyntaxHighlighter::is_navigatable(u64 token_type) const
 | |
| {
 | |
|     return static_cast<GUI::AutocompleteProvider::TokenInfo::SemanticType>(token_type) == GUI::AutocompleteProvider::TokenInfo::SemanticType::IncludePath;
 | |
| }
 | |
| 
 | |
| }
 |