Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat custom diagram #159

Open
wants to merge 24 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
a584e6f
feat: Implement custom_diagram
huyenngn Nov 9, 2024
c65647a
fix: Diagram target dictates unified edge direction
huyenngn Nov 9, 2024
cf6fdca
feat(custom_diagram): Add recursion option
huyenngn Nov 10, 2024
40c009c
fix: Move edges to owners
huyenngn Nov 11, 2024
bfa43bc
docs: Add docs for custom diagram
huyenngn Nov 11, 2024
be8a5b8
feat(custom_diagram): Add nested recursion and recursion depth
huyenngn Nov 12, 2024
810ca66
docs: Add expamles for custom_diagram
huyenngn Nov 12, 2024
8e6baea
fix: Fix recursion depth
huyenngn Nov 18, 2024
d7e94b9
fix: Straighten target edge
huyenngn Nov 18, 2024
d6592a0
feat(context-diagram): Add support for PhysicalPorts
huyenngn Nov 19, 2024
261ea06
docs: Add PhysicalPort to docs
huyenngn Nov 19, 2024
2a5f966
fix: Apply code review suggestions
huyenngn Nov 25, 2024
6305008
fix: Apply suggestions from code review
huyenngn Nov 25, 2024
e4dfd0d
fix: Fix minimum size calculation of boxes
huyenngn Nov 26, 2024
f091a65
refactor: Implement generic make_owner_boxes function
huyenngn Nov 26, 2024
bed6a30
fix: Fix mutability bug
huyenngn Nov 27, 2024
c744a6b
feat: Add support for python lambda filter
huyenngn Nov 27, 2024
3f99c27
fix: Add custom_diagram attribute to Class elements
huyenngn Dec 2, 2024
5398b91
refactor: Add generic make_owner_box method
huyenngn Dec 9, 2024
ad39f99
refactor(custom_diagram): Take iterable as collection
huyenngn Dec 9, 2024
d2162d4
docs(custom_diagram): Add example to docs
huyenngn Dec 9, 2024
51772a8
docs: Update custom_diagram docs
huyenngn Dec 16, 2024
fd3b4b1
fix: Remove duplicates from port context
ewuerger Jan 6, 2025
e492ffe
fix: Change custom diagram names
huyenngn Jan 8, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
39 changes: 39 additions & 0 deletions capellambse_context_diagrams/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -65,10 +65,12 @@ def init() -> None:
"""Initialize the extension."""
register_classes()
register_interface_context()
register_physical_port_context()
register_tree_view()
register_realization_view()
register_data_flow_view()
register_cable_tree_view()
register_custom_diagram()
# register_functional_context() XXX: Future


Expand Down Expand Up @@ -247,6 +249,15 @@ def register_functional_context() -> None:
)


def register_physical_port_context() -> None:
"""Add the `context_diagram` attribute to `PhysicalPort`s."""
m.set_accessor(
cs.PhysicalPort,
ATTR_NAME,
context.PhysicalPortContextAccessor(DiagramType.PAB.value, {}),
)


def register_tree_view() -> None:
"""Add the ``tree_view`` attribute to ``Class``es."""
m.set_accessor(
Expand Down Expand Up @@ -313,3 +324,31 @@ def register_cable_tree_view() -> None:
{},
),
)


def register_custom_diagram() -> None:
"""Add the `custom_diagram` attribute to `ModelObject`s."""
supported_classes: list[tuple[type[m.ModelElement], DiagramType]] = [
(oa.Entity, DiagramType.OAB),
(oa.OperationalActivity, DiagramType.OAB),
(oa.OperationalCapability, DiagramType.OCB),
(oa.CommunicationMean, DiagramType.OAB),
(sa.Mission, DiagramType.MCB),
(sa.Capability, DiagramType.MCB),
(sa.SystemComponent, DiagramType.SAB),
(sa.SystemFunction, DiagramType.SAB),
(la.LogicalComponent, DiagramType.LAB),
(la.LogicalFunction, DiagramType.LAB),
(pa.PhysicalComponent, DiagramType.PAB),
(pa.PhysicalFunction, DiagramType.PAB),
(cs.PhysicalLink, DiagramType.PAB),
(cs.PhysicalPort, DiagramType.PAB),
(fa.ComponentExchange, DiagramType.SAB),
(information.Class, DiagramType.CDB),
]
for class_, dgcls in supported_classes:
m.set_accessor(
class_,
"custom_diagram",
context.CustomContextAccessor(dgcls.value, {}),
)
299 changes: 299 additions & 0 deletions capellambse_context_diagrams/collectors/custom.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,299 @@
# SPDX-FileCopyrightText: 2022 Copyright DB InfraGO AG and the capellambse-context-diagrams contributors
# SPDX-License-Identifier: Apache-2.0

