Compare commits

..

No commits in common. "ab9d902cc5e63ee503a627c8406240c86a4556f9" and "3efb2f9f4ec8f949efee071248d49c25b50582e0" have entirely different histories.

8 changed files with 23 additions and 265 deletions

View file

@ -27,9 +27,7 @@ from .values import ArrayValue, ObjectValue, Value, VariantValue
class Property(AstNode):
grammar = Statement(
UseIdent("name"),
":",
AnyOf(Binding, VariantValue, ObjectValue, Value, ArrayValue),
UseIdent("name"), ":", AnyOf(Binding, VariantValue, ObjectValue, Value, ArrayValue)
)
@property

View file

@ -98,7 +98,7 @@ class MenuAttribute(AstNode):
return self.tokens["name"]
@property
def value(self) -> T.Union[StringValue, VariantValue]:
def value(self) -> StringValue | VariantValue:
if len(self.children[StringValue]) > 0:
return self.children[StringValue][0]
elif len(self.children[VariantValue]) > 0:
@ -137,10 +137,7 @@ menu_attribute = Group(
[
UseIdent("name"),
":",
Err(
AnyOf(StringValue, VariantValue),
"Expected string, translated string, or variant",
),
Err(AnyOf(StringValue, VariantValue), "Expected string or translated string"),
Match(";").expected(),
],
)

View file

