mirror of
https://gitlab.gnome.org/jwestman/blueprint-compiler.git
synced 2025-05-04 15:59:08 -04:00
Add completions for namespaces in the typelib path that can be imported. Accepting the completion automatically adds an import statement.
160 lines
4.7 KiB
Python
160 lines
4.7 KiB
Python
# ui.py
|
|
#
|
|
# Copyright 2022 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
|
|
|
|
from functools import cached_property
|
|
|
|
from .. import gir
|
|
from .common import *
|
|
from .contexts import ScopeCtx
|
|
from .gobject_object import Object
|
|
from .gtk_menu import Menu, menu
|
|
from .gtkbuilder_template import Template
|
|
from .imports import GtkDirective, Import
|
|
from .translation_domain import TranslationDomain
|
|
from .types import TypeName
|
|
|
|
|
|
class UI(AstNode):
|
|
"""The AST node for the entire file"""
|
|
|
|
grammar = [
|
|
GtkDirective,
|
|
ZeroOrMore(Import),
|
|
Optional(TranslationDomain),
|
|
Until(
|
|
AnyOf(
|
|
Template,
|
|
menu,
|
|
Object,
|
|
),
|
|
Eof(),
|
|
),
|
|
]
|
|
|
|
@cached_property
|
|
def gir(self) -> gir.GirContext:
|
|
gir_ctx = gir.GirContext()
|
|
self._gir_errors = []
|
|
|
|
try:
|
|
if gtk := self.children[GtkDirective][0].gir_namespace:
|
|
gir_ctx.add_namespace(gtk)
|
|
except CompileError as e:
|
|
self._gir_errors.append(e)
|
|
|
|
for i in self.children[Import]:
|
|
try:
|
|
if i.gir_namespace is not None:
|
|
gir_ctx.add_namespace(i.gir_namespace)
|
|
else:
|
|
gir_ctx.not_found_namespaces.add(i.namespace)
|
|
except CompileError as e:
|
|
e.range = i.range
|
|
self._gir_errors.append(e)
|
|
|
|
return gir_ctx
|
|
|
|
@property
|
|
def using(self) -> T.List[Import]:
|
|
return self.children[Import]
|
|
|
|
@property
|
|
def gtk_decl(self) -> GtkDirective:
|
|
return self.children[GtkDirective][0]
|
|
|
|
@property
|
|
def translation_domain(self) -> T.Optional[TranslationDomain]:
|
|
domains = self.children[TranslationDomain]
|
|
if len(domains):
|
|
return domains[0]
|
|
else:
|
|
return None
|
|
|
|
@property
|
|
def contents(self) -> T.List[T.Union[Object, Template, Menu]]:
|
|
return [
|
|
child
|
|
for child in self.children
|
|
if isinstance(child, Object)
|
|
or isinstance(child, Template)
|
|
or isinstance(child, Menu)
|
|
]
|
|
|
|
@property
|
|
def template(self) -> T.Optional[Template]:
|
|
if len(self.children[Template]):
|
|
return self.children[Template][0]
|
|
else:
|
|
return None
|
|
|
|
def is_legacy_template(self, id: str) -> bool:
|
|
return (
|
|
id not in self.context[ScopeCtx].objects
|
|
and self.template is not None
|
|
and self.template.class_name.glib_type_name == id
|
|
)
|
|
|
|
def import_range(self, ns: str):
|
|
"""Returns a range to insert a new import statement"""
|
|
pos = self.children[GtkDirective][0].range.end
|
|
|
|
# try to insert alphabetically
|
|
for import_ in self.children[Import]:
|
|
if ns.lower() > import_.namespace.lower():
|
|
pos = import_.range.end
|
|
|
|
return Range(pos, pos, self.group.text)
|
|
|
|
def import_code_action(self, ns: str, version: str) -> CodeAction:
|
|
return CodeAction(
|
|
f"Import {ns} {version}",
|
|
f"\nusing {ns} {version};",
|
|
self.import_range(ns),
|
|
)
|
|
|
|
@cached_property
|
|
def used_imports(self) -> T.Optional[T.Set[str]]:
|
|
def _iter_recursive(node: AstNode):
|
|
yield node
|
|
for child in node.children:
|
|
if isinstance(child, AstNode):
|
|
yield from _iter_recursive(child)
|
|
|
|
result = set()
|
|
for node in _iter_recursive(self):
|
|
if isinstance(node, TypeName):
|
|
ns = node.gir_ns
|
|
if ns is not None:
|
|
result.add(ns.name)
|
|
return result
|
|
|
|
@context(ScopeCtx)
|
|
def scope_ctx(self) -> ScopeCtx:
|
|
return ScopeCtx(node=self)
|
|
|
|
@validate()
|
|
def gir_errors(self):
|
|
# make sure gir is loaded
|
|
self.gir
|
|
if len(self._gir_errors):
|
|
raise MultipleErrors(self._gir_errors)
|
|
|
|
@validate()
|
|
def unique_ids(self):
|
|
self.context[ScopeCtx].validate_unique_ids()
|