"""This module defines the collector for the CustomDiagram."""
from __future__ import annotations

import builtins
import collections.abc as cabc
import copy
import typing as t

import capellambse
import capellambse.model as m

from .. import _elkjs, context
from . import generic, makers


def _is_edge(obj: m.ModelElement) -> bool:
if hasattr(obj, "source") and hasattr(obj, "target"):
return True
return False


def _is_port(obj: m.ModelElement) -> bool:
if obj.xtype.endswith("Port"):
return True
return False


class CustomCollector:
"""Collect the context for a custom diagram."""

def __init__(
self,
diagram: context.CustomDiagram,
params: dict[str, t.Any],
) -> None:
self.diagram = diagram
self.target: m.ModelElement = self.diagram.target

self.boxable_target: m.ModelElement
if _is_port(self.target):
self.boxable_target = self.target.owner
elif _is_edge(self.target):
self.boxable_target = self.target.source.owner
else:
self.boxable_target = self.target

self.data = makers.make_diagram(diagram)
self.params = params
self.collection = self.diagram._collect
self.boxes: dict[str, _elkjs.ELKInputChild] = {}
self.edges: dict[str, _elkjs.ELKInputEdge] = {}
self.ports: dict[str, _elkjs.ELKInputPort] = {}
self.boxes_to_delete: set[str] = set()

if self.diagram._display_parent_relation:
self.edge_owners: dict[str, str] = {}
self.diagram_target_owners = list(
generic.get_all_owners(self.boxable_target)
)
self.common_owners: set[str] = set()

if self.diagram._unify_edge_direction != "NONE":
self.directions: dict[str, bool] = {}
self.min_heights: dict[str, dict[str, float]] = {}

def __call__(self) -> _elkjs.ELKInputData:
if _is_port(self.target):
self._make_port_and_owner(self.target)
else:
self._make_target(self.target)

if target_edge := self.edges.get(self.target.uuid):
target_edge.layoutOptions = copy.deepcopy(
_elkjs.EDGE_STRAIGHTENING_LAYOUT_OPTIONS
)

if self.diagram._unify_edge_direction == "UNIFORM":
self.directions[self.boxable_target.uuid] = False

for elem in self.collection:
self._make_target(elem)

if self.diagram._display_parent_relation:
current = self.boxable_target
while (
current
and self.common_owners
and getattr(current, "owner", None) is not None
and not isinstance(current.owner, generic.PackageTypes)
):
self.common_owners.discard(current.uuid)
current = generic.make_owner_box(
current, self._make_box, self.boxes, self.boxes_to_delete
)
self.common_owners.discard(current.uuid)
huyenngn marked this conversation as resolved.
Show resolved Hide resolved
for edge_uuid, box_uuid in self.edge_owners.items():
if box := self.boxes.get(box_uuid):
box.edges.append(self.edges.pop(edge_uuid))

self._fix_box_heights()
for uuid in self.boxes_to_delete:
del self.boxes[uuid]
return self._get_data()

def _get_data(self) -> t.Any:
self.data.children = list(self.boxes.values())
self.data.edges = list(self.edges.values())
return self.data

def _fix_box_heights(self) -> None:
if self.diagram._unify_edge_direction != "NONE":
for uuid, min_heights in self.min_heights.items():
box = self.boxes[uuid]
box.height = max(box.height, sum(min_heights.values()))
else:
for uuid, min_heights in self.min_heights.items():
box = self.boxes[uuid]
box.height = max([box.height] + list(min_heights.values()))

def _make_target(
self, obj: m.ModelElement
) -> _elkjs.ELKInputChild | _elkjs.ELKInputEdge | None:
if _is_edge(obj):
return self._make_edge_and_ports(obj)
return self._make_box(obj, slim_width=self.diagram._slim_center_box)

def _make_box(
self,
obj: m.ModelElement,
**kwargs: t.Any,
) -> _elkjs.ELKInputChild:
if box := self.boxes.get(obj.uuid):
return box
box = makers.make_box(
obj,
no_symbol=self.diagram._display_symbols_as_boxes,
**kwargs,
)
self.boxes[obj.uuid] = box
if self.diagram._display_unused_ports:
for attr in generic.DIAGRAM_TYPE_TO_CONNECTOR_NAMES[
self.diagram.type
]:
for port in getattr(obj, attr, []):
self._make_port_and_owner(port)
if self.diagram._display_parent_relation:
self.common_owners.add(
generic.make_owner_boxes(
obj,
self.diagram_target_owners,
self._make_box,
self.boxes,
self.boxes_to_delete,
)
)
return box

def _make_edge_and_ports(
self,
edge_obj: m.ModelElement,
) -> _elkjs.ELKInputEdge | None:
if self.edges.get(edge_obj.uuid):
return None
src_obj = edge_obj.source
tgt_obj = edge_obj.target
src_owner = src_obj.owner
tgt_owner = tgt_obj.owner
src_owners = list(generic.get_all_owners(src_obj))
tgt_owners = list(generic.get_all_owners(tgt_obj))
if self.diagram._hide_direct_children:
if (
self.boxable_target.uuid in src_owners
or self.boxable_target.uuid in tgt_owners
):
return None
if self.diagram._display_parent_relation:
common_owner = None
for owner in src_owners:
if owner in tgt_owners:
common_owner = owner
break
if common_owner:
self.edge_owners[edge_obj.uuid] = common_owner

if self._need_switch(
src_owners, tgt_owners, src_owner.uuid, tgt_owner.uuid
):
src_obj, tgt_obj = tgt_obj, src_obj
src_owner, tgt_owner = tgt_owner, src_owner

if not self.ports.get(src_obj.uuid):
port = self._make_port_and_owner(src_obj)
self._update_min_heights(src_owner.uuid, "right", port)
if not self.ports.get(tgt_obj.uuid):
port = self._make_port_and_owner(tgt_obj)
self._update_min_heights(tgt_owner.uuid, "left", port)

edge = _elkjs.ELKInputEdge(
id=edge_obj.uuid,
sources=[src_obj.uuid],
targets=[tgt_obj.uuid],
labels=makers.make_label(
edge_obj.name,
),
)
self.edges[edge_obj.uuid] = edge
return edge

def _update_min_heights(
self, owner_uuid: str, side: str, port: _elkjs.ELKInputPort
) -> None:
self.min_heights.setdefault(owner_uuid, {"left": 0.0, "right": 0.0})[
side
] += makers.PORT_SIZE + max(
2 * makers.PORT_PADDING,
sum(label.height for label in port.labels),
)

def _need_switch(
self,
src_owners: list[str],
tgt_owners: list[str],
src_uuid: str,
tgt_uuid: str,
) -> bool:
if self.diagram._unify_edge_direction == "SMART":
if src_uuid != self.boxable_target.uuid:
src_uncommon = [
owner for owner in src_owners if owner not in tgt_owners
][-1]
src_dir = self.directions.setdefault(src_uncommon, False)
else:
src_dir = None
if tgt_uuid != self.boxable_target.uuid:
tgt_uncommon = [
owner for owner in tgt_owners if owner not in src_owners
][-1]
tgt_dir = self.directions.setdefault(tgt_uncommon, True)
else:
tgt_dir = None
if (src_dir is True) or (tgt_dir is False):
return True
elif self.diagram._unify_edge_direction == "UNIFORM":
src_dir = self.directions.get(src_uuid)
tgt_dir = self.directions.get(tgt_uuid)
if (src_dir is None) and (tgt_dir is None):
self.directions[src_uuid] = False
self.directions[tgt_uuid] = True
elif src_dir is None:
self.directions[src_uuid] = not tgt_dir
elif tgt_dir is None:
self.directions[tgt_uuid] = not src_dir
if self.directions[src_uuid]:
return True
elif self.diagram._unify_edge_direction == "TREE":
src_dir = self.directions.get(src_uuid)
tgt_dir = self.directions.get(tgt_uuid)
if (src_dir is None) and (tgt_dir is None):
self.directions[src_uuid] = True
self.directions[tgt_uuid] = True
elif src_dir is None:
self.directions[src_uuid] = True
return True
elif tgt_dir is None:
self.directions[tgt_uuid] = True
return False

def _make_port_and_owner(
self, port_obj: m.ModelElement
) -> _elkjs.ELKInputPort:
owner_obj = port_obj.owner
box = self._make_box(
owner_obj,
layout_options=makers.DEFAULT_LABEL_LAYOUT_OPTIONS,
)
if port := self.ports.get(port_obj.uuid):
return port
port = makers.make_port(port_obj.uuid)
if self.diagram._display_port_labels:
text = port_obj.name or "UNKNOWN"
port.labels = makers.make_label(text)
huyenngn marked this conversation as resolved.
Show resolved Hide resolved
_plp = self.diagram._port_label_position
if not (plp := getattr(_elkjs.PORT_LABEL_POSITION, _plp, None)):
raise ValueError(f"Invalid port label position '{_plp}'.")
assert isinstance(plp, _elkjs.PORT_LABEL_POSITION)
box.layoutOptions["portLabels.placement"] = plp.name
box.ports.append(port)
self.ports[port_obj.uuid] = port
return port


def collector(
diagram: context.ContextDiagram, params: dict[str, t.Any]
) -> _elkjs.ELKInputData:
"""Collect data for rendering a custom diagram."""
return CustomCollector(diagram, params)()
Loading
Loading