@ -26,12 +26,6 @@ from .common import *
from .contexts import ScopeCtx, ValueTypeCtx
from .gobject_object import Object
from .types import TypeName
from .variant import VarContent
import gi
gi.require_version("GLib", "2.0")
from gi.repository import GLib
class Translated(AstNode):
@ -377,7 +371,6 @@ class IdentLiteral(AstNode):
else:
return None
class VariantValue(AstNode):
grammar = [
"variant",
@ -385,8 +378,8 @@ class VariantValue(AstNode):
UseQuoted("type"),
">",
"(",
Err(VarContent, "Invalid variant content!"),
")",
UseQuoted("value"),
")"
]
@property
@ -395,7 +388,7 @@ class VariantValue(AstNode):
@property
def var_value(self) -> str:
return self.children[0].content
return self.tokens["value"]
@validate()
def validate_for_type(self) -> None:
@ -411,34 +404,12 @@ class VariantValue(AstNode):
raise CompileError(f"Cannot convert variant to number")
elif isinstance(expected_type, gir.StringType):
raise CompileError("Cannot convert variant to string")
elif (
isinstance(expected_type, gir.Boxed)
and expected_type.full_name == "GLib.Variant"
):
elif isinstance(expected_type, gir.Boxed) and expected_type.full_name == "GLib.Variant":
pass
else:
raise CompileError(f"Cannot convert variant into {expected_type.full_name}")
pass
@validate("type")
def validate_type(self):
if not GLib.VariantType.string_is_valid(self.var_type):
raise CompileError(f"`{self.var_type}` is not a valid variant type")
@validate()
def validate_content(self):
if not GLib.VariantType.string_is_valid(self.var_type):
return
try:
var_ty = GLib.VariantType.new(self.var_type)
var_val = GLib.Variant.parse(var_ty, self.var_value)
except GLib.GError as error:
raise CompileError(f"Variant did not match specified type: {error}")
pass
class Literal(AstNode):
grammar = AnyOf(
TypeLiteral,

View file

@ -1,122 +0,0 @@
import typing as T
from blueprintcompiler.gir import ArrayType
from blueprintcompiler.lsp_utils import SemanticToken
from .common import *
from .contexts import ScopeCtx, ValueTypeCtx
from .gobject_object import Object
from .types import TypeName
VAR_CONTENT_HOOKS: list[T.Any] = []
class VarContent(AstNode):
grammar = AnyOf(*VAR_CONTENT_HOOKS)
@property
def content(self) -> str:
return self.children[0].content
class VarContentBool(AstNode):
grammar = AnyOf(
[Keyword("true"), UseLiteral("value", True)],
[Keyword("false"), UseLiteral("value", False)],
)
@property
def content(self) -> str:
if self.tokens["value"]:
return "true"
else:
return "false"
class VarContentString(AstNode):
grammar = UseQuoted("value")
@property
def content(self) -> str:
return utils.escape_quote(self.tokens["value"])
class VarContentNumber(AstNode):
grammar = UseNumberText("value")
@property
def content(self) -> str:
return self.tokens["value"]
class VarContentTuple(AstNode):
grammar = ["(", Delimited(VarContent, ","), ")"]
@property
def content(self) -> str:
inner = ", ".join(child.content for child in self.children)
return f"({inner})"
class VarContentArray(AstNode):
grammar = ["[", Delimited(VarContent, ","), "]"]
@property
def content(self) -> str:
inner = ", ".join(child.content for child in self.children)
return f"[{inner}]"
class VarContentDictEntry(AstNode):
grammar = ["{", VarContent, ",", VarContent, "}"]
@property
def content(self):
return f"{{{self.children[0].content}, {self.children[1].content}}}"
class VarContentDict(AstNode):
grammar = ["{", Delimited([VarContent, ":", VarContent], ","), "}"]
@property
def content(self) -> str:
inner = ", ".join(
f"{key.content}: {value.content}"
for (key, value) in utils.iter_batched(self.children, 2, strict=True)
)
return f"{{{inner}}}"
class VarContentVariant(AstNode):
grammar = ["<", VarContent, ">"]
@property
def content(self) -> str:
return f"<{self.children[0].content}>"
class VarContentMaybe(AstNode):
grammar = AnyOf(
[Keyword("just"), VarContent],
[Keyword("nothing")],
)
@property
def content(self) -> str:
if self.children[0] is not None:
return f"just {self.children[0].content}"
else:
return "nothing"
VarContent.grammar.children = [
VarContentString,
VarContentNumber,
VarContentBool,
VarContentMaybe,
VarContentTuple,
VarContentDict,
VarContentDictEntry,
VarContentArray,
VarContentVariant,
]

View file

@ -19,7 +19,6 @@
import typing as T
from dataclasses import dataclass
import itertools
class Colors:
@ -155,18 +154,3 @@ def unescape_quote(string: str) -> str:
i += 1
return result
def iter_batched(iterable, n, *, strict=False):
"""
Replacement for `itertools.batched()` since the testing infrastructure
uses Python 3.9 at the moment. Copied directly off of the Python docs.
"""
# batched('ABCDEFG', 3) → ABC DEF G
if n < 1:
raise ValueError("n must be at least one")
iterator = iter(iterable)
while batch := tuple(itertools.islice(iterator, n)):
if strict and len(batch) != n:
raise ValueError("batched(): incomplete batch")
yield batch

View file

@ -1,42 +0,0 @@
using Gtk 4.0;
$BlueprintTestObject {
// test-one: variant<"s">("one");
test-zero: variant<"b">(true);
test-one: variant<"s">("one");
test-two: variant<"i">(2);
test-three: variant<"(ii)">((3, 4));
test-four: variant<"ai">([5, 6]);
test-five: variant<"{sv}">({"key", <"value">});
test-six: variant<"a{ss}">({
"GLib": "2.24",
"Gtk": "4.16"
});
test-seven: variant<"ams">([just "2", nothing]);
}
menu test_menu {
submenu {
label: "Test menu";
item {
label: "Option 1";
action: "app.test_menu.set_action";
target: variant<"y">(1);
}
item {
label: "Option 2";
action: "app.test_menu.set_action";
target: variant<"y">(2);
}
item {
label: "Option 3";
action: "app.test_menu.set_action";
target: variant<"y">(3);
}
item {
label: "Option 4";
action: "app.test_menu.set_action";
target: variant<"y">(4);
}
}
}

View file

@ -1,44 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<!--
DO NOT EDIT!
This file was @generated by blueprint-compiler. Instead, edit the
corresponding .blp file and regenerate this file with blueprint-compiler.
-->
<interface>
<requires lib="gtk" version="4.0"/>
<object class="BlueprintTestObject">
<property name="test-zero" type="b">true</property>
<property name="test-one" type="s">"one"</property>
<property name="test-two" type="i">2</property>
<property name="test-three" type="(ii)">(3, 4)</property>
<property name="test-four" type="ai">[5, 6]</property>
<property name="test-five" type="{sv}">{"key", &lt;"value"&gt;}</property>
<property name="test-six" type="a{ss}">{"GLib": "2.24", "Gtk": "4.16"}</property>
<property name="test-seven" type="ams">[just "2", nothing]</property>
</object>
<menu id="test_menu">
<submenu>
<attribute name="label">Test menu</attribute>
<item>
<attribute name="label">Option 1</attribute>
<attribute name="action">app.test_menu.set_action</attribute>
<attribute name="target" type="y">1</attribute>
</item>
<item>
<attribute name="label">Option 2</attribute>
<attribute name="action">app.test_menu.set_action</attribute>
<attribute name="target" type="y">2</attribute>
</item>
<item>
<attribute name="label">Option 3</attribute>
<attribute name="action">app.test_menu.set_action</attribute>
<attribute name="target" type="y">3</attribute>
</item>
<item>
<attribute name="label">Option 4</attribute>
<attribute name="action">app.test_menu.set_action</attribute>
<attribute name="target" type="y">4</attribute>
</item>
</submenu>
</menu>
</interface>

16
variant-test.blp Normal file
View file

@ -0,0 +1,16 @@
using Gtk 4.0;
menu root {
submenu {
name: "one";
item {
action: "app.foo_bar";
target: variant<"s">("\"one\"");
}
}
}
Button {
action-name: "app.shave_yak";
action-target: variant<"y">("8");
}