mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 10:12:07 +00:00

Specifically: * Properties must precede sub-object declarations. * There shouldn't be any newlines in-between properties. * There should be a newline in-between each sub-object. * Object declarations must include the curly braces. * There should be a newline between the properties of an object and it's sub-objects.
338 lines
10 KiB
C++
338 lines
10 KiB
C++
/*
|
|
* Copyright (c) 2022, kleines Filmröllchen <filmroellchen@serenityos.org>.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/Concepts.h>
|
|
#include <AK/Error.h>
|
|
#include <AK/Forward.h>
|
|
#include <AK/HashMap.h>
|
|
#include <AK/IterationDecision.h>
|
|
#include <AK/JsonArray.h>
|
|
#include <AK/JsonValue.h>
|
|
#include <AK/NonnullRefPtr.h>
|
|
#include <AK/NonnullRefPtrVector.h>
|
|
#include <AK/RefCounted.h>
|
|
#include <AK/RefPtr.h>
|
|
#include <AK/String.h>
|
|
#include <AK/StringBuilder.h>
|
|
#include <AK/TypeCasts.h>
|
|
#include <LibGUI/GML/Lexer.h>
|
|
|
|
namespace GUI::GML {
|
|
|
|
class Comment;
|
|
class JsonValueNode;
|
|
|
|
// Base of the GML Abstract Syntax Tree (AST).
|
|
class Node : public RefCounted<Node> {
|
|
public:
|
|
virtual ~Node() = default;
|
|
template<typename NodeT>
|
|
requires(IsBaseOf<Node, NodeT>) static ErrorOr<NonnullRefPtr<NodeT>> from_token(Token token)
|
|
{
|
|
return try_make_ref_counted<NodeT>(token.m_view);
|
|
}
|
|
|
|
String to_string() const
|
|
{
|
|
StringBuilder builder;
|
|
format(builder, 0, false);
|
|
return builder.to_string();
|
|
}
|
|
|
|
// Format this AST node with the builder at the given indentation level.
|
|
// is_inline controls whether we are starting on a new line.
|
|
virtual void format(StringBuilder& builder, size_t indentation, bool is_inline) const = 0;
|
|
|
|
// FIXME: We can't change the kind of indentation right now.
|
|
static void indent(StringBuilder& builder, size_t indentation)
|
|
{
|
|
for (size_t i = 0; i < indentation; ++i)
|
|
builder.append(" ");
|
|
}
|
|
};
|
|
|
|
// AST nodes that actually hold data and can be in a KeyValuePair.
|
|
class ValueNode
|
|
: public Node {
|
|
public:
|
|
virtual ~ValueNode() = default;
|
|
};
|
|
|
|
// Single line comments with //.
|
|
class Comment : public Node {
|
|
public:
|
|
Comment(String text)
|
|
: m_text(move(text))
|
|
{
|
|
}
|
|
|
|
virtual void format(StringBuilder& builder, size_t indentation, bool is_inline) const override
|
|
{
|
|
if (is_inline) {
|
|
builder.append(m_text);
|
|
} else {
|
|
indent(builder, indentation);
|
|
builder.append(m_text);
|
|
}
|
|
builder.append("\n");
|
|
}
|
|
virtual ~Comment() override = default;
|
|
|
|
private:
|
|
String m_text {};
|
|
};
|
|
|
|
// Any JSON-like key: value pair.
|
|
class KeyValuePair : public Node {
|
|
public:
|
|
KeyValuePair(String key, NonnullRefPtr<ValueNode> value)
|
|
: m_key(move(key))
|
|
, m_value(move(value))
|
|
{
|
|
}
|
|
virtual ~KeyValuePair() override = default;
|
|
virtual void format(StringBuilder& builder, size_t indentation, bool is_inline) const override
|
|
{
|
|
if (!is_inline)
|
|
indent(builder, indentation);
|
|
builder.appendff("{}: ", m_key);
|
|
m_value->format(builder, indentation, true);
|
|
if (!is_inline)
|
|
builder.append("\n");
|
|
}
|
|
|
|
String key() const { return m_key; }
|
|
NonnullRefPtr<ValueNode> value() const { return m_value; }
|
|
|
|
private:
|
|
String m_key;
|
|
NonnullRefPtr<ValueNode> m_value;
|
|
};
|
|
|
|
// Just a mixin so that we can use JSON values in the AST
|
|
// FIXME: Use a specialized value type for all the possible GML property values. Right now that's all possible JSON values (?)
|
|
class JsonValueNode : public ValueNode
|
|
, public JsonValue {
|
|
|
|
public:
|
|
JsonValueNode(JsonValue const& value)
|
|
: JsonValue(value)
|
|
{
|
|
}
|
|
|
|
virtual void format(StringBuilder& builder, size_t indentation, bool is_inline) const override
|
|
{
|
|
if (!is_inline)
|
|
indent(builder, indentation);
|
|
if (is_array()) {
|
|
// custom array serialization as AK's doesn't pretty-print
|
|
// objects and arrays (we only care about arrays (for now))
|
|
builder.append("[");
|
|
auto first = true;
|
|
as_array().for_each([&](auto& value) {
|
|
if (!first)
|
|
builder.append(", ");
|
|
first = false;
|
|
value.serialize(builder);
|
|
});
|
|
builder.append("]");
|
|
} else {
|
|
serialize(builder);
|
|
}
|
|
if (!is_inline)
|
|
builder.append("\n");
|
|
}
|
|
};
|
|
|
|
// GML class declaration, starting with '@'
|
|
class Object : public ValueNode {
|
|
public:
|
|
Object() = default;
|
|
Object(String name, NonnullRefPtrVector<Node> properties, NonnullRefPtrVector<Node> sub_objects)
|
|
: m_properties(move(properties))
|
|
, m_sub_objects(move(sub_objects))
|
|
, m_name(move(name))
|
|
{
|
|
}
|
|
|
|
virtual ~Object() override = default;
|
|
|
|
StringView name() const { return m_name; }
|
|
void set_name(String name) { m_name = move(name); }
|
|
|
|
ErrorOr<void> add_sub_object_child(NonnullRefPtr<Node> child)
|
|
{
|
|
VERIFY(is<Object>(child.ptr()) || is<Comment>(child.ptr()));
|
|
return m_sub_objects.try_append(move(child));
|
|
}
|
|
|
|
ErrorOr<void> add_property_child(NonnullRefPtr<Node> child)
|
|
{
|
|
VERIFY(is<KeyValuePair>(child.ptr()) || is<Comment>(child.ptr()));
|
|
return m_properties.try_append(move(child));
|
|
}
|
|
|
|
// Does not return key-value pair `layout: ...`!
|
|
template<typename Callback>
|
|
void for_each_property(Callback callback)
|
|
{
|
|
for (auto const& child : m_properties) {
|
|
if (is<KeyValuePair>(child)) {
|
|
auto const& property = static_cast<KeyValuePair const&>(child);
|
|
if (property.key() != "layout" && is<JsonValueNode>(property.value().ptr()))
|
|
callback(property.key(), static_ptr_cast<JsonValueNode>(property.value()));
|
|
}
|
|
}
|
|
}
|
|
|
|
template<typename Callback>
|
|
void for_each_child_object(Callback callback)
|
|
{
|
|
for (NonnullRefPtr<Node> child : m_sub_objects) {
|
|
// doesn't capture layout as intended, as that's behind a kv-pair
|
|
if (is<Object>(child.ptr())) {
|
|
auto object = static_ptr_cast<Object>(child);
|
|
callback(object);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Uses IterationDecision to allow the callback to interrupt the iteration, like a for-loop break.
|
|
template<IteratorFunction<NonnullRefPtr<Object>> Callback>
|
|
void for_each_child_object_interruptible(Callback callback)
|
|
{
|
|
for (NonnullRefPtr<Node> child : m_sub_objects) {
|
|
// doesn't capture layout as intended, as that's behind a kv-pair
|
|
if (is<Object>(child.ptr())) {
|
|
auto object = static_ptr_cast<Object>(child);
|
|
if (callback(object) == IterationDecision::Break)
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
RefPtr<Object> layout_object() const
|
|
{
|
|
for (NonnullRefPtr<Node> child : m_properties) {
|
|
if (is<KeyValuePair>(child.ptr())) {
|
|
auto property = static_ptr_cast<KeyValuePair>(child);
|
|
if (property->key() == "layout") {
|
|
VERIFY(is<Object>(property->value().ptr()));
|
|
return static_ptr_cast<Object>(property->value());
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<ValueNode> get_property(StringView property_name)
|
|
{
|
|
for (NonnullRefPtr<Node> child : m_properties) {
|
|
if (is<KeyValuePair>(child.ptr())) {
|
|
auto property = static_ptr_cast<KeyValuePair>(child);
|
|
if (property->key() == property_name)
|
|
return property->value();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
virtual void format(StringBuilder& builder, size_t indentation, bool is_inline) const override
|
|
{
|
|
if (!is_inline)
|
|
indent(builder, indentation);
|
|
builder.append('@');
|
|
builder.append(m_name);
|
|
builder.append(" {");
|
|
if (!m_properties.is_empty() || !m_sub_objects.is_empty()) {
|
|
builder.append('\n');
|
|
|
|
for (auto const& property : m_properties)
|
|
property.format(builder, indentation + 1, false);
|
|
|
|
if (!m_properties.is_empty() && !m_sub_objects.is_empty())
|
|
builder.append('\n');
|
|
|
|
// This loop is necessary as we need to know what the last child is.
|
|
for (size_t i = 0; i < m_sub_objects.size(); ++i) {
|
|
auto const& child = m_sub_objects[i];
|
|
child.format(builder, indentation + 1, false);
|
|
|
|
if (is<Object>(child) && i != m_sub_objects.size() - 1)
|
|
builder.append('\n');
|
|
}
|
|
|
|
indent(builder, indentation);
|
|
}
|
|
builder.append('}');
|
|
if (!is_inline)
|
|
builder.append('\n');
|
|
}
|
|
|
|
private:
|
|
// Properties and comments
|
|
NonnullRefPtrVector<Node> m_properties;
|
|
// Sub objects and comments
|
|
NonnullRefPtrVector<Node> m_sub_objects;
|
|
String m_name {};
|
|
};
|
|
|
|
class GMLFile : public Node {
|
|
public:
|
|
virtual ~GMLFile() override = default;
|
|
|
|
ErrorOr<void> add_child(NonnullRefPtr<Node> child)
|
|
{
|
|
if (!has_main_class()) {
|
|
if (is<Comment>(child.ptr())) {
|
|
return m_leading_comments.try_append(*static_ptr_cast<Comment>(child));
|
|
}
|
|
if (is<Object>(child.ptr())) {
|
|
m_main_class = static_ptr_cast<Object>(child);
|
|
return {};
|
|
}
|
|
return Error::from_string_literal("Unexpected data before main class");
|
|
}
|
|
// After the main class, only comments are allowed.
|
|
if (!is<Comment>(child.ptr()))
|
|
return Error::from_string_literal("Data not allowed after main class");
|
|
return m_trailing_comments.try_append(*static_ptr_cast<Comment>(child));
|
|
}
|
|
|
|
bool has_main_class() const { return m_main_class != nullptr; }
|
|
|
|
NonnullRefPtrVector<Comment> leading_comments() const { return m_leading_comments; }
|
|
Object& main_class()
|
|
{
|
|
VERIFY(!m_main_class.is_null());
|
|
return *m_main_class.ptr();
|
|
}
|
|
NonnullRefPtrVector<Comment> trailing_comments() const { return m_trailing_comments; }
|
|
|
|
virtual void format(StringBuilder& builder, size_t indentation, [[maybe_unused]] bool is_inline) const override
|
|
{
|
|
for (auto const& comment : m_leading_comments)
|
|
comment.format(builder, indentation, false);
|
|
|
|
if (!m_leading_comments.is_empty())
|
|
builder.append('\n');
|
|
m_main_class->format(builder, indentation, false);
|
|
if (!m_trailing_comments.is_empty())
|
|
builder.append('\n');
|
|
|
|
for (auto const& comment : m_trailing_comments)
|
|
comment.format(builder, indentation, false);
|
|
}
|
|
|
|
private:
|
|
NonnullRefPtrVector<Comment> m_leading_comments;
|
|
RefPtr<Object> m_main_class;
|
|
NonnullRefPtrVector<Comment> m_trailing_comments;
|
|
};
|
|
|
|
}
|