Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 95f4d051d6 | |||
| c79ad8e87d | |||
| 9797dfbfc0 | |||
| 63b00a6257 | |||
| 36091845a6 | |||
| 89e1855e2c | |||
| b7f5725282 | |||
| dc8c8cedf4 | |||
| 31e8b838dd | |||
| baf7f32c2a | |||
| ad1d58bd4c | |||
| ec86688750 | |||
| 84098f4c42 | |||
| 77f9d73698 | |||
| 812e8dd2f3 |
@@ -7,7 +7,7 @@
|
|||||||
</h3>
|
</h3>
|
||||||
<p align="center"><b>Speckle</b> is the data infrastructure for the AEC industry.</p><br/>
|
<p align="center"><b>Speckle</b> is the data infrastructure for the AEC industry.</p><br/>
|
||||||
|
|
||||||
<p align="center"><a href="https://twitter.com/SpeckleSystems"><img src="https://img.shields.io/twitter/follow/SpeckleSystems?style=social" alt="Twitter Follow"></a> <a href="https://speckle.community"><img src="https://img.shields.io/discourse/users?server=https%3A%2F%2Fspeckle.community&style=flat-square&logo=discourse&logoColor=white" alt="Community forum users"></a> <a href="https://speckle.systems"><img src="https://img.shields.io/badge/https://-speckle.systems-royalblue?style=flat-square" alt="website"></a> <a href="https://speckle.guide/dev/"><img src="https://img.shields.io/badge/docs-speckle.guide-orange?style=flat-square&logo=read-the-docs&logoColor=white" alt="docs"></a></p>
|
<p align="center"><a href="https://twitter.com/SpeckleSystems"><img src="https://img.shields.io/twitter/follow/SpeckleSystems?style=social" alt="Twitter Follow"></a> <a href="https://speckle.community"><img src="https://img.shields.io/discourse/users?server=https%3A%2F%2Fspeckle.community&style=flat-square&logo=discourse&logoColor=white" alt="Community forum users"></a> <a href="https://speckle.systems"><img src="https://img.shields.io/badge/https://-speckle.systems-royalblue?style=flat-square" alt="website"></a> <a href="https://docs.speckle.systems/dev/"><img src="https://img.shields.io/badge/docs-docs.speckle.systems-orange?style=flat-square&logo=read-the-docs&logoColor=white" alt="docs"></a></p>
|
||||||
<p align="center"><a href="https://github.com/specklesystems/speckle-blender/"><img src="https://circleci.com/gh/specklesystems/speckle-blender.svg?style=svg&circle-token=76eabd350ea243575cbb258b746ed3f471f7ac29" alt="Speckle-Next"></a> </p>
|
<p align="center"><a href="https://github.com/specklesystems/speckle-blender/"><img src="https://circleci.com/gh/specklesystems/speckle-blender.svg?style=svg&circle-token=76eabd350ea243575cbb258b746ed3f471f7ac29" alt="Speckle-Next"></a> </p>
|
||||||
|
|
||||||
# About Speckle
|
# About Speckle
|
||||||
@@ -25,20 +25,19 @@ What is Speckle? Check our ](https://speckle.xyz) ⇒ creating an account at our public server
|
- [](https://app.speckle.systems) ⇒ creating an account at our public server
|
||||||
- [](https://marketplace.digitalocean.com/apps/speckle-server?refcode=947a2b5d7dc1) ⇒ deploying an instance in 1 click
|
|
||||||
|
|
||||||
### Resources
|
### Resources
|
||||||
|
|
||||||
- [](https://speckle.community) for help, feature requests or just to hang with other speckle enthusiasts, check out our community forum!
|
- [](https://speckle.community) for help, feature requests or just to hang with other speckle enthusiasts, check out our community forum!
|
||||||
- [](https://speckle.systems) our tutorials portal is full of resources to get you started using Speckle
|
- [](https://speckle.systems) our tutorials portal is full of resources to get you started using Speckle
|
||||||
- [](https://speckle.guide/user/blender.html) reference on almost any end-user and developer functionality
|
- [](https://docs.speckle.systems/connectors/blender) reference on almost any end-user and developer functionality
|
||||||
|
|
||||||
|
|
||||||
# Blender Connector
|
# Blender Connector
|
||||||
|
|||||||
@@ -10,7 +10,11 @@ from specklepy.core.api import host_applications
|
|||||||
|
|
||||||
from ..utils.get_ascendants import get_ascendants
|
from ..utils.get_ascendants import get_ascendants
|
||||||
from ..utils.account_manager import _client_cache
|
from ..utils.account_manager import _client_cache
|
||||||
from ...converter.utils import find_object_by_id, get_project_workspace_id
|
from ...converter.utils import (
|
||||||
|
find_object_by_id,
|
||||||
|
get_project_workspace_id,
|
||||||
|
build_object_id_map,
|
||||||
|
)
|
||||||
from ...converter.to_native import (
|
from ...converter.to_native import (
|
||||||
convert_to_native,
|
convert_to_native,
|
||||||
render_material_proxy_to_native,
|
render_material_proxy_to_native,
|
||||||
@@ -78,11 +82,17 @@ def load_operation(
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Build object ID map once
|
||||||
|
object_id_map = build_object_id_map(version_data)
|
||||||
|
|
||||||
# Create material mapping first
|
# Create material mapping first
|
||||||
material_mapping = render_material_proxy_to_native(version_data)
|
material_mapping = render_material_proxy_to_native(version_data)
|
||||||
|
|
||||||
definition_collections, definition_objects = instance_definition_proxy_to_native(
|
definition_collections, definition_objects = instance_definition_proxy_to_native(
|
||||||
version_data, material_mapping, instance_loading_mode=instance_loading_mode
|
version_data,
|
||||||
|
material_mapping,
|
||||||
|
instance_loading_mode=instance_loading_mode,
|
||||||
|
object_id_map=object_id_map,
|
||||||
)
|
)
|
||||||
|
|
||||||
definitions_root_collection = None
|
definitions_root_collection = None
|
||||||
@@ -96,7 +106,8 @@ def load_operation(
|
|||||||
for definition in find_instance_definitions(version_data).values():
|
for definition in find_instance_definitions(version_data).values():
|
||||||
definition_object_ids.update(definition.objects)
|
definition_object_ids.update(definition.objects)
|
||||||
for obj_id in definition.objects:
|
for obj_id in definition.objects:
|
||||||
found_obj = find_object_by_id(version_data, obj_id)
|
# Use ID map
|
||||||
|
found_obj = object_id_map.get(obj_id)
|
||||||
if found_obj:
|
if found_obj:
|
||||||
if hasattr(found_obj, "id"):
|
if hasattr(found_obj, "id"):
|
||||||
definition_object_ids.add(found_obj.id)
|
definition_object_ids.add(found_obj.id)
|
||||||
|
|||||||
@@ -123,7 +123,11 @@ def update_workspaces_list(context: Context) -> None:
|
|||||||
workspace: speckle_workspace = wm.speckle_workspaces.add()
|
workspace: speckle_workspace = wm.speckle_workspaces.add()
|
||||||
workspace.id = id
|
workspace.id = id
|
||||||
workspace.name = name
|
workspace.name = name
|
||||||
wm.selected_workspace.id = get_active_workspace(wm.selected_account_id)["id"]
|
active_workspace = get_active_workspace(wm.selected_account_id)
|
||||||
|
if active_workspace:
|
||||||
|
wm.selected_workspace.id = active_workspace["id"]
|
||||||
|
else:
|
||||||
|
wm.selected_workspace.id = "personal"
|
||||||
print("Updated Workspaces List!")
|
print("Updated Workspaces List!")
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -120,10 +120,13 @@ class SPECKLE_OT_project_selection_dialog(bpy.types.Operator):
|
|||||||
if wm.selected_account_id == "":
|
if wm.selected_account_id == "":
|
||||||
wm.selected_account_id = get_default_account_id()
|
wm.selected_account_id = get_default_account_id()
|
||||||
|
|
||||||
wm.selected_workspace.id = get_active_workspace(wm.selected_account_id)["id"]
|
active_workspace = get_active_workspace(wm.selected_account_id)
|
||||||
wm.selected_workspace.name = get_active_workspace(wm.selected_account_id)[
|
if active_workspace:
|
||||||
"name"
|
wm.selected_workspace.id = active_workspace["id"]
|
||||||
]
|
wm.selected_workspace.name = active_workspace["name"]
|
||||||
|
else:
|
||||||
|
wm.selected_workspace.id = "personal"
|
||||||
|
wm.selected_workspace.name = "Personal Projects"
|
||||||
|
|
||||||
# Fetch projects from server
|
# Fetch projects from server
|
||||||
projects: List[Tuple[str, str, str, str, bool]] = get_projects_for_account(
|
projects: List[Tuple[str, str, str, str, bool]] = get_projects_for_account(
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
from typing import Any, Dict, Optional
|
||||||
|
|
||||||
import bpy
|
import bpy
|
||||||
from bpy.types import Context
|
from bpy.types import Context
|
||||||
from typing import Dict, Any, Optional
|
|
||||||
from ..utils.property_groups import speckle_model_card
|
from ..utils.property_groups import speckle_model_card
|
||||||
|
|
||||||
|
|
||||||
@@ -316,11 +318,17 @@ def update_model_card_objects(
|
|||||||
if isinstance(converted_objects, list):
|
if isinstance(converted_objects, list):
|
||||||
converted_objects = {obj.name: obj for obj in converted_objects}
|
converted_objects = {obj.name: obj for obj in converted_objects}
|
||||||
|
|
||||||
|
# Using a set keeps lookup O(1)
|
||||||
|
object_names = set()
|
||||||
|
collection_names = set()
|
||||||
|
|
||||||
for obj in converted_objects.values():
|
for obj in converted_objects.values():
|
||||||
# Handle collections
|
# Handle collections
|
||||||
if isinstance(obj, bpy.types.Collection):
|
if isinstance(obj, bpy.types.Collection):
|
||||||
if obj.name in (o.name for o in model_card.collections):
|
if obj.name in collection_names:
|
||||||
continue
|
continue
|
||||||
|
collection_names.add(obj.name)
|
||||||
|
|
||||||
s_col = model_card.collections.add()
|
s_col = model_card.collections.add()
|
||||||
s_col.name = obj.name
|
s_col.name = obj.name
|
||||||
s_col.applicationId = obj.get("applicationId", "")
|
s_col.applicationId = obj.get("applicationId", "")
|
||||||
@@ -334,8 +342,10 @@ def update_model_card_objects(
|
|||||||
|
|
||||||
# Handle objects
|
# Handle objects
|
||||||
elif isinstance(obj, bpy.types.Object):
|
elif isinstance(obj, bpy.types.Object):
|
||||||
if obj.name in (o.name for o in model_card.objects):
|
if obj.name in object_names:
|
||||||
continue
|
continue
|
||||||
|
object_names.add(obj.name)
|
||||||
|
|
||||||
s_obj = model_card.objects.add()
|
s_obj = model_card.objects.add()
|
||||||
s_obj.name = obj.name
|
s_obj.name = obj.name
|
||||||
s_obj.applicationId = obj.get("applicationId", "")
|
s_obj.applicationId = obj.get("applicationId", "")
|
||||||
|
|||||||
@@ -159,7 +159,14 @@ def convert_to_native(
|
|||||||
else:
|
else:
|
||||||
# Fallback to display value if direct conversion not supported
|
# Fallback to display value if direct conversion not supported
|
||||||
mesh, children = display_value_to_native(
|
mesh, children = display_value_to_native(
|
||||||
speckle_object, object_name, data_block_name, scale, material_mapping
|
speckle_object,
|
||||||
|
object_name,
|
||||||
|
data_block_name,
|
||||||
|
scale,
|
||||||
|
material_mapping,
|
||||||
|
definition_collections,
|
||||||
|
root_collection,
|
||||||
|
instance_loading_mode,
|
||||||
)
|
)
|
||||||
if mesh:
|
if mesh:
|
||||||
# Create a mesh object with the object_name (simple name) and mesh data
|
# Create a mesh object with the object_name (simple name) and mesh data
|
||||||
@@ -176,7 +183,11 @@ def convert_to_native(
|
|||||||
# Ensure the converted object has the correct name (especially for DataObjects)
|
# Ensure the converted object has the correct name (especially for DataObjects)
|
||||||
if isinstance(speckle_object, DataObject):
|
if isinstance(speckle_object, DataObject):
|
||||||
converted_object.name = object_name
|
converted_object.name = object_name
|
||||||
data_block_name = converted_object.data.name
|
if (
|
||||||
|
hasattr(converted_object, "data")
|
||||||
|
and converted_object.data is not None
|
||||||
|
):
|
||||||
|
data_block_name = converted_object.data.name
|
||||||
|
|
||||||
# If there are multiple objects, parent remaining ones to the first
|
# If there are multiple objects, parent remaining ones to the first
|
||||||
for child in children[1:]:
|
for child in children[1:]:
|
||||||
@@ -197,6 +208,9 @@ def display_value_to_native(
|
|||||||
data_block_name: str,
|
data_block_name: str,
|
||||||
scale: float,
|
scale: float,
|
||||||
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
||||||
|
definition_collections: Optional[Dict[str, bpy.types.Collection]] = None,
|
||||||
|
root_collection: Optional[bpy.types.Collection] = None,
|
||||||
|
instance_loading_mode: str = "INSTANCE_PROXIES",
|
||||||
) -> Tuple[Optional[bpy.types.Mesh], List[Object]]:
|
) -> Tuple[Optional[bpy.types.Mesh], List[Object]]:
|
||||||
"""
|
"""
|
||||||
fallback conversion mechanism using displayValue if present
|
fallback conversion mechanism using displayValue if present
|
||||||
@@ -215,6 +229,9 @@ def display_value_to_native(
|
|||||||
DISPLAY_VALUE_PROPERTY_ALIASES,
|
DISPLAY_VALUE_PROPERTY_ALIASES,
|
||||||
True,
|
True,
|
||||||
material_mapping,
|
material_mapping,
|
||||||
|
definition_collections,
|
||||||
|
root_collection,
|
||||||
|
instance_loading_mode,
|
||||||
)
|
)
|
||||||
|
|
||||||
# If the parent had an applicationId and we created a mesh, apply the material
|
# If the parent had an applicationId and we created a mesh, apply the material
|
||||||
@@ -247,6 +264,9 @@ def elements_to_native(
|
|||||||
data_block_name: str,
|
data_block_name: str,
|
||||||
scale: float,
|
scale: float,
|
||||||
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
||||||
|
definition_collections: Optional[Dict[str, bpy.types.Collection]] = None,
|
||||||
|
root_collection: Optional[bpy.types.Collection] = None,
|
||||||
|
instance_loading_mode: str = "INSTANCE_PROXIES",
|
||||||
) -> List[Object]:
|
) -> List[Object]:
|
||||||
"""
|
"""
|
||||||
convert elements collection of a speckle object
|
convert elements collection of a speckle object
|
||||||
@@ -259,6 +279,9 @@ def elements_to_native(
|
|||||||
ELEMENTS_PROPERTY_ALIASES,
|
ELEMENTS_PROPERTY_ALIASES,
|
||||||
False,
|
False,
|
||||||
material_mapping,
|
material_mapping,
|
||||||
|
definition_collections,
|
||||||
|
root_collection,
|
||||||
|
instance_loading_mode,
|
||||||
)
|
)
|
||||||
return elements
|
return elements
|
||||||
|
|
||||||
@@ -271,12 +294,16 @@ def _members_to_native(
|
|||||||
members: Iterable[str],
|
members: Iterable[str],
|
||||||
combineMeshes: bool,
|
combineMeshes: bool,
|
||||||
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
material_mapping: Optional[Dict[str, bpy.types.Material]] = None,
|
||||||
|
definition_collections: Optional[Dict[str, bpy.types.Collection]] = None,
|
||||||
|
root_collection: Optional[bpy.types.Collection] = None,
|
||||||
|
instance_loading_mode: str = "INSTANCE_PROXIES",
|
||||||
) -> Tuple[Optional[bpy.types.Mesh], List[Object]]:
|
) -> Tuple[Optional[bpy.types.Mesh], List[Object]]:
|
||||||
"""
|
"""
|
||||||
converts a given speckle_object by converting specified members
|
converts a given speckle_object by converting specified members
|
||||||
"""
|
"""
|
||||||
meshes: List[Mesh] = []
|
meshes: List[Mesh] = []
|
||||||
others: List[Base] = []
|
others: List[Base] = []
|
||||||
|
instance_proxies: List[InstanceProxy] = []
|
||||||
|
|
||||||
for alias in members:
|
for alias in members:
|
||||||
display = getattr(speckle_object, alias, None)
|
display = getattr(speckle_object, alias, None)
|
||||||
@@ -285,10 +312,13 @@ def _members_to_native(
|
|||||||
MAX_DEPTH = 255 # some large value, to prevent infinite recursion
|
MAX_DEPTH = 255 # some large value, to prevent infinite recursion
|
||||||
|
|
||||||
def separate(value: Any) -> bool:
|
def separate(value: Any) -> bool:
|
||||||
nonlocal meshes, others, count, MAX_DEPTH
|
nonlocal meshes, others, instance_proxies, count, MAX_DEPTH
|
||||||
|
|
||||||
if combineMeshes and isinstance(value, Mesh):
|
if combineMeshes and isinstance(value, Mesh):
|
||||||
meshes.append(value)
|
meshes.append(value)
|
||||||
|
elif isinstance(value, InstanceProxy):
|
||||||
|
# Handle InstanceProxy objects separately - they need definition_collections
|
||||||
|
instance_proxies.append(value)
|
||||||
elif isinstance(value, Base):
|
elif isinstance(value, Base):
|
||||||
others.append(value)
|
others.append(value)
|
||||||
elif isinstance(value, list):
|
elif isinstance(value, list):
|
||||||
@@ -318,10 +348,28 @@ def _members_to_native(
|
|||||||
# Check if the original object is a DataObject
|
# Check if the original object is a DataObject
|
||||||
is_data_object = isinstance(speckle_object, DataObject)
|
is_data_object = isinstance(speckle_object, DataObject)
|
||||||
|
|
||||||
|
# Process InstanceProxy objects - do not add to children list as they are already
|
||||||
|
for item in instance_proxies:
|
||||||
|
try:
|
||||||
|
convert_to_native(
|
||||||
|
item,
|
||||||
|
material_mapping,
|
||||||
|
definition_collections=definition_collections,
|
||||||
|
root_collection=root_collection,
|
||||||
|
instance_loading_mode="LINKED_DUPLICATES", # always use Linked Duplicates for displayValue proxies
|
||||||
|
)
|
||||||
|
except Exception as ex:
|
||||||
|
print(f"Failed to convert instance proxy in display value {item}: {ex}")
|
||||||
|
|
||||||
|
# Process other objects
|
||||||
for item in others:
|
for item in others:
|
||||||
try:
|
try:
|
||||||
blender_object = convert_to_native(
|
blender_object = convert_to_native(
|
||||||
item, material_mapping, instance_loading_mode="INSTANCE_PROXIES"
|
item,
|
||||||
|
material_mapping,
|
||||||
|
definition_collections=definition_collections,
|
||||||
|
root_collection=root_collection,
|
||||||
|
instance_loading_mode=instance_loading_mode,
|
||||||
)
|
)
|
||||||
if blender_object:
|
if blender_object:
|
||||||
# If the parent is a DataObject, override the name of the converted child
|
# If the parent is a DataObject, override the name of the converted child
|
||||||
@@ -987,7 +1035,14 @@ def curve_to_native(
|
|||||||
):
|
):
|
||||||
print("curve_to_native: degree 2 curve, falling back to displayValue")
|
print("curve_to_native: degree 2 curve, falling back to displayValue")
|
||||||
mesh, children = display_value_to_native(
|
mesh, children = display_value_to_native(
|
||||||
speckle_curve, object_name, data_block_name, scale
|
speckle_curve,
|
||||||
|
object_name,
|
||||||
|
data_block_name,
|
||||||
|
scale,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
"INSTANCE_PROXIES",
|
||||||
)
|
)
|
||||||
if mesh:
|
if mesh:
|
||||||
curve_obj = bpy.data.objects.new(object_name, mesh)
|
curve_obj = bpy.data.objects.new(object_name, mesh)
|
||||||
@@ -1059,7 +1114,14 @@ def polycurve_to_native(
|
|||||||
and speckle_polycurve.displayValue
|
and speckle_polycurve.displayValue
|
||||||
):
|
):
|
||||||
mesh, children = display_value_to_native(
|
mesh, children = display_value_to_native(
|
||||||
speckle_polycurve, object_name, data_block_name, scale
|
speckle_polycurve,
|
||||||
|
object_name,
|
||||||
|
data_block_name,
|
||||||
|
scale,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
"INSTANCE_PROXIES",
|
||||||
)
|
)
|
||||||
if mesh:
|
if mesh:
|
||||||
curve_obj = bpy.data.objects.new(object_name, mesh)
|
curve_obj = bpy.data.objects.new(object_name, mesh)
|
||||||
@@ -1211,6 +1273,7 @@ def instance_definition_proxy_to_native(
|
|||||||
material_mapping: Dict[str, Any],
|
material_mapping: Dict[str, Any],
|
||||||
processed_definitions: Dict[str, Any] = None,
|
processed_definitions: Dict[str, Any] = None,
|
||||||
instance_loading_mode: str = "INSTANCE_PROXIES",
|
instance_loading_mode: str = "INSTANCE_PROXIES",
|
||||||
|
object_id_map: Optional[Dict[str, Base]] = None,
|
||||||
) -> Tuple[Dict[str, bpy.types.Collection], Dict[str, Any]]:
|
) -> Tuple[Dict[str, bpy.types.Collection], Dict[str, Any]]:
|
||||||
"""
|
"""
|
||||||
converts instance definition proxies to Blender collections recursively
|
converts instance definition proxies to Blender collections recursively
|
||||||
@@ -1262,7 +1325,8 @@ def instance_definition_proxy_to_native(
|
|||||||
# Process objects, including nested instances
|
# Process objects, including nested instances
|
||||||
if hasattr(definition, "objects") and isinstance(definition.objects, list):
|
if hasattr(definition, "objects") and isinstance(definition.objects, list):
|
||||||
for obj_id in definition.objects:
|
for obj_id in definition.objects:
|
||||||
found_obj = find_object_by_id(root_object, obj_id)
|
# Use the ID map for lookup
|
||||||
|
found_obj = object_id_map.get(obj_id) if object_id_map else None
|
||||||
|
|
||||||
if found_obj:
|
if found_obj:
|
||||||
try:
|
try:
|
||||||
@@ -1362,7 +1426,8 @@ def instance_proxy_to_linked_duplicates(
|
|||||||
print(f"Definition collection not found for instance {speckle_instance.id}")
|
print(f"Definition collection not found for instance {speckle_instance.id}")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
unit_scale = proxy_scale(speckle_instance)
|
# Use the scale from the parent context
|
||||||
|
unit_scale = scale
|
||||||
|
|
||||||
# convert transformation matrix
|
# convert transformation matrix
|
||||||
matrix = mathutils.Matrix(
|
matrix = mathutils.Matrix(
|
||||||
@@ -1397,7 +1462,6 @@ def instance_proxy_to_linked_duplicates(
|
|||||||
location, rotation, scale_vector = matrix.decompose()
|
location, rotation, scale_vector = matrix.decompose()
|
||||||
location = location * unit_scale
|
location = location * unit_scale
|
||||||
|
|
||||||
# create transformation matrix
|
|
||||||
final_matrix = (
|
final_matrix = (
|
||||||
mathutils.Matrix.Translation(location)
|
mathutils.Matrix.Translation(location)
|
||||||
@ rotation.to_matrix().to_4x4()
|
@ rotation.to_matrix().to_4x4()
|
||||||
@@ -1409,10 +1473,8 @@ def instance_proxy_to_linked_duplicates(
|
|||||||
parent_empty.empty_display_type = "PLAIN_AXES"
|
parent_empty.empty_display_type = "PLAIN_AXES"
|
||||||
parent_empty.empty_display_size = 0.1
|
parent_empty.empty_display_size = 0.1
|
||||||
|
|
||||||
parent_empty.matrix_world = final_matrix
|
|
||||||
|
|
||||||
# link parent to root collection
|
|
||||||
root_collection.objects.link(parent_empty)
|
root_collection.objects.link(parent_empty)
|
||||||
|
parent_empty.matrix_world = final_matrix
|
||||||
|
|
||||||
parent_empty["speckle_id"] = speckle_instance.id
|
parent_empty["speckle_id"] = speckle_instance.id
|
||||||
parent_empty["speckle_type"] = speckle_instance.speckle_type
|
parent_empty["speckle_type"] = speckle_instance.speckle_type
|
||||||
@@ -1422,15 +1484,14 @@ def instance_proxy_to_linked_duplicates(
|
|||||||
|
|
||||||
duplicated_objects = []
|
duplicated_objects = []
|
||||||
for obj in definition_collection.objects:
|
for obj in definition_collection.objects:
|
||||||
# create a copy of the object with linked data
|
|
||||||
duplicate_obj = obj.copy()
|
duplicate_obj = obj.copy()
|
||||||
|
|
||||||
duplicate_obj.name = f"{obj.name}_{speckle_instance.id[:8]}"
|
duplicate_obj.name = f"{obj.name}_{speckle_instance.id[:8]}"
|
||||||
|
|
||||||
root_collection.objects.link(duplicate_obj)
|
root_collection.objects.link(duplicate_obj)
|
||||||
|
|
||||||
# apply the instance transformation directly to each object
|
duplicate_obj.parent = parent_empty
|
||||||
duplicate_obj.matrix_world = final_matrix @ obj.matrix_world
|
duplicate_obj.matrix_parent_inverse.identity()
|
||||||
|
duplicate_obj.matrix_basis = obj.matrix_world
|
||||||
|
|
||||||
duplicated_objects.append(duplicate_obj)
|
duplicated_objects.append(duplicate_obj)
|
||||||
|
|
||||||
@@ -1450,7 +1511,8 @@ def instance_proxy_to_native(
|
|||||||
print(f"Definition collection not found for instance {speckle_instance.id}")
|
print(f"Definition collection not found for instance {speckle_instance.id}")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
unit_scale = proxy_scale(speckle_instance)
|
# Use the scale from the parent context
|
||||||
|
unit_scale = scale
|
||||||
|
|
||||||
# convert transformation matrix
|
# convert transformation matrix
|
||||||
matrix = mathutils.Matrix(
|
matrix = mathutils.Matrix(
|
||||||
@@ -1483,35 +1545,24 @@ def instance_proxy_to_native(
|
|||||||
)
|
)
|
||||||
|
|
||||||
location, rotation, scale_vector = matrix.decompose()
|
location, rotation, scale_vector = matrix.decompose()
|
||||||
|
|
||||||
location = location * unit_scale
|
location = location * unit_scale
|
||||||
|
instance_name = f"Instance_{speckle_instance.id}"
|
||||||
bpy.ops.object.collection_instance_add(
|
instance_obj = bpy.data.objects.new(instance_name, None)
|
||||||
collection=definition_collection.name,
|
instance_obj.instance_type = "COLLECTION"
|
||||||
align="WORLD",
|
instance_obj.instance_collection = definition_collection
|
||||||
location=(0, 0, 0),
|
|
||||||
rotation=(0, 0, 0),
|
|
||||||
scale=(1, 1, 1),
|
|
||||||
)
|
|
||||||
|
|
||||||
instance_obj = bpy.context.active_object
|
|
||||||
|
|
||||||
instance_obj.empty_display_size = 0
|
instance_obj.empty_display_size = 0
|
||||||
|
|
||||||
instance_name = f"Instance_{speckle_instance.id}"
|
# Link to root collection
|
||||||
instance_obj.name = instance_name
|
root_collection.objects.link(instance_obj)
|
||||||
|
|
||||||
if instance_obj.name not in root_collection.objects:
|
|
||||||
for coll in instance_obj.users_collection:
|
|
||||||
coll.objects.unlink(instance_obj)
|
|
||||||
root_collection.objects.link(instance_obj)
|
|
||||||
|
|
||||||
|
# Store metadata
|
||||||
instance_obj["speckle_id"] = speckle_instance.id
|
instance_obj["speckle_id"] = speckle_instance.id
|
||||||
instance_obj["speckle_type"] = speckle_instance.speckle_type
|
instance_obj["speckle_type"] = speckle_instance.speckle_type
|
||||||
instance_obj["definition_id"] = speckle_instance.definitionId
|
instance_obj["definition_id"] = speckle_instance.definitionId
|
||||||
if hasattr(speckle_instance, "maxDepth"):
|
if hasattr(speckle_instance, "maxDepth"):
|
||||||
instance_obj["max_depth"] = speckle_instance.maxDepth
|
instance_obj["max_depth"] = speckle_instance.maxDepth
|
||||||
|
|
||||||
|
# Apply transformation
|
||||||
final_matrix = (
|
final_matrix = (
|
||||||
mathutils.Matrix.Translation(location)
|
mathutils.Matrix.Translation(location)
|
||||||
@ rotation.to_matrix().to_4x4()
|
@ rotation.to_matrix().to_4x4()
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from typing import Tuple, List, Optional
|
from typing import Tuple, List, Optional, Dict
|
||||||
import bpy
|
import bpy
|
||||||
import mathutils
|
import mathutils
|
||||||
from specklepy.objects import Base
|
from specklepy.objects import Base
|
||||||
@@ -118,6 +118,25 @@ def transform_matrix(transform: List[float]) -> mathutils.Matrix:
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def build_object_id_map(root_object: Base) -> Dict[str, Base]:
|
||||||
|
"""
|
||||||
|
Builds a dictionary mapping object IDs (both id and applicationId) to objects.
|
||||||
|
"""
|
||||||
|
id_map = {}
|
||||||
|
traversal_function = create_default_traversal_function()
|
||||||
|
|
||||||
|
for traversal_item in traversal_function.traverse(root_object):
|
||||||
|
obj = traversal_item.current
|
||||||
|
|
||||||
|
if hasattr(obj, "id") and obj.id:
|
||||||
|
id_map[obj.id] = obj
|
||||||
|
|
||||||
|
if hasattr(obj, "applicationId") and obj.applicationId:
|
||||||
|
id_map[obj.applicationId] = obj
|
||||||
|
|
||||||
|
return id_map
|
||||||
|
|
||||||
|
|
||||||
def find_object_by_id(root_object: Base, target_id: str) -> Optional[Base]:
|
def find_object_by_id(root_object: Base, target_id: str) -> Optional[Base]:
|
||||||
"""
|
"""
|
||||||
finds an object using traversal, checking both id and applicationId
|
finds an object using traversal, checking both id and applicationId
|
||||||
|
|||||||
+1
-1
@@ -5,7 +5,7 @@ description = "Next-Gen Speckle connector for Blender!"
|
|||||||
requires-python = ">=3.11.9, <4.0.0"
|
requires-python = ">=3.11.9, <4.0.0"
|
||||||
license = "Apache-2.0"
|
license = "Apache-2.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"specklepy>=3.0.3",
|
"specklepy>=3.0.4",
|
||||||
]
|
]
|
||||||
|
|
||||||
[dependency-groups]
|
[dependency-groups]
|
||||||
|
|||||||
Reference in New Issue
Block a user