mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 16:22:07 +00:00 
			
		
		
		
	 1682f0b760
			
		
	
	
		1682f0b760
		
	
	
	
	
		
			
			SPDX License Identifiers are a more compact / standardized way of representing file license information. See: https://spdx.dev/resources/use/#identifiers This was done with the `ambr` search and replace tool. ambr --no-parent-ignore --key-from-file --rep-from-file key.txt rep.txt *
		
			
				
	
	
		
			155 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			155 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "GMLLexer.h"
 | |
| #include <AK/Vector.h>
 | |
| #include <ctype.h>
 | |
| 
 | |
| namespace GUI {
 | |
| 
 | |
| GMLLexer::GMLLexer(const StringView& input)
 | |
|     : m_input(input)
 | |
| {
 | |
| }
 | |
| 
 | |
| char GMLLexer::peek(size_t offset) const
 | |
| {
 | |
|     if ((m_index + offset) >= m_input.length())
 | |
|         return 0;
 | |
|     return m_input[m_index + offset];
 | |
| }
 | |
| 
 | |
| char GMLLexer::consume()
 | |
| {
 | |
|     VERIFY(m_index < m_input.length());
 | |
|     char ch = m_input[m_index++];
 | |
|     m_previous_position = m_position;
 | |
|     if (ch == '\n') {
 | |
|         m_position.line++;
 | |
|         m_position.column = 0;
 | |
|     } else {
 | |
|         m_position.column++;
 | |
|     }
 | |
|     return ch;
 | |
| }
 | |
| 
 | |
| static bool is_valid_identifier_start(char ch)
 | |
| {
 | |
|     return isalpha(ch) || ch == '_';
 | |
| }
 | |
| 
 | |
| static bool is_valid_identifier_character(char ch)
 | |
| {
 | |
|     return isalnum(ch) || ch == '_';
 | |
| }
 | |
| 
 | |
| static bool is_valid_class_character(char ch)
 | |
| {
 | |
|     return isalnum(ch) || ch == '_' || ch == ':';
 | |
| }
 | |
| 
 | |
| Vector<GMLToken> GMLLexer::lex()
 | |
| {
 | |
|     Vector<GMLToken> tokens;
 | |
| 
 | |
|     size_t token_start_index = 0;
 | |
|     GMLPosition token_start_position;
 | |
| 
 | |
|     auto begin_token = [&] {
 | |
|         token_start_index = m_index;
 | |
|         token_start_position = m_position;
 | |
|     };
 | |
| 
 | |
|     auto commit_token = [&](auto type) {
 | |
|         GMLToken token;
 | |
|         token.m_view = m_input.substring_view(token_start_index, m_index - token_start_index);
 | |
|         token.m_type = type;
 | |
|         token.m_start = token_start_position;
 | |
|         token.m_end = m_previous_position;
 | |
|         tokens.append(token);
 | |
|     };
 | |
| 
 | |
|     auto consume_class = [&] {
 | |
|         begin_token();
 | |
|         consume();
 | |
|         commit_token(GMLToken::Type::ClassMarker);
 | |
|         begin_token();
 | |
|         while (is_valid_class_character(peek()))
 | |
|             consume();
 | |
|         commit_token(GMLToken::Type::ClassName);
 | |
|     };
 | |
| 
 | |
|     while (m_index < m_input.length()) {
 | |
|         if (isspace(peek(0))) {
 | |
|             begin_token();
 | |
|             while (isspace(peek()))
 | |
|                 consume();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         // C++ style comments
 | |
|         if (peek(0) && peek(0) == '/' && peek(1) == '/') {
 | |
|             begin_token();
 | |
|             while (peek() && peek() != '\n')
 | |
|                 consume();
 | |
|             commit_token(GMLToken::Type::Comment);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (peek(0) == '{') {
 | |
|             begin_token();
 | |
|             consume();
 | |
|             commit_token(GMLToken::Type::LeftCurly);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (peek(0) == '}') {
 | |
|             begin_token();
 | |
|             consume();
 | |
|             commit_token(GMLToken::Type::RightCurly);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (peek(0) == '@') {
 | |
|             consume_class();
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (is_valid_identifier_start(peek(0))) {
 | |
|             begin_token();
 | |
|             consume();
 | |
|             while (is_valid_identifier_character(peek(0)))
 | |
|                 consume();
 | |
|             commit_token(GMLToken::Type::Identifier);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (peek(0) == ':') {
 | |
|             begin_token();
 | |
|             consume();
 | |
|             commit_token(GMLToken::Type::Colon);
 | |
| 
 | |
|             while (isspace(peek()))
 | |
|                 consume();
 | |
| 
 | |
|             if (peek(0) == '@') {
 | |
|                 consume_class();
 | |
|             } else {
 | |
|                 begin_token();
 | |
|                 while (peek() && peek() != '\n')
 | |
|                     consume();
 | |
|                 commit_token(GMLToken::Type::JsonValue);
 | |
|             }
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         consume();
 | |
|         commit_token(GMLToken::Type::Unknown);
 | |
|     }
 | |
|     return tokens;
 | |
| }
 | |
| 
 | |
| }
 |