mirror of
https://gitlab.gnome.org/jwestman/blueprint-compiler.git
synced 2025-05-04 15:59:08 -04:00
Add dialog actions
This commit is contained in:
parent
bac008296a
commit
8ba898e354
10 changed files with 115 additions and 3 deletions
|
@ -47,6 +47,9 @@ class AstNode:
|
||||||
self.tokens = ChainMap(tokens, defaultdict(lambda: None))
|
self.tokens = ChainMap(tokens, defaultdict(lambda: None))
|
||||||
self.incomplete = incomplete
|
self.incomplete = incomplete
|
||||||
|
|
||||||
|
self._unique_id: str | None = None
|
||||||
|
self._next_unique_id: int = 0
|
||||||
|
|
||||||
self.parent = None
|
self.parent = None
|
||||||
for child in self.children:
|
for child in self.children:
|
||||||
child.parent = self
|
child.parent = self
|
||||||
|
@ -71,6 +74,13 @@ class AstNode:
|
||||||
else:
|
else:
|
||||||
return self.parent.parent_by_type(type)
|
return self.parent.parent_by_type(type)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def unique_id(self):
|
||||||
|
if self._unique_id is None:
|
||||||
|
self.root._next_unique_id += 1
|
||||||
|
self._unique_id = "__" + str(self.root._next_unique_id)
|
||||||
|
return self._unique_id
|
||||||
|
|
||||||
@lazy_prop
|
@lazy_prop
|
||||||
def errors(self):
|
def errors(self):
|
||||||
return list(self._get_errors())
|
return list(self._get_errors())
|
||||||
|
|
|
@ -41,6 +41,15 @@ class ObjectContent(AstNode):
|
||||||
for x in self.children:
|
for x in self.children:
|
||||||
x.emit_xml(xml)
|
x.emit_xml(xml)
|
||||||
|
|
||||||
|
if self.parent.gir_class and self.parent.gir_class.assignable_to(self.root.gir.get_type("Dialog", "Gtk")):
|
||||||
|
action_widgets = [widget for widget in self.children if hasattr(widget, "is_action_widget") and widget.is_action_widget]
|
||||||
|
if len(action_widgets):
|
||||||
|
xml.start_tag("action-widgets")
|
||||||
|
for widget in action_widgets:
|
||||||
|
widget.emit_action_widget(xml)
|
||||||
|
xml.end_tag()
|
||||||
|
|
||||||
|
|
||||||
class Object(AstNode):
|
class Object(AstNode):
|
||||||
grammar: T.Any = [
|
grammar: T.Any = [
|
||||||
class_name,
|
class_name,
|
||||||
|
@ -48,6 +57,13 @@ class Object(AstNode):
|
||||||
ObjectContent,
|
ObjectContent,
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self):
|
||||||
|
if self.tokens["id"] is None:
|
||||||
|
if hasattr(self.parent, "child_needs_id") and self.parent.child_needs_id:
|
||||||
|
return self.unique_id
|
||||||
|
return self.tokens["id"]
|
||||||
|
|
||||||
@validate("namespace")
|
@validate("namespace")
|
||||||
def gir_ns_exists(self):
|
def gir_ns_exists(self):
|
||||||
if not self.tokens["ignore_gir"]:
|
if not self.tokens["ignore_gir"]:
|
||||||
|
@ -84,7 +100,7 @@ class Object(AstNode):
|
||||||
def emit_xml(self, xml: XmlEmitter):
|
def emit_xml(self, xml: XmlEmitter):
|
||||||
xml.start_tag("object", **{
|
xml.start_tag("object", **{
|
||||||
"class": self.gir_class.glib_type_name if self.gir_class else self.tokens["class_name"],
|
"class": self.gir_class.glib_type_name if self.gir_class else self.tokens["class_name"],
|
||||||
"id": self.tokens["id"],
|
"id": self.id,
|
||||||
})
|
})
|
||||||
for child in self.children:
|
for child in self.children:
|
||||||
child.emit_xml(xml)
|
child.emit_xml(xml)
|
||||||
|
|
|
@ -26,17 +26,44 @@ class Child(AstNode):
|
||||||
grammar = [
|
grammar = [
|
||||||
Optional([
|
Optional([
|
||||||
"[",
|
"[",
|
||||||
|
AnyOf(
|
||||||
|
[
|
||||||
|
Keyword("action"),
|
||||||
|
"response", "=", AnyOf(UseNumber("response_id"), UseIdent("response_enum")),
|
||||||
|
],
|
||||||
|
[
|
||||||
Optional(["internal-child", UseLiteral("internal_child", True)]),
|
Optional(["internal-child", UseLiteral("internal_child", True)]),
|
||||||
UseIdent("child_type").expected("a child type"),
|
UseIdent("child_type"),
|
||||||
|
]
|
||||||
|
),
|
||||||
"]",
|
"]",
|
||||||
]),
|
]),
|
||||||
Object,
|
Object,
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def child_needs_id(self):
|
||||||
|
return self.is_action_widget
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_action_widget(self):
|
||||||
|
return self.tokens["action"] is not None
|
||||||
|
|
||||||
|
@validate("action")
|
||||||
|
def action_widget(self):
|
||||||
|
if self.is_action_widget:
|
||||||
|
parent = self.parent_by_type(Object).gir_class
|
||||||
|
dialog = self.root.gir.get_type("Dialog", "Gtk")
|
||||||
|
info_bar = self.root.gir.get_type("InfoBar", "Gtk")
|
||||||
|
if not (parent is None or parent.assignable_to(dialog) or parent.assignable_to(info_bar)):
|
||||||
|
raise CompileError(f"Parent type {parent.full_name} does not have action widgets")
|
||||||
|
|
||||||
def emit_xml(self, xml: XmlEmitter):
|
def emit_xml(self, xml: XmlEmitter):
|
||||||
child_type = internal_child = None
|
child_type = internal_child = None
|
||||||
if self.tokens["internal_child"]:
|
if self.tokens["internal_child"]:
|
||||||
internal_child = self.tokens["child_type"]
|
internal_child = self.tokens["child_type"]
|
||||||
|
elif self.tokens["action"]:
|
||||||
|
child_type = "action"
|
||||||
else:
|
else:
|
||||||
child_type = self.tokens["child_type"]
|
child_type = self.tokens["child_type"]
|
||||||
xml.start_tag("child", type=child_type, internal_child=internal_child)
|
xml.start_tag("child", type=child_type, internal_child=internal_child)
|
||||||
|
@ -44,6 +71,24 @@ class Child(AstNode):
|
||||||
child.emit_xml(xml)
|
child.emit_xml(xml)
|
||||||
xml.end_tag()
|
xml.end_tag()
|
||||||
|
|
||||||
|
@validate("response_enum")
|
||||||
|
def valid_response_enum(self):
|
||||||
|
if response := self.tokens["response_enum"]:
|
||||||
|
if response not in self.root.gir.get_type("ResponseType", "Gtk").members:
|
||||||
|
raise CompileError(f"{response} is not a member of Gtk.ResponseType")
|
||||||
|
|
||||||
|
@docs("response_enum")
|
||||||
|
def response_enum_docs(self):
|
||||||
|
member = self.root.gir.get_type("ResponseType", "Gtk").members.get(self.tokens["response_enum"])
|
||||||
|
if member:
|
||||||
|
return member.doc
|
||||||
|
|
||||||
|
def emit_action_widget(self, xml: XmlEmitter):
|
||||||
|
if self.is_action_widget:
|
||||||
|
xml.start_tag("action-widget", response=self.tokens["response_id"] or self.tokens["response_enum"])
|
||||||
|
xml.put_text(self.children[Object][0].id)
|
||||||
|
xml.end_tag()
|
||||||
|
|
||||||
|
|
||||||
@decompiler("child")
|
@decompiler("child")
|
||||||
def decompile_child(ctx, gir, type=None, internal_child=None):
|
def decompile_child(ctx, gir, type=None, internal_child=None):
|
||||||
|
|
6
tests/sample_errors/action_parent.blp
Normal file
6
tests/sample_errors/action_parent.blp
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
using Gtk 4.0;
|
||||||
|
|
||||||
|
Box {
|
||||||
|
[action response=ok]
|
||||||
|
Label {}
|
||||||
|
}
|
1
tests/sample_errors/action_parent.err
Normal file
1
tests/sample_errors/action_parent.err
Normal file
|
@ -0,0 +1 @@
|
||||||
|
4,4,6,Parent type Gtk.Box does not have action widgets
|
6
tests/sample_errors/action_response.blp
Normal file
6
tests/sample_errors/action_response.blp
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
using Gtk 4.0;
|
||||||
|
|
||||||
|
Dialog {
|
||||||
|
[action response=foo]
|
||||||
|
Button {}
|
||||||
|
}
|
1
tests/sample_errors/action_response.err
Normal file
1
tests/sample_errors/action_response.err
Normal file
|
@ -0,0 +1 @@
|
||||||
|
4,20,3,foo is not a member of Gtk.ResponseType
|
8
tests/samples/actions.blp
Normal file
8
tests/samples/actions.blp
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
using Gtk 4.0;
|
||||||
|
|
||||||
|
Gtk.Dialog {
|
||||||
|
[action response=1]
|
||||||
|
Button {}
|
||||||
|
[action response=cancel]
|
||||||
|
Button cancel {}
|
||||||
|
}
|
16
tests/samples/actions.ui
Normal file
16
tests/samples/actions.ui
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<interface>
|
||||||
|
<requires lib="gtk" version="4.0"/>
|
||||||
|
<object class="GtkDialog">
|
||||||
|
<child type="action">
|
||||||
|
<object class="GtkButton" id="__1"></object>
|
||||||
|
</child>
|
||||||
|
<child type="action">
|
||||||
|
<object class="GtkButton" id="cancel"></object>
|
||||||
|
</child>
|
||||||
|
<action-widgets>
|
||||||
|
<action-widget response="1">__1</action-widget>
|
||||||
|
<action-widget response="cancel">cancel</action-widget>
|
||||||
|
</action-widgets>
|
||||||
|
</object>
|
||||||
|
</interface>
|
|
@ -111,6 +111,7 @@ class TestSamples(unittest.TestCase):
|
||||||
|
|
||||||
def test_samples(self):
|
def test_samples(self):
|
||||||
self.assert_sample("accessibility")
|
self.assert_sample("accessibility")
|
||||||
|
self.assert_sample("actions")
|
||||||
self.assert_sample("binding")
|
self.assert_sample("binding")
|
||||||
self.assert_sample("child_type")
|
self.assert_sample("child_type")
|
||||||
self.assert_sample("combo_box_text")
|
self.assert_sample("combo_box_text")
|
||||||
|
@ -141,6 +142,8 @@ class TestSamples(unittest.TestCase):
|
||||||
self.assert_sample_error("a11y_prop_dne")
|
self.assert_sample_error("a11y_prop_dne")
|
||||||
self.assert_sample_error("a11y_prop_obj_dne")
|
self.assert_sample_error("a11y_prop_obj_dne")
|
||||||
self.assert_sample_error("a11y_prop_type")
|
self.assert_sample_error("a11y_prop_type")
|
||||||
|
self.assert_sample_error("action_parent")
|
||||||
|
self.assert_sample_error("action_response")
|
||||||
self.assert_sample_error("class_assign")
|
self.assert_sample_error("class_assign")
|
||||||
self.assert_sample_error("class_dne")
|
self.assert_sample_error("class_dne")
|
||||||
self.assert_sample_error("consecutive_unexpected_tokens")
|
self.assert_sample_error("consecutive_unexpected_tokens")
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue