Module sverchok.utils.tree_structure
Expand source code
# This file is part of project Sverchok. It's copyrighted by the contributors
# recorded in the version control history of the file, available from
# its original location https://github.com/nortikin/sverchok/commit/master
#
# SPDX-License-Identifier: GPL3
# License-Filename: LICENSE
from __future__ import annotations
from collections.abc import Mapping
from typing import List, Iterable, TypeVar, TYPE_CHECKING, Dict, Any, Generic, Optional, Union
import bpy
import sverchok.utils.tree_walk as tw
if TYPE_CHECKING:
from sverchok.core.node_group import SvGroupTree
from sverchok.node_tree import SverchCustomTreeNode
SvNode = Union[SverchCustomTreeNode, bpy.types.Node]
class Node(tw.Node):
def __init__(self, name: str, index: int, tree: Tree, bl_node):
self.name = name
self._inputs: List[Socket] = []
self._outputs: List[Socket] = []
self._index = index
self._tree = tree
self.is_input_changed = False
self.is_updated = False
self.is_output_changed = False
# cash
self.bl_tween = bl_node
# @property
# def bl_tween(self) -> SvNode:
# """Quite expansive function, 1ms = 800 calls, it's better to cash, is potentially dangerous"""
# return self._tree.bl_tween.nodes[self._index]
@property
def index(self):
"""Index of node location in Blender collection from which it was copied"""
return self._index
@property
def inputs(self) -> List[Socket]:
return self._inputs
@property
def outputs(self) -> List[Socket]:
return self._outputs
@property
def next_nodes(self) -> Iterable[Node]:
"""Returns all nodes which are linked wia the node output sockets"""
return {other_s.node for s in self.outputs for other_s in s.linked_sockets}
@property
def last_nodes(self) -> Iterable[Node]:
"""Returns all nodes which are linked wia the node input sockets"""
return {other_s.node for s in self.inputs for other_s in s.linked_sockets}
def get_bl_node(self, tree: SvGroupTree) -> bpy.types.Node:
"""
Will return the node from given tree with the same name
It is slower then `bl_tween` attribute but can return node from another given tree
"""
return tree.nodes[self.name]
def get_input_socket(self, identifier: str, default=None) -> Optional[Socket]:
"""Search input socket by its identifier"""
for socket in self._inputs:
if socket.identifier == identifier:
return socket
return default
def get_output_socket(self, identifier: str, default=None) -> Optional[Socket]:
"""Search output socket by its identifier"""
for socket in self._outputs:
if socket.identifier == identifier:
return socket
return default
@classmethod
def from_bl_node(cls, bl_node: bpy.types.Node, index: int, tree: Tree) -> Node:
"""Generate node and its sockets from Blender node instance"""
node = cls(bl_node.name, index, tree, bl_node)
for in_socket in bl_node.inputs:
node.inputs.append(Socket.from_bl_socket(node, in_socket))
for out_socket in bl_node.outputs:
node.outputs.append(Socket.from_bl_socket(node, out_socket))
return node
def __repr__(self):
return f'Node:"{self.name}"'
NodeType = TypeVar('NodeType', bound=Node)
class Tree(tw.Tree[NodeType]):
"""
Structure similar to blender node groups but is more efficient in searching neighbours
Each time when nodes or links collections are changed the instance of the tree should be recreated
so it is immutable (topologically) data structure
"""
def __init__(self, bl_tree: SvGroupTree):
# it means that the tree has correct topology (during a tree initialization it's always true)
self.is_updated = True
self._tree_id = bl_tree.tree_id
self._nodes = NodesCollection(bl_tree, self)
self._links = LinksCollection(bl_tree, self)
# if the tree is created in the same time with the class initialization (loading file)
# the index of the tree in node_groups collection will be not found (-1)
self._index = bpy.data.node_groups.find(bl_tree.name)
# add links between wifi nodes
self._handle_wifi_nodes()
@property
def id(self) -> str:
return self._tree_id
@property
def bl_tween(self) -> SvGroupTree:
return bpy.data.node_groups[self._index] # todo should keep real tree object instead
@property
def nodes(self) -> NodesCollection:
return self._nodes
@property
def links(self) -> LinksCollection:
return self._links
def _handle_wifi_nodes(self):
"""The idea is to convert wifi nodes into regular nodes with sockets and links between them"""
# todo the code is very bad and should be removed later wifi node refactoring
# the method knows too match about wifi nodes
var_name_wifi_in = dict()
# add sockets to wifi "from nodes" if it has variable
for node in self._nodes:
if node.bl_tween.bl_idname == 'WifiInNode' and node.bl_tween.var_name:
socket = Socket(node, True, "Virtual wifi socket")
node.outputs.append(socket)
var_name_wifi_in[node.bl_tween.var_name] = node
# add sockets to wifi "to nodes" and connect them to wifi "from nodes" if there is wifi node with such variable
if var_name_wifi_in:
for to_node in self._nodes:
if to_node.bl_tween.bl_idname == 'WifiOutNode' and to_node.bl_tween.var_name in var_name_wifi_in:
socket = Socket(to_node, False, "Virtual wifi socket")
to_node.inputs.append(socket)
from_node = var_name_wifi_in[to_node.bl_tween.var_name]
from_socket = from_node.outputs[0]
to_socket = to_node.inputs[0]
self.links._dict[
(from_node.name, from_socket.identifier,
to_node.name, to_socket.identifier)] = Link(from_socket, to_socket, len(self.links))
Element = TypeVar('Element')
class TreeCollections(Mapping, Generic[Element]):
"""
The idea of this collection is to have access to its elements by their identifier
and meantime to have access to their indexes of true blender collections
so to get fast mapping between python and blender collections
downside is that this collection is immutable
because it impossible to predict order of Blender collection after changing of their content
"""
def __init__(self):
self._dict: Dict[Any, Element] = dict()
def __getitem__(self, item) -> Element:
return self._dict[item]
def __iter__(self) -> Iterable[Element]:
return iter(self._dict.values())
def __len__(self):
return len(self._dict)
def __contains__(self, item):
try:
return item.name in self._dict
except AttributeError:
return item in self._dict
def __sub__(self, other) -> List[Element]:
if isinstance(other, TreeCollections):
return [self._dict[k] for k in (self._dict.keys() - other._dict.keys())]
else:
return NotImplemented
class NodesCollection(TreeCollections[NodeType]):
def __init__(self, bl_tree: SvGroupTree, tree: Tree):
super().__init__()
self._active_input: Optional[Node] = None
self._active_output: Optional[Node] = None
for i, bl_node in enumerate(bl_tree.nodes):
node = Node.from_bl_node(bl_node, i, tree)
self._dict[bl_node.name] = node
# https://developer.blender.org/T82350
if bl_node.bl_idname == 'NodeGroupInput':
self._active_input = node
if bl_node.bl_idname == 'NodeGroupOutput':
self._active_output = node
@property
def active_input(self) -> Optional[Node]:
return self._active_input
@property
def active_output(self) -> Optional[Node]:
return self._active_output
class LinksCollection(TreeCollections):
def __init__(self, bl_tree: SvGroupTree, tree: Tree):
super().__init__()
for i, bl_link in enumerate(bl_tree.links):
# new in 2.93, it is the same as if there was no the link (is_hidden was added before 2.93)
if hasattr(bl_link, 'is_muted') and bl_link.is_muted:
# or bl_link.is_hidden: # it does not call update method of a tree https://developer.blender.org/T89109
continue
from_node = tree.nodes[bl_link.from_node.name]
from_socket = from_node.get_output_socket(bl_link.from_socket.identifier)
to_node = tree.nodes[bl_link.to_node.name]
to_socket = to_node.get_input_socket(bl_link.to_socket.identifier)
self._dict[(bl_link.from_node.name, bl_link.from_socket.identifier,
bl_link.to_node.name, bl_link.to_socket.identifier)] = Link(from_socket, to_socket, i)
def __iter__(self) -> Iterable[Link]: return super().__iter__()
class Socket:
def __init__(self, node: Node, is_output: bool, identifier: str):
self.is_output = is_output
self.identifier = identifier
self._node = node
self._links = []
@property
def bl_tween(self) -> bpy.types.NodeSocket:
return getattr(self._node.bl_tween, 'outputs' if self.is_output else 'inputs')[self.index]
@property
def index(self) -> int:
"""Index of the node in inputs or outputs collection"""
return getattr(self.node, 'outputs' if self.is_output else 'inputs').index(self)
@property
def node(self) -> Node:
return self._node
@property
def links(self) -> List[Link]:
return self._links
def get_bl_socket(self, bl_tree: SvGroupTree) -> bpy.types.NodeSocket:
"""Search socket in given tree by its identifier"""
bl_node = self.node.get_bl_node(bl_tree)
for bl_socket in bl_node.outputs if self.is_output else bl_node.inputs:
if bl_socket.identifier == self.identifier:
return bl_socket
raise LookupError(f'Socket "{self.identifier}" was not found in Node "{bl_node.name}"'
f'in {"outputs" if self.is_output else "inputs"} sockets:'
f'"{[s.identefier for s in (bl_node.outputs if self.is_output else bl_node.inputs)]}"')
@property
def linked_sockets(self) -> List[Socket]:
"""All sockets which share the same links"""
return [link.to_socket if self.is_output else link.from_socket for link in self.links]
@classmethod
def from_bl_socket(cls, node: Node, bl_socket: bpy.types.NodeSocket) -> Socket:
"""Generate socket from Blender socket instance"""
return cls(node, bl_socket.is_output, bl_socket.identifier)
def __repr__(self):
return f'Socket "{self.identifier}"'
class Link:
def __init__(self, from_socket: Socket, to_socket: Socket, index: int):
self.from_socket: Socket = from_socket
self.to_socket: Socket = to_socket
from_socket.links.append(self)
to_socket.links.append(self)
self._index = index
@property
def index(self):
"""Index of the link location in Blender collection from which it was copied"""
return self._index
@property
def from_node(self) -> Node:
return self.from_socket.node
@property
def to_node(self) -> Node:
return self.to_socket.node
def __repr__(self):
return f'FROM "{self.from_node.name}.{self.from_socket.identifier}" ' \
f'TO "{self.to_node.name}.{self.to_socket.identifier}"'
Classes
class Link (from_socket: Socket, to_socket: Socket, index: int)
-
Expand source code
class Link: def __init__(self, from_socket: Socket, to_socket: Socket, index: int): self.from_socket: Socket = from_socket self.to_socket: Socket = to_socket from_socket.links.append(self) to_socket.links.append(self) self._index = index @property def index(self): """Index of the link location in Blender collection from which it was copied""" return self._index @property def from_node(self) -> Node: return self.from_socket.node @property def to_node(self) -> Node: return self.to_socket.node def __repr__(self): return f'FROM "{self.from_node.name}.{self.from_socket.identifier}" ' \ f'TO "{self.to_node.name}.{self.to_socket.identifier}"'
Instance variables
var from_node : Node
-
Expand source code
@property def from_node(self) -> Node: return self.from_socket.node
var index
-
Index of the link location in Blender collection from which it was copied
Expand source code
@property def index(self): """Index of the link location in Blender collection from which it was copied""" return self._index
var to_node : Node
-
Expand source code
@property def to_node(self) -> Node: return self.to_socket.node
class LinksCollection (bl_tree: SvGroupTree, tree: Tree)
-
The idea of this collection is to have access to its elements by their identifier and meantime to have access to their indexes of true blender collections so to get fast mapping between python and blender collections downside is that this collection is immutable because it impossible to predict order of Blender collection after changing of their content
Expand source code
class LinksCollection(TreeCollections): def __init__(self, bl_tree: SvGroupTree, tree: Tree): super().__init__() for i, bl_link in enumerate(bl_tree.links): # new in 2.93, it is the same as if there was no the link (is_hidden was added before 2.93) if hasattr(bl_link, 'is_muted') and bl_link.is_muted: # or bl_link.is_hidden: # it does not call update method of a tree https://developer.blender.org/T89109 continue from_node = tree.nodes[bl_link.from_node.name] from_socket = from_node.get_output_socket(bl_link.from_socket.identifier) to_node = tree.nodes[bl_link.to_node.name] to_socket = to_node.get_input_socket(bl_link.to_socket.identifier) self._dict[(bl_link.from_node.name, bl_link.from_socket.identifier, bl_link.to_node.name, bl_link.to_socket.identifier)] = Link(from_socket, to_socket, i) def __iter__(self) -> Iterable[Link]: return super().__iter__()
Ancestors
- TreeCollections
- collections.abc.Mapping
- collections.abc.Collection
- collections.abc.Sized
- collections.abc.Iterable
- collections.abc.Container
- typing.Generic
class Node (name: str, index: int, tree: Tree, bl_node)
-
Helper class that provides a standard way to create an ABC using inheritance.
Expand source code
class Node(tw.Node): def __init__(self, name: str, index: int, tree: Tree, bl_node): self.name = name self._inputs: List[Socket] = [] self._outputs: List[Socket] = [] self._index = index self._tree = tree self.is_input_changed = False self.is_updated = False self.is_output_changed = False # cash self.bl_tween = bl_node # @property # def bl_tween(self) -> SvNode: # """Quite expansive function, 1ms = 800 calls, it's better to cash, is potentially dangerous""" # return self._tree.bl_tween.nodes[self._index] @property def index(self): """Index of node location in Blender collection from which it was copied""" return self._index @property def inputs(self) -> List[Socket]: return self._inputs @property def outputs(self) -> List[Socket]: return self._outputs @property def next_nodes(self) -> Iterable[Node]: """Returns all nodes which are linked wia the node output sockets""" return {other_s.node for s in self.outputs for other_s in s.linked_sockets} @property def last_nodes(self) -> Iterable[Node]: """Returns all nodes which are linked wia the node input sockets""" return {other_s.node for s in self.inputs for other_s in s.linked_sockets} def get_bl_node(self, tree: SvGroupTree) -> bpy.types.Node: """ Will return the node from given tree with the same name It is slower then `bl_tween` attribute but can return node from another given tree """ return tree.nodes[self.name] def get_input_socket(self, identifier: str, default=None) -> Optional[Socket]: """Search input socket by its identifier""" for socket in self._inputs: if socket.identifier == identifier: return socket return default def get_output_socket(self, identifier: str, default=None) -> Optional[Socket]: """Search output socket by its identifier""" for socket in self._outputs: if socket.identifier == identifier: return socket return default @classmethod def from_bl_node(cls, bl_node: bpy.types.Node, index: int, tree: Tree) -> Node: """Generate node and its sockets from Blender node instance""" node = cls(bl_node.name, index, tree, bl_node) for in_socket in bl_node.inputs: node.inputs.append(Socket.from_bl_socket(node, in_socket)) for out_socket in bl_node.outputs: node.outputs.append(Socket.from_bl_socket(node, out_socket)) return node def __repr__(self): return f'Node:"{self.name}"'
Ancestors
- Node
- abc.ABC
Static methods
def from_bl_node(bl_node: bpy.types.Node, index: int, tree: Tree) ‑> Node
-
Generate node and its sockets from Blender node instance
Expand source code
@classmethod def from_bl_node(cls, bl_node: bpy.types.Node, index: int, tree: Tree) -> Node: """Generate node and its sockets from Blender node instance""" node = cls(bl_node.name, index, tree, bl_node) for in_socket in bl_node.inputs: node.inputs.append(Socket.from_bl_socket(node, in_socket)) for out_socket in bl_node.outputs: node.outputs.append(Socket.from_bl_socket(node, out_socket)) return node
Instance variables
var index
-
Index of node location in Blender collection from which it was copied
Expand source code
@property def index(self): """Index of node location in Blender collection from which it was copied""" return self._index
var inputs : List[Socket]
-
Expand source code
@property def inputs(self) -> List[Socket]: return self._inputs
var last_nodes : Iterable[Node]
-
Returns all nodes which are linked wia the node input sockets
Expand source code
@property def last_nodes(self) -> Iterable[Node]: """Returns all nodes which are linked wia the node input sockets""" return {other_s.node for s in self.inputs for other_s in s.linked_sockets}
var next_nodes : Iterable[Node]
-
Returns all nodes which are linked wia the node output sockets
Expand source code
@property def next_nodes(self) -> Iterable[Node]: """Returns all nodes which are linked wia the node output sockets""" return {other_s.node for s in self.outputs for other_s in s.linked_sockets}
var outputs : List[Socket]
-
Expand source code
@property def outputs(self) -> List[Socket]: return self._outputs
Methods
def get_bl_node(self, tree: SvGroupTree) ‑> bpy.types.Node
-
Will return the node from given tree with the same name It is slower then
bl_tween
attribute but can return node from another given treeExpand source code
def get_bl_node(self, tree: SvGroupTree) -> bpy.types.Node: """ Will return the node from given tree with the same name It is slower then `bl_tween` attribute but can return node from another given tree """ return tree.nodes[self.name]
def get_input_socket(self, identifier: str, default=None) ‑> Optional[Socket]
-
Search input socket by its identifier
Expand source code
def get_input_socket(self, identifier: str, default=None) -> Optional[Socket]: """Search input socket by its identifier""" for socket in self._inputs: if socket.identifier == identifier: return socket return default
def get_output_socket(self, identifier: str, default=None) ‑> Optional[Socket]
-
Search output socket by its identifier
Expand source code
def get_output_socket(self, identifier: str, default=None) -> Optional[Socket]: """Search output socket by its identifier""" for socket in self._outputs: if socket.identifier == identifier: return socket return default
Inherited members
class NodesCollection (bl_tree: SvGroupTree, tree: Tree)
-
The idea of this collection is to have access to its elements by their identifier and meantime to have access to their indexes of true blender collections so to get fast mapping between python and blender collections downside is that this collection is immutable because it impossible to predict order of Blender collection after changing of their content
Expand source code
class NodesCollection(TreeCollections[NodeType]): def __init__(self, bl_tree: SvGroupTree, tree: Tree): super().__init__() self._active_input: Optional[Node] = None self._active_output: Optional[Node] = None for i, bl_node in enumerate(bl_tree.nodes): node = Node.from_bl_node(bl_node, i, tree) self._dict[bl_node.name] = node # https://developer.blender.org/T82350 if bl_node.bl_idname == 'NodeGroupInput': self._active_input = node if bl_node.bl_idname == 'NodeGroupOutput': self._active_output = node @property def active_input(self) -> Optional[Node]: return self._active_input @property def active_output(self) -> Optional[Node]: return self._active_output
Ancestors
- TreeCollections
- collections.abc.Mapping
- collections.abc.Collection
- collections.abc.Sized
- collections.abc.Iterable
- collections.abc.Container
- typing.Generic
Instance variables
var active_input : Optional[Node]
-
Expand source code
@property def active_input(self) -> Optional[Node]: return self._active_input
var active_output : Optional[Node]
-
Expand source code
@property def active_output(self) -> Optional[Node]: return self._active_output
class Socket (node: Node, is_output: bool, identifier: str)
-
Expand source code
class Socket: def __init__(self, node: Node, is_output: bool, identifier: str): self.is_output = is_output self.identifier = identifier self._node = node self._links = [] @property def bl_tween(self) -> bpy.types.NodeSocket: return getattr(self._node.bl_tween, 'outputs' if self.is_output else 'inputs')[self.index] @property def index(self) -> int: """Index of the node in inputs or outputs collection""" return getattr(self.node, 'outputs' if self.is_output else 'inputs').index(self) @property def node(self) -> Node: return self._node @property def links(self) -> List[Link]: return self._links def get_bl_socket(self, bl_tree: SvGroupTree) -> bpy.types.NodeSocket: """Search socket in given tree by its identifier""" bl_node = self.node.get_bl_node(bl_tree) for bl_socket in bl_node.outputs if self.is_output else bl_node.inputs: if bl_socket.identifier == self.identifier: return bl_socket raise LookupError(f'Socket "{self.identifier}" was not found in Node "{bl_node.name}"' f'in {"outputs" if self.is_output else "inputs"} sockets:' f'"{[s.identefier for s in (bl_node.outputs if self.is_output else bl_node.inputs)]}"') @property def linked_sockets(self) -> List[Socket]: """All sockets which share the same links""" return [link.to_socket if self.is_output else link.from_socket for link in self.links] @classmethod def from_bl_socket(cls, node: Node, bl_socket: bpy.types.NodeSocket) -> Socket: """Generate socket from Blender socket instance""" return cls(node, bl_socket.is_output, bl_socket.identifier) def __repr__(self): return f'Socket "{self.identifier}"'
Static methods
def from_bl_socket(node: Node, bl_socket: bpy.types.NodeSocket) ‑> Socket
-
Generate socket from Blender socket instance
Expand source code
@classmethod def from_bl_socket(cls, node: Node, bl_socket: bpy.types.NodeSocket) -> Socket: """Generate socket from Blender socket instance""" return cls(node, bl_socket.is_output, bl_socket.identifier)
Instance variables
var bl_tween : bpy_types.NodeSocket
-
Expand source code
@property def bl_tween(self) -> bpy.types.NodeSocket: return getattr(self._node.bl_tween, 'outputs' if self.is_output else 'inputs')[self.index]
var index : int
-
Index of the node in inputs or outputs collection
Expand source code
@property def index(self) -> int: """Index of the node in inputs or outputs collection""" return getattr(self.node, 'outputs' if self.is_output else 'inputs').index(self)
var linked_sockets : List[Socket]
-
All sockets which share the same links
Expand source code
@property def linked_sockets(self) -> List[Socket]: """All sockets which share the same links""" return [link.to_socket if self.is_output else link.from_socket for link in self.links]
var links : List[Link]
-
Expand source code
@property def links(self) -> List[Link]: return self._links
var node : Node
-
Expand source code
@property def node(self) -> Node: return self._node
Methods
def get_bl_socket(self, bl_tree: SvGroupTree) ‑> bpy.types.NodeSocket
-
Search socket in given tree by its identifier
Expand source code
def get_bl_socket(self, bl_tree: SvGroupTree) -> bpy.types.NodeSocket: """Search socket in given tree by its identifier""" bl_node = self.node.get_bl_node(bl_tree) for bl_socket in bl_node.outputs if self.is_output else bl_node.inputs: if bl_socket.identifier == self.identifier: return bl_socket raise LookupError(f'Socket "{self.identifier}" was not found in Node "{bl_node.name}"' f'in {"outputs" if self.is_output else "inputs"} sockets:' f'"{[s.identefier for s in (bl_node.outputs if self.is_output else bl_node.inputs)]}"')
class Tree (bl_tree: SvGroupTree)
-
Structure similar to blender node groups but is more efficient in searching neighbours Each time when nodes or links collections are changed the instance of the tree should be recreated so it is immutable (topologically) data structure
Expand source code
class Tree(tw.Tree[NodeType]): """ Structure similar to blender node groups but is more efficient in searching neighbours Each time when nodes or links collections are changed the instance of the tree should be recreated so it is immutable (topologically) data structure """ def __init__(self, bl_tree: SvGroupTree): # it means that the tree has correct topology (during a tree initialization it's always true) self.is_updated = True self._tree_id = bl_tree.tree_id self._nodes = NodesCollection(bl_tree, self) self._links = LinksCollection(bl_tree, self) # if the tree is created in the same time with the class initialization (loading file) # the index of the tree in node_groups collection will be not found (-1) self._index = bpy.data.node_groups.find(bl_tree.name) # add links between wifi nodes self._handle_wifi_nodes() @property def id(self) -> str: return self._tree_id @property def bl_tween(self) -> SvGroupTree: return bpy.data.node_groups[self._index] # todo should keep real tree object instead @property def nodes(self) -> NodesCollection: return self._nodes @property def links(self) -> LinksCollection: return self._links def _handle_wifi_nodes(self): """The idea is to convert wifi nodes into regular nodes with sockets and links between them""" # todo the code is very bad and should be removed later wifi node refactoring # the method knows too match about wifi nodes var_name_wifi_in = dict() # add sockets to wifi "from nodes" if it has variable for node in self._nodes: if node.bl_tween.bl_idname == 'WifiInNode' and node.bl_tween.var_name: socket = Socket(node, True, "Virtual wifi socket") node.outputs.append(socket) var_name_wifi_in[node.bl_tween.var_name] = node # add sockets to wifi "to nodes" and connect them to wifi "from nodes" if there is wifi node with such variable if var_name_wifi_in: for to_node in self._nodes: if to_node.bl_tween.bl_idname == 'WifiOutNode' and to_node.bl_tween.var_name in var_name_wifi_in: socket = Socket(to_node, False, "Virtual wifi socket") to_node.inputs.append(socket) from_node = var_name_wifi_in[to_node.bl_tween.var_name] from_socket = from_node.outputs[0] to_socket = to_node.inputs[0] self.links._dict[ (from_node.name, from_socket.identifier, to_node.name, to_socket.identifier)] = Link(from_socket, to_socket, len(self.links))
Ancestors
- Tree
- abc.ABC
- typing.Generic
Instance variables
var bl_tween : SvGroupTree
-
Expand source code
@property def bl_tween(self) -> SvGroupTree: return bpy.data.node_groups[self._index] # todo should keep real tree object instead
var id : str
-
Expand source code
@property def id(self) -> str: return self._tree_id
var links : LinksCollection
-
Expand source code
@property def links(self) -> LinksCollection: return self._links
var nodes : NodesCollection
-
Expand source code
@property def nodes(self) -> NodesCollection: return self._nodes
Inherited members
class TreeCollections
-
The idea of this collection is to have access to its elements by their identifier and meantime to have access to their indexes of true blender collections so to get fast mapping between python and blender collections downside is that this collection is immutable because it impossible to predict order of Blender collection after changing of their content
Expand source code
class TreeCollections(Mapping, Generic[Element]): """ The idea of this collection is to have access to its elements by their identifier and meantime to have access to their indexes of true blender collections so to get fast mapping between python and blender collections downside is that this collection is immutable because it impossible to predict order of Blender collection after changing of their content """ def __init__(self): self._dict: Dict[Any, Element] = dict() def __getitem__(self, item) -> Element: return self._dict[item] def __iter__(self) -> Iterable[Element]: return iter(self._dict.values()) def __len__(self): return len(self._dict) def __contains__(self, item): try: return item.name in self._dict except AttributeError: return item in self._dict def __sub__(self, other) -> List[Element]: if isinstance(other, TreeCollections): return [self._dict[k] for k in (self._dict.keys() - other._dict.keys())] else: return NotImplemented
Ancestors
- collections.abc.Mapping
- collections.abc.Collection
- collections.abc.Sized
- collections.abc.Iterable
- collections.abc.Container
- typing.Generic
Subclasses