feat: use basic uuid to identify node inputs

This commit is contained in:
Swann 2020-11-11 17:52:32 +01:00
parent 30d734c2c1
commit cef45dad3c
No known key found for this signature in database
GPG Key ID: E1D3641A7C43AACB
2 changed files with 84 additions and 52 deletions
README.md
multi_user/bl_types

@ -36,6 +36,7 @@ Currently, not all data-block are supported for replication over the wire. The f
| image | ✔️ | | | image | ✔️ | |
| mesh | ✔️ | | | mesh | ✔️ | |
| material | ✔️ | | | material | ✔️ | |
| node_groups | ❗ | Material only |
| metaball | ✔️ | | | metaball | ✔️ | |
| object | ✔️ | | | object | ✔️ | |
| texts | ✔️ | | | texts | ✔️ | |

@ -21,6 +21,8 @@ import mathutils
import logging import logging
import re import re
from uuid import uuid4
from .dump_anything import Loader, Dumper from .dump_anything import Loader, Dumper
from .bl_datablock import BlDatablock, get_datablock_from_uuid from .bl_datablock import BlDatablock, get_datablock_from_uuid
@ -44,25 +46,27 @@ def load_node(node_data, node_tree):
if image_uuid and not target_node.image: if image_uuid and not target_node.image:
target_node.image = get_datablock_from_uuid(image_uuid, None) target_node.image = get_datablock_from_uuid(image_uuid, None)
if node_tree_uuid: if node_tree_uuid:
target_node.node_tree = get_datablock_from_uuid(node_tree_uuid, None) target_node.node_tree = get_datablock_from_uuid(node_tree_uuid, None)
for idx, inpt in enumerate(node_data['inputs']): inputs = node_data.get('inputs')
if hasattr(target_node.inputs[idx], "default_value"): if inputs:
try: for idx, inpt in enumerate(inputs):
target_node.inputs[idx].default_value = inpt["default_value"] if hasattr(target_node.inputs[idx], "default_value"):
except: try:
logging.error( target_node.inputs[idx].default_value = inpt["default_value"]
f"Material {inpt.keys()} parameter not supported, skipping") except:
logging.error(f"Material input {inpt.keys()} parameter not supported, skipping")
for idx, output in enumerate(node_data['outputs']): outputs = node_data.get('outputs')
if hasattr(target_node.outputs[idx], "default_value"): if outputs:
try: for idx, output in enumerate(outputs):
target_node.outputs[idx].default_value = output["default_value"] if hasattr(target_node.outputs[idx], "default_value"):
except: try:
logging.error( target_node.outputs[idx].default_value = output["default_value"]
f"Material {output.keys()} parameter not supported, skipping") except:
logging.error(f"Material output {output.keys()} parameter not supported, skipping")
def load_links(links_data, node_tree): def load_links(links_data, node_tree):
@ -152,9 +156,6 @@ def dump_node(node):
io_dumper.depth = 2 io_dumper.depth = 2
io_dumper.include_filter = ["default_value"] io_dumper.include_filter = ["default_value"]
if node.type in ['GROUP_INPUT', 'GROUP_OUTPUT']:
io_dumper.include_filter.extend(['name','type'])
for idx, inpt in enumerate(node.inputs): for idx, inpt in enumerate(node.inputs):
if hasattr(inpt, 'default_value'): if hasattr(inpt, 'default_value'):
dumped_node['inputs'].append(io_dumper.dump(inpt)) dumped_node['inputs'].append(io_dumper.dump(inpt))
@ -190,7 +191,7 @@ def dump_node(node):
return dumped_node return dumped_node
def dump_shader_node_tree(node_tree:bpy.types.ShaderNodeTree)->dict: def dump_shader_node_tree(node_tree: bpy.types.ShaderNodeTree) -> dict:
""" Dump a shader node_tree to a dict including links and nodes """ Dump a shader node_tree to a dict including links and nodes
:arg node_tree: dumped shader node tree :arg node_tree: dumped shader node tree
@ -198,26 +199,74 @@ def dump_shader_node_tree(node_tree:bpy.types.ShaderNodeTree)->dict:
:return: dict :return: dict
""" """
node_tree_data = { node_tree_data = {
'nodes': {node.name: dump_node(node) for node in node_tree.nodes}, 'nodes': {node.name: dump_node(node) for node in node_tree.nodes},
'links': dump_links(node_tree.links), 'links': dump_links(node_tree.links),
'name': node_tree.name, 'name': node_tree.name,
'type': type(node_tree).__name__ 'type': type(node_tree).__name__
} }
if node_tree.inputs: for socket_id in ['inputs', 'outputs']:
node_tree_data['inputs'] = [(i.name, i.bl_socket_idname, i.identifier) for i in node_tree.inputs] socket_collection = getattr(node_tree, socket_id)
if node_tree.outputs: node_tree_data[socket_id] = dump_node_tree_sockets(socket_collection)
node_tree_data['outputs'] = [(o.name, o.bl_socket_idname)for o in node_tree.outputs]
return node_tree_data return node_tree_data
def load_shader_node_tree(node_tree_data:dict, target_node_tree:bpy.types.ShaderNodeTree)->dict:
"""
:arg node_tree: dumped node data def dump_node_tree_sockets(sockets: bpy.types.Collection)->dict:
:type node_data: dict """ dump sockets of a shader_node_tree
:arg node_tree: target node_tree
:type node_tree: bpy.types.NodeTree :arg target_node_tree: target node_tree
:type target_node_tree: bpy.types.NodeTree
:arg socket_id: socket identifer
:type socket_id: str
:return: dict
"""
sockets_data = []
for socket in sockets:
try:
socket_uuid = socket['uuid']
except Exception:
socket_uuid = str(uuid4())
socket['uuid'] = socket_uuid
sockets_data.append((socket.name, socket.bl_socket_idname, socket_uuid))
return sockets_data
def load_node_tree_sockets(sockets: bpy.types.Collection,
sockets_data: dict):
""" load sockets of a shader_node_tree
:arg target_node_tree: target node_tree
:type target_node_tree: bpy.types.NodeTree
:arg socket_id: socket identifer
:type socket_id: str
:arg socket_data: dumped socket data
:type socket_data: dict
"""
# Check for removed sockets
for socket in sockets:
if not [s for s in sockets_data if socket['uuid'] == s[2]]:
sockets.remove(socket)
# Check for new sockets
for idx, socket_data in enumerate(sockets_data):
try:
checked_socket = sockets[idx]
if checked_socket.name != socket_data[0]:
checked_socket.name = socket_data[0]
except Exception:
s = sockets.new(socket_data[1], socket_data[0])
s['uuid'] = socket_data[2]
def load_shader_node_tree(node_tree_data:dict, target_node_tree:bpy.types.ShaderNodeTree)->dict:
"""Load a shader node_tree from dumped data
:arg node_tree_data: dumped node data
:type node_tree_data: dict
:arg target_node_tree: target node_tree
:type target_node_tree: bpy.types.NodeTree
""" """
# TODO: load only required nodes # TODO: load only required nodes
target_node_tree.nodes.clear() target_node_tree.nodes.clear()
@ -226,30 +275,12 @@ def load_shader_node_tree(node_tree_data:dict, target_node_tree:bpy.types.Shader
target_node_tree.name = node_tree_data['name'] target_node_tree.name = node_tree_data['name']
if 'inputs' in node_tree_data: if 'inputs' in node_tree_data:
for inpt in target_node_tree.inputs: socket_collection = getattr(target_node_tree, 'inputs')
if not [i for i in node_tree_data['inputs'] if inpt.identifier == i[2]]: load_node_tree_sockets(socket_collection, node_tree_data['inputs'])
target_node_tree.inputs.remove(inpt)
for idx, socket_data in enumerate(node_tree_data['inputs']):
try:
checked_input = target_node_tree.inputs[idx]
if checked_input.name != socket_data[0]:
checked_input.name = socket_data[0]
except Exception:
target_node_tree.inputs.new(socket_data[1], socket_data[0])
if 'outputs' in node_tree_data: if 'outputs' in node_tree_data:
for inpt in target_node_tree.outputs: socket_collection = getattr(target_node_tree, 'outputs')
if not [o for o in node_tree_data['outputs'] if inpt.identifier == o[2]]: load_node_tree_sockets(socket_collection,node_tree_data['outputs'])
target_node_tree.outputs.remove(inpt)
for idx, socket_data in enumerate(node_tree_data['outputs']):
try:
checked_outputs = target_node_tree.outputs[idx]
if checked_outputs.name != socket_data[0]:
checked_outputs.name = socket_data[0]
except Exception:
target_node_tree.outputs.new(socket_data[1], socket_data[0])
# Load nodes # Load nodes
for node in node_tree_data["nodes"]: for node in node_tree_data["nodes"]: