/* * Copyright (c) 2020, the SerenityOS developers. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include namespace GUI { static TextStyle style_for_token_type(Gfx::Palette palette, JS::TokenType type) { switch (type) { case JS::TokenType::Invalid: case JS::TokenType::Eof: return { palette.syntax_comment() }; case JS::TokenType::NumericLiteral: case JS::TokenType::BigIntLiteral: return { palette.syntax_number() }; case JS::TokenType::StringLiteral: case JS::TokenType::TemplateLiteralStart: case JS::TokenType::TemplateLiteralEnd: case JS::TokenType::TemplateLiteralString: case JS::TokenType::RegexLiteral: case JS::TokenType::RegexFlags: case JS::TokenType::UnterminatedStringLiteral: return { palette.syntax_string() }; case JS::TokenType::BracketClose: case JS::TokenType::BracketOpen: case JS::TokenType::Comma: case JS::TokenType::CurlyClose: case JS::TokenType::CurlyOpen: case JS::TokenType::ParenClose: case JS::TokenType::ParenOpen: case JS::TokenType::Semicolon: case JS::TokenType::TemplateLiteralExprStart: case JS::TokenType::TemplateLiteralExprEnd: return { palette.syntax_punctuation() }; case JS::TokenType::Ampersand: case JS::TokenType::AmpersandEquals: case JS::TokenType::Asterisk: case JS::TokenType::DoubleAsteriskEquals: case JS::TokenType::AsteriskEquals: case JS::TokenType::Caret: case JS::TokenType::CaretEquals: case JS::TokenType::DoubleAmpersand: case JS::TokenType::DoubleAsterisk: case JS::TokenType::DoublePipe: case JS::TokenType::DoubleQuestionMark: case JS::TokenType::Equals: case JS::TokenType::EqualsEquals: case JS::TokenType::EqualsEqualsEquals: case JS::TokenType::ExclamationMark: case JS::TokenType::ExclamationMarkEquals: case JS::TokenType::ExclamationMarkEqualsEquals: case JS::TokenType::GreaterThan: case JS::TokenType::GreaterThanEquals: case JS::TokenType::LessThan: case JS::TokenType::LessThanEquals: case JS::TokenType::Minus: case JS::TokenType::MinusEquals: case JS::TokenType::MinusMinus: case JS::TokenType::Percent: case JS::TokenType::PercentEquals: case JS::TokenType::Period: case JS::TokenType::Pipe: case JS::TokenType::PipeEquals: case JS::TokenType::Plus: case JS::TokenType::PlusEquals: case JS::TokenType::PlusPlus: case JS::TokenType::QuestionMark: case JS::TokenType::QuestionMarkPeriod: case JS::TokenType::ShiftLeft: case JS::TokenType::ShiftLeftEquals: case JS::TokenType::ShiftRight: case JS::TokenType::ShiftRightEquals: case JS::TokenType::Slash: case JS::TokenType::SlashEquals: case JS::TokenType::Tilde: case JS::TokenType::UnsignedShiftRight: case JS::TokenType::UnsignedShiftRightEquals: return { palette.syntax_operator() }; case JS::TokenType::BoolLiteral: case JS::TokenType::Class: case JS::TokenType::Const: case JS::TokenType::Delete: case JS::TokenType::Debugger: case JS::TokenType::Extends: case JS::TokenType::Function: case JS::TokenType::In: case JS::TokenType::Instanceof: case JS::TokenType::Interface: case JS::TokenType::Let: case JS::TokenType::New: case JS::TokenType::NullLiteral: case JS::TokenType::Super: case JS::TokenType::Typeof: case JS::TokenType::Var: case JS::TokenType::Void: return { palette.syntax_keyword(), &Gfx::Font::default_bold_fixed_width_font() }; case JS::TokenType::Await: case JS::TokenType::Catch: case JS::TokenType::Do: case JS::TokenType::Else: case JS::TokenType::Finally: case JS::TokenType::For: case JS::TokenType::If: case JS::TokenType::Return: case JS::TokenType::Try: case JS::TokenType::While: case JS::TokenType::Yield: return { palette.syntax_control_keyword(), &Gfx::Font::default_bold_fixed_width_font() }; case JS::TokenType::Identifier: return { palette.syntax_identifier() }; default: return { palette.base_text() }; } } bool JSSyntaxHighlighter::is_identifier(void* token) const { auto js_token = static_cast(reinterpret_cast(token)); return js_token == JS::TokenType::Identifier; } bool JSSyntaxHighlighter::is_navigatable(void* token) const { (void)token; return false; } void JSSyntaxHighlighter::rehighlight(Gfx::Palette palette) { ASSERT(m_editor); auto text = m_editor->text(); JS::Lexer lexer(text); Vector spans; GUI::TextPosition position { 0, 0 }; GUI::TextPosition start { 0, 0 }; auto advance_position = [&position](char ch) { if (ch == '\n') { position.set_line(position.line() + 1); position.set_column(0); } else position.set_column(position.column() + 1); }; auto append_token = [&](StringView str, const JS::Token& token, bool is_trivia) { if (str.is_empty()) return; start = position; for (size_t i = 0; i < str.length() - 1; ++i) advance_position(str[i]); GUI::TextDocumentSpan span; span.range.set_start(start); span.range.set_end({ position.line(), position.column() }); auto type = is_trivia ? JS::TokenType::Invalid : token.type(); auto style = style_for_token_type(palette, type); span.color = style.color; span.font = style.font; span.is_skippable = is_trivia; span.data = reinterpret_cast(static_cast(type)); spans.append(span); advance_position(str[str.length() - 1]); #ifdef DEBUG_SYNTAX_HIGHLIGHTING dbg() << token.name() << (is_trivia ? " (trivia) @ \"" : " @ \"") << token.value() << "\" " << span.range.start().line() << ":" << span.range.start().column() << " - " << span.range.end().line() << ":" << span.range.end().column(); #endif }; bool was_eof = false; for (auto token = lexer.next(); !was_eof; token = lexer.next()) { append_token(token.trivia(), token, true); append_token(token.value(), token, false); if (token.type() == JS::TokenType::Eof) was_eof = true; } m_editor->document().set_spans(spans); m_has_brace_buddies = false; highlight_matching_token_pair(); m_editor->update(); } Vector JSSyntaxHighlighter::matching_token_pairs() const { static Vector pairs; if (pairs.is_empty()) { pairs.append({ reinterpret_cast(JS::TokenType::CurlyOpen), reinterpret_cast(JS::TokenType::CurlyClose) }); pairs.append({ reinterpret_cast(JS::TokenType::ParenOpen), reinterpret_cast(JS::TokenType::ParenClose) }); pairs.append({ reinterpret_cast(JS::TokenType::BracketOpen), reinterpret_cast(JS::TokenType::BracketClose) }); } return pairs; } bool JSSyntaxHighlighter::token_types_equal(void* token1, void* token2) const { return static_cast(reinterpret_cast(token1)) == static_cast(reinterpret_cast(token2)); } JSSyntaxHighlighter::~JSSyntaxHighlighter() { } }