mirror of
https://gitlab.gnome.org/jwestman/blueprint-compiler.git
synced 2025-05-04 15:59:08 -04:00
329 lines
8.1 KiB
Python
329 lines
8.1 KiB
Python
# gtk_menus.py
|
|
#
|
|
# Copyright 2021 James Westman <james@jwestman.net>
|
|
#
|
|
# This file is free software; you can redistribute it and/or modify it
|
|
# under the terms of the GNU Lesser General Public License as
|
|
# published by the Free Software Foundation; either version 3 of the
|
|
# License, or (at your option) any later version.
|
|
#
|
|
# This file is distributed in the hope that it will be useful, but
|
|
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
# SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
import typing as T
|
|
|
|
from blueprintcompiler.language.values import StringValue
|
|
|
|
from .common import *
|
|
from .contexts import ValueTypeCtx
|
|
from .gobject_object import RESERVED_IDS
|
|
|
|
|
|
class Menu(AstNode):
|
|
@property
|
|
def gir_class(self):
|
|
return self.root.gir.namespaces["Gtk"].lookup_type("Gio.Menu")
|
|
|
|
@property
|
|
def id(self) -> str:
|
|
return self.tokens["id"]
|
|
|
|
@property
|
|
def signature(self) -> str:
|
|
if self.id:
|
|
return f"Gio.Menu {self.id}"
|
|
else:
|
|
return "Gio.Menu"
|
|
|
|
@property
|
|
def document_symbol(self) -> DocumentSymbol:
|
|
return DocumentSymbol(
|
|
self.tokens["tag"],
|
|
SymbolKind.Object,
|
|
self.range,
|
|
self.group.tokens[self.tokens["tag"]].range,
|
|
self.id,
|
|
)
|
|
|
|
@property
|
|
def tag(self) -> str:
|
|
return self.tokens["tag"]
|
|
|
|
@property
|
|
def items(self) -> T.List[T.Union["Menu", "MenuAttribute"]]:
|
|
return self.children
|
|
|
|
@validate("menu")
|
|
def has_id(self):
|
|
if self.tokens["tag"] == "menu" and self.tokens["id"] is None:
|
|
raise CompileError("Menu requires an ID")
|
|
|
|
@validate("id")
|
|
def object_id_not_reserved(self):
|
|
if self.id in RESERVED_IDS:
|
|
raise CompileWarning(f"{self.id} may be a confusing object ID")
|
|
|
|
@docs("menu")
|
|
def ref_docs_menu(self):
|
|
return get_docs_section("Syntax Menu")
|
|
|
|
@docs("section")
|
|
def ref_docs_section(self):
|
|
return get_docs_section("Syntax Menu")
|
|
|
|
@docs("submenu")
|
|
def ref_docs_submenu(self):
|
|
return get_docs_section("Syntax Menu")
|
|
|
|
@docs("item")
|
|
def ref_docs_item(self):
|
|
if self.tokens["shorthand"]:
|
|
return get_docs_section("Syntax MenuItemShorthand")
|
|
else:
|
|
return get_docs_section("Syntax Menu")
|
|
|
|
|
|
class MenuAttribute(AstNode):
|
|
tag_name = "attribute"
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
return self.tokens["name"]
|
|
|
|
@property
|
|
def value(self) -> StringValue:
|
|
return self.children[StringValue][0]
|
|
|
|
@property
|
|
def document_symbol(self) -> DocumentSymbol:
|
|
return DocumentSymbol(
|
|
self.name,
|
|
SymbolKind.Field,
|
|
self.range,
|
|
(
|
|
self.group.tokens["name"].range
|
|
if self.group.tokens["name"]
|
|
else self.range
|
|
),
|
|
self.value.range.text,
|
|
)
|
|
|
|
@context(ValueTypeCtx)
|
|
def value_type(self) -> ValueTypeCtx:
|
|
return ValueTypeCtx(None)
|
|
|
|
@validate("name")
|
|
def unique(self):
|
|
self.validate_unique_in_parent(
|
|
f"Duplicate attribute '{self.name}'", lambda x: x.name == self.name
|
|
)
|
|
|
|
|
|
menu_child = AnyOf()
|
|
|
|
menu_attribute = Group(
|
|
MenuAttribute,
|
|
[
|
|
UseIdent("name"),
|
|
":",
|
|
Err(StringValue, "Expected string or translated string"),
|
|
Match(";").expected(),
|
|
],
|
|
)
|
|
|
|
menu_section = Group(
|
|
Menu,
|
|
[
|
|
Keyword("section"),
|
|
UseLiteral("tag", "section"),
|
|
Optional(UseIdent("id")),
|
|
Match("{").expected(),
|
|
Until(AnyOf(menu_child, menu_attribute), "}"),
|
|
],
|
|
)
|
|
|
|
menu_submenu = Group(
|
|
Menu,
|
|
[
|
|
Keyword("submenu"),
|
|
UseLiteral("tag", "submenu"),
|
|
Optional(UseIdent("id")),
|
|
Match("{").expected(),
|
|
Until(AnyOf(menu_child, menu_attribute), "}"),
|
|
],
|
|
)
|
|
|
|
menu_item = Group(
|
|
Menu,
|
|
[
|
|
Keyword("item"),
|
|
UseLiteral("tag", "item"),
|
|
Match("{").expected(),
|
|
Until(menu_attribute, "}"),
|
|
],
|
|
)
|
|
|
|
menu_item_shorthand = Group(
|
|
Menu,
|
|
[
|
|
Keyword("item"),
|
|
UseLiteral("tag", "item"),
|
|
UseLiteral("shorthand", True),
|
|
"(",
|
|
Group(
|
|
MenuAttribute,
|
|
[UseLiteral("name", "label"), StringValue],
|
|
),
|
|
Optional(
|
|
[
|
|
",",
|
|
Optional(
|
|
[
|
|
Group(
|
|
MenuAttribute,
|
|
[UseLiteral("name", "action"), StringValue],
|
|
),
|
|
Optional(
|
|
[
|
|
",",
|
|
Group(
|
|
MenuAttribute,
|
|
[UseLiteral("name", "icon"), StringValue],
|
|
),
|
|
]
|
|
),
|
|
]
|
|
),
|
|
]
|
|
),
|
|
Match(")").expected(),
|
|
],
|
|
)
|
|
|
|
menu_child.children = [
|
|
menu_section,
|
|
menu_submenu,
|
|
menu_item_shorthand,
|
|
menu_item,
|
|
]
|
|
|
|
menu: Group = Group(
|
|
Menu,
|
|
[
|
|
Keyword("menu"),
|
|
UseLiteral("tag", "menu"),
|
|
Optional(UseIdent("id")),
|
|
[
|
|
Match("{"),
|
|
Until(
|
|
AnyOf(
|
|
menu_child,
|
|
Fail(
|
|
menu_attribute,
|
|
"Attributes are not permitted at the top level of a menu",
|
|
),
|
|
),
|
|
"}",
|
|
),
|
|
],
|
|
],
|
|
)
|
|
|
|
from .ui import UI
|
|
|
|
|
|
@completer(
|
|
applies_in=[UI],
|
|
matches=new_statement_patterns,
|
|
)
|
|
def menu_completer(_ctx: CompletionContext):
|
|
yield Completion("menu", CompletionItemKind.Snippet, snippet="menu {\n $0\n}")
|
|
|
|
|
|
@completer(
|
|
applies_in=[Menu],
|
|
matches=new_statement_patterns,
|
|
)
|
|
def menu_content_completer(_ctx: CompletionContext):
|
|
yield Completion(
|
|
"submenu",
|
|
CompletionItemKind.Snippet,
|
|
snippet="submenu {\n $0\n}",
|
|
sort_text=get_sort_key(CompletionPriority.CLASS, "1 submenu"),
|
|
)
|
|
yield Completion(
|
|
"section",
|
|
CompletionItemKind.Snippet,
|
|
snippet="section {\n $0\n}",
|
|
sort_text=get_sort_key(CompletionPriority.CLASS, "1 section"),
|
|
)
|
|
yield Completion(
|
|
"item",
|
|
CompletionItemKind.Snippet,
|
|
snippet="item {\n $0\n}",
|
|
sort_text=get_sort_key(CompletionPriority.CLASS, "1 item"),
|
|
)
|
|
yield Completion(
|
|
"item (shorthand)",
|
|
CompletionItemKind.Snippet,
|
|
snippet='item (_("${1:Label}"), "${2:action-name}", "${3:icon-name}")',
|
|
sort_text=get_sort_key(CompletionPriority.CLASS, "0 item (shorthand)"),
|
|
)
|
|
|
|
yield Completion(
|
|
"label",
|
|
CompletionItemKind.Snippet,
|
|
snippet="label: $0;",
|
|
sort_text=get_sort_key(CompletionPriority.OBJECT_MEMBER, "label"),
|
|
)
|
|
yield Completion(
|
|
"action",
|
|
CompletionItemKind.Snippet,
|
|
snippet='action: "$0";',
|
|
sort_text=get_sort_key(CompletionPriority.OBJECT_MEMBER, "action"),
|
|
)
|
|
yield Completion(
|
|
"icon",
|
|
CompletionItemKind.Snippet,
|
|
snippet='icon: "$0";',
|
|
sort_text=get_sort_key(CompletionPriority.OBJECT_MEMBER, "icon"),
|
|
)
|
|
|
|
|
|
@decompiler("menu")
|
|
def decompile_menu(ctx, gir, id=None):
|
|
if id:
|
|
ctx.print(f"menu {id} {{")
|
|
else:
|
|
ctx.print("menu {")
|
|
|
|
|
|
@decompiler("submenu")
|
|
def decompile_submenu(ctx, gir, id=None):
|
|
if id:
|
|
ctx.print(f"submenu {id} {{")
|
|
else:
|
|
ctx.print("submenu {")
|
|
|
|
|
|
@decompiler("item", parent_tag="menu")
|
|
def decompile_item(ctx, gir, id=None):
|
|
if id:
|
|
ctx.print(f"item {id} {{")
|
|
else:
|
|
ctx.print("item {")
|
|
|
|
|
|
@decompiler("section")
|
|
def decompile_section(ctx, gir, id=None):
|
|
if id:
|
|
ctx.print(f"section {id} {{")
|
|
else:
|
|
ctx.print("section {")
|