ladybird/Userland/Libraries/LibGUI/INILexer.cpp
Brian Gianforcaro 1682f0b760 Everything: Move to SPDX license identifiers in all files.
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 *
2021-04-22 11:22:27 +02:00

140 lines
3.1 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (c) 2020, Hüseyin Aslıtürk <asliturk@hotmail.com>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "INILexer.h"
#include <AK/Vector.h>
#include <ctype.h>
namespace GUI {
IniLexer::IniLexer(const StringView& input)
: m_input(input)
{
}
char IniLexer::peek(size_t offset) const
{
if ((m_index + offset) >= m_input.length())
return 0;
return m_input[m_index + offset];
}
char IniLexer::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;
}
Vector<IniToken> IniLexer::lex()
{
Vector<IniToken> tokens;
size_t token_start_index = 0;
IniPosition token_start_position;
auto emit_token = [&](auto type) {
IniToken token;
token.m_type = type;
token.m_start = m_position;
token.m_end = m_position;
tokens.append(token);
consume();
};
auto begin_token = [&] {
token_start_index = m_index;
token_start_position = m_position;
};
auto commit_token = [&](auto type) {
IniToken token;
token.m_type = type;
token.m_start = token_start_position;
token.m_end = m_previous_position;
tokens.append(token);
};
while (m_index < m_input.length()) {
auto ch = peek();
if (isspace(ch)) {
begin_token();
while (isspace(peek()))
consume();
commit_token(IniToken::Type::Whitespace);
continue;
}
// ;Comment
if (ch == ';') {
begin_token();
while (peek() && peek() != '\n')
consume();
commit_token(IniToken::Type::Comment);
continue;
}
// [Section]
if (ch == '[') {
// [ Token
begin_token();
consume();
commit_token(IniToken::Type::LeftBracket);
// Section
begin_token();
while (peek() && !(peek() == ']' || peek() == '\n'))
consume();
commit_token(IniToken::Type::section);
// ] Token
if (peek() && peek() == ']') {
begin_token();
consume();
commit_token(IniToken::Type::RightBracket);
}
continue;
}
// Empty Line
if (ch == '\n') {
consume();
emit_token(IniToken::Type::Unknown);
continue;
}
// Name=Value
begin_token();
while (peek() && !(peek() == '=' || peek() == '\n'))
consume();
commit_token(IniToken::Type::Name);
if (peek() && peek() == '=') {
begin_token();
consume();
commit_token(IniToken::Type::Equal);
}
if (peek()) {
begin_token();
while (peek() && peek() != '\n')
consume();
commit_token(IniToken::Type::Value);
}
}
return tokens;
}
}