/
Blender Optimization package (BOP)

Blender Optimization package (BOP)

Welcome on documentation page for a Blender addon called Blender Optimization package. Shortly BOP. Documentation provide all necessary explanations including tutorial how to install addon


 


 

What is BOP ?

BOP stands for Blender Optimization package. It is a python script made in Blender and is meant to be used for a Twinzo clients and automatization processes specific for Twinzo use-case. Script uses multiple Build in functions which can greatly help with 3D model optimization and save time.

 

How it looks like ?

image-20240529-120605.png

What individual functions does script use ?

Script uses these functions:

It will delete all mesh faces which normals are facing down (Z-). Function also ignore objects with active modifiers

It will smooth out mesh normals to default value. Usually its 30°

Custom Split Normals are a way to tweak/fake shading by pointing them towards other directions than default, auto-computed ones. By deleting them 3D model switch to default one provided by Blender

It will delete duplicated meshes which are roughly at the same location and same shape

It will convert Triangulated mesh into the Quads for a Unity

It will merge connected or unconnected vertices which positions are exact or very close (Threshold is 0.05)

It will simplify your mesh by dissolving vertices and edges separating flat regions. Reduces detail on planar faces and linear edges

It will make sure that all non necessary or unused data are deleted so exported model will be clean (Applies on whole scene)

It will put all objects inside scene under the empty parent object in order to function properly in Unity (Applies on all objects) Also it will keep hierarchy for multi-storey buildings

It will recalculate the normals of selected faces so that they point outside the volume that the face belongs to

It will rotate 3D model in such a way, that side with higher average number of normals facing corresponding direction will be rotated to facing up (Z+). Function is intended to use on Scanned meshes!

It will set objects origin point to their relative median position

It will reset object scale values in scene to 1 (Ignores empty and instances)

It will merge similar materials based on their values and colors (Applies on all objects in project)

It will delete unnecessary types of objects like (Empty, Lights, Objects without mesh data). (Its best to use it before Object Instancer and Reparent function!)

It will delete any armature or bones in scene (Be careful if there is no selected object, it will clear all parents in scene!)

It will delete animations in scene

It will Bake Animation data into objects itself. This will allow easier manipulation with objects

It will adjust LOD names to ensure that Unity will recognize them

It will enable Automatic packing of external data such as textures and others into .blend file. Default state is False. Since its toggle, if you press it twice, it will disable function. Additionally you can check it in File - External data - Automatically Pack Resources

It will Delete non visible geometry via camera array and ray casting


How to install addon ?

 

Software needed:

 

  1. Download zip file called BOP

  2. Then we head to Blender and install addon. Click on edit panel in top left corner, then choose Preferences.

    image-20240529-120642.png

    1. If you are using Blender 4.1 or older

      1. Then choose Add-ons and click on Install in top right corner.

        image-20240529-120712.png

      2. Find the downloaded ZIP file and click again on Install button.

        image-20240529-120746.png

      3. After that make sure you enabled addon.

        image-20240529-120807.png

    2. If you are using Blender 4.2 or newer

      1. Then head to Extensions, click on drop down arrow in top right corner and choose option Install Legacy Add-on or Install from Disk

        image-20240529-120831.png

      2. Find downloaded ZIP file and click again on Install button.

        image-20240529-120859.png

      3. After that make sure you enabled addon

        image-20240529-120933.png


How to use it ?

  1. After we installed addon we need to press N to bring up side panel and locate it under Twinzo section

image-20240529-121003.png
  1. Now you should see this panel:

    image-20240529-121120.png

  2. There are 4 Buttons.

    1. FIX my Classic Model

    2. FIX my Scanned Model

    3. RESET

    4. Need help ?

  3. There are 8 drop down menus

    1. Individual Functions

    2. Object Separator

    3. Object Instancer

    4. Analyze scene

    5. Decimator v5

    6. Exporter

    7. Documentation

    8. Updater

FIX my Classic Model button:

This button execute multiple functions which are specifically designed for a Classic Models like Demo Factory or Demo Office which you can check in app

You can use this button also for a Scanned Models but its not specifically designed for them

This button execute individual functions automatically in this order

  1. Bake Animation data

  2. Reset Origin

  3. Reset Scale to 1

  4. Recalculate Mesh Normals

  5. Auto Smooth All

  6. Delete Custom split Normals

  7. Convert triangulated mesh into quads

  8. Merge Close vertices

  9. Remove Unused datablocks

  10. Delete Bloat

  11. Merge Similar Materials

  12. Delete Downward faces

  13. Create Instanced meshes

  14. Decimator_v5

  15. Adjust LOD names

  16. Reparent

FIX my Scanned Model button:

This button execute multiple functions which are suitable for a Scanned Models via Lidar and similar technologies. Which are scanned models ? You can check Construction house or Twinzo office inside Twinzo app

You can use this button also for a Classic Models but its not specifically designed for them

  1. Delete Animations

  2. Delete any Armature and Bones

  3. Reset Origin

  4. Reset Scale to 1

  5. Delete Custom split Normals

  6. Merge Close vertices

  7. Remove Unused datablocks

  8. Delete Bloat

  9. Fix Scan rotation

  10. Merge Similar Materials

  11. Decimator_v5

  12. Adjust LOD names

  13. Reparent

RESET button:

Reset button will reset whole scene into the original state. Depending on your previous steps it may not resets everything as you expect.

Use this button with CAUTION!

Need help ? button:

Need help ? button is located under the drop down menu of Documentation and it will bring you to the addon documentation webpage


Individual functions

 

Drop down menu contain all individual functions which can be executed manually in any order as you like

Almost all functions works either on selected objects or when is nothing selected it will apply on all objects in scene

You can also search for each individual function by pressing F3 or Spacebar

Blender BOP vyhladavanie scriptov.gif


Object Separator

Object Separator will separate objects by loose parts (Mesh parts which are not connected to anything) and deletes meshes with Z-dimensions equal to 0

Its best to use it before Object Instancer


Object Instancer

Object Instancer will make identic or very similar objects as instanced one. Its doing it by checking distances between mesh vertices. Script will also remake already instanced meshes.

Instanced objects are very important in our workflow.


Analyze scene

Analyze scene will do exactly what is says. Its divided into 2 categories: Classical mesh and Scan Lidar mesh. Each category holds analyze scene tools designed for specific mesh type. Currently there are 2 types of tools: Heatmap and Statistic chart.

Heatmap:

Heatmap is graphical representation of data visualized in different colors, based on your model condition. You can check your model condition by using various methods:

It will assign colors to objects based on number of polygons

It will assign colors to objects based on face orientation (Backface culling)

It will assing colors to multi user objects (Instanced meshes) and non multi user objects (Not Instanced meshes)

It will assign colors to objects based on their mesh density. In more complex scenes, this function can take a some time

Statistic chart:

By pressing button Calculate Statistics, script will analyze scene and write statistics down bellow. There are two panels:

  1. Statistics

    1. In statistics panel are stored various information about currently opened scene

  2. Results

    1. In Results panel are stored Potentional Issues which are based on collected statistics in currently opened scene. Potentional Issues means that your model exceeded certain limit and may not work in Twinzo app


Decimator v5

Decimator v5 is a tool, designed to reduce vertex count while mostly preserving model shape and keeping textures mapping untouched. Tool is compatible with multi-user objects (Meshes with shared geometry). You can use Decimator v5 even on Scanned meshes but if your scan has significantly less vertexes than 750K, function can be too aggressive.

Tool is utilizing Blender native Decimate (Collapse) function with dynamic ratio values. Ratio values are dynamically calculated based on object face count and dimensions.

Conditions:

  1. If object has less faces, decimate function will be less aggressive

  2. If object dimensions are smaller than specified limit, decimate function will use aggressive decimate ratio since those objects are prompt to be less visible in app

  3. Non multi-user objects are processed separately

More details can be found in addon code

Reducing vertex count is heavy process and can take several minutes to compute. If you are not satisfied with results, you can always press CTRL+Z to revert changes.


Exporter

Exporter will do exactly what is says. In order to export your model, you need to specify valid export location. You can do that by pressing folder icon next to predefined text "Choose path".

image-20240529-121208.png

Couple things to know:

  1. Exported file format will be always FBX

  2. You don't need to write .fbx file extension while choosing export destination and file name. Exporter will automatically add it for you

  3. Why my exported file include name "BOP" before file format extension ?

    1. "BOP" text before file format extension will let us know that you used our addon and that can speed up processing if file will be uploaded through automatic Twinzo platform

If you set location which is not accessible or doesn't exist, addon will prompt you with error.

 


Documentation

Documentation contain button which direct you to this page


Updater

Updater drop down menu contain button for checking available updates


Known limitations

Since every 3D model is build differently with unique purpose in mind, sometimes addon will not perform as you expected. BOP is build to cover most of the general optimization steps to help you prepare your 3D model for a Twinzo app. If addon fails in some point to process your 3D model, you may consider to try fix it manually or try to use functions in different order. We also provide guideline to help you with it: https://twinzo.atlassian.net/l/cp/ZLbu2p6W

When addon tends to get slow or fail:

  1. When trying to optimize large number of objects at once (like 6000 and more)

  2. When trying to optimize very dense meshes with millions of polygons (1M and more), (Doesn't count for Lidar or photogrammetry based meshes)

  3. Function Merge close vertices relies on model scale units. If due to unit conversion, model scaling is about the values like 0.001 or negative -0.001 and more, it may require manual adjustment

 

Code snippet (Python)

# This script is not licensed and its primary designed for Twinzo use-cases. # However you are free to use, modify, and share this script, with or without modifications #----------------------------Purpose----------------------------# # Script is made in mind to help and automate optimization processes #----------------------------Disclaimer----------------------------# # Since every 3D model is made differently and for different purpose # Some parts of the script may not work or behave differently than expected # Depending on scene and model complexity, some scripts take some time to execute # Enjoy bl_info = { "name": "BOptimization package", "author": "Twinzo", "version": (2, 9), "blender": (3, 4, 0), "location": "View > N > Twinzo panel", "description": "Optimization package for process automatization", "doc_url": "https://twinzo.atlassian.net/wiki/spaces/PUBD/pages/104398849/Blender+Optimization+package+BOP", "category": "Object" } import bpy import bmesh import webbrowser import math import re import os import random import mathutils import bpy.utils.previews from mathutils import Vector, bvhtree from mathutils import Matrix import threading #----------------------------Enable developer extras----------------------------# # Enable developer extras to be able to search for custom operators def enable_developer_extras(): # Access user preferences prefs = bpy.context.preferences # Access the view section of user preferences view_prefs = prefs.view # Enable developer extras view_prefs.show_developer_ui = True #----------------------------Update checking----------------------------# # Check for addon updates def check_for_update(): bpy.context.window_manager.popup_menu(update_notification_menu, title="Addon Updater", icon='INFO') # Update notification menu def update_notification_menu(self, context): layout = self.layout layout.label(text=f"Current Version: {bl_info['version'][0]}.{bl_info['version'][1]}") layout.label(text="Click on button bellow to visit addon page") layout.operator("wm.url_open", text="Open Website").url = "https://twinzo.sharepoint.com/:f:/s/Publicdocumentation/Eo7iFbv43WxJh76Tn9b-dm8BTi_fMkn2sZjEr8m3U0VW4Q?e=o0bkPh" #----------------------------Defined functions----------------------------# # Function to delete faces with normals facing downward (treshold is defined in class) def delete_downward_faces(threshold=0.4): try: selected_objects = bpy.context.selected_objects # Check if any objects are selected if selected_objects: for obj in selected_objects: if obj.type == 'MESH': bpy.context.view_layer.objects.active = obj bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.normals_make_consistent(inside=False) bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.mode_set(mode='EDIT') bm = bmesh.from_edit_mesh(obj.data) faces_to_delete = [] for face in bm.faces: # Calculate the face normal normal = face.normal normal_world = obj.matrix_world.inverted().transposed() @ normal # Check if the face normal has a valid length if normal.length > 0 and normal_world.length > 0: # Calculate the angle between the face normal and the negative Z-axis angle = normal_world.angle(Vector((0, 0, -1))) if normal_world.z <= -threshold and angle <= threshold: faces_to_delete.append(face) if faces_to_delete: # Deselect all faces and select the faces to delete bpy.ops.mesh.select_all(action='DESELECT') for face in faces_to_delete: face.select = True # Delete the selected faces bpy.ops.mesh.delete(type='FACE') # Update the mesh after deleting the faces bmesh.update_edit_mesh(obj.data) bpy.ops.object.mode_set(mode='OBJECT') # Deselect all objects at the end bpy.ops.object.select_all(action='DESELECT') else: # Function to check if a face is facing downward def is_face_facing_downward(face, obj, threshold=0.2): z_vector = Vector((0, 0, -1)) # Negative Z-axis as a Vector world_matrix = obj.matrix_world normal = face.normal.copy() # Transform the face normal into world space normal = world_matrix.to_3x3() @ normal # Ensure the normal is normalized for consistent comparison normal.normalize() # Skip zero-length vectors to avoid the angle calculation error if normal.length == 0: return False # Compare the angle between the face normal and the downward direction (negative Z) angle = normal.angle(z_vector) # Tighten the angle threshold and z-component check for stricter deletion criteria return normal.z <= threshold and angle <= 0.523 # 0.523 radians = 30 degrees # Dictionary to keep track of processed multi-user objects processed_multi_user_objects = {} # Loop through visible objects and delete downward-facing faces for obj in bpy.context.visible_objects: if obj.type == 'MESH' and not obj.modifiers and not obj.is_instancer: # Check if object data is shared among multiple objects if obj.data.users > 1: multi_user_type = obj.data.name # Use the data name as a unique identifier # Process only once per multi-user mesh if multi_user_type not in processed_multi_user_objects: bm = bmesh.new() bm.from_mesh(obj.data) # Identify faces to delete faces_to_delete = [face for face in bm.faces if is_face_facing_downward(face, obj, threshold=0.2)] # Delete faces directly in bmesh if faces_to_delete: bmesh.ops.delete(bm, geom=faces_to_delete, context='FACES') # Write changes back to the mesh bm.to_mesh(obj.data) bm.free() # Mark the object as processed processed_multi_user_objects[multi_user_type] = obj else: # For non-multi-user objects, apply the logic directly bm = bmesh.new() bm.from_mesh(obj.data) # Identify faces to delete faces_to_delete = [face for face in bm.faces if is_face_facing_downward(face, obj, threshold=0.2)] # Delete faces directly in bmesh if faces_to_delete: bmesh.ops.delete(bm, geom=faces_to_delete, context='FACES') # Write changes back to the mesh bm.to_mesh(obj.data) bm.free() # Deselect all objects at the end bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("An error occurred:", str(e)) print("Continuing") # Define Function to apply "Convert Tris to Quads" to all objects def tris_to_quads(): try: selected_objects = bpy.context.selected_objects if selected_objects: # Switch to edit mode bpy.ops.object.mode_set(mode='EDIT') if not bpy.context.scene.tool_settings.use_transform_correct_face_attributes: bpy.context.scene.tool_settings.use_transform_correct_face_attributes = True if not bpy.context.scene.tool_settings.use_transform_correct_keep_connected: bpy.context.scene.tool_settings.use_transform_correct_keep_connected = True bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='FACE') bpy.ops.mesh.tris_convert_to_quads(face_threshold=0.698132, shape_threshold=0.698132, uvs=False) # Switch back to object mode bpy.ops.object.mode_set(mode='OBJECT') # Deselect all bpy.ops.object.select_all(action='DESELECT') else: # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Get all visible mesh objects in the scene visible_mesh_objects = [ obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get() ] if not visible_mesh_objects: print("No visible mesh objects in the scene.") return # Exit if no visible mesh objects are found # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_mesh_objects: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_mesh_objects) bpy.context.view_layer.objects.active = active_object # Switch to object mode to apply selection bpy.ops.object.mode_set(mode='OBJECT') # Switch to edit mode for all selected mesh objects and convert triangles to quads bpy.ops.object.mode_set(mode='EDIT') if not bpy.context.scene.tool_settings.use_transform_correct_face_attributes: bpy.context.scene.tool_settings.use_transform_correct_face_attributes = True if not bpy.context.scene.tool_settings.use_transform_correct_keep_connected: bpy.context.scene.tool_settings.use_transform_correct_keep_connected = True bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.tris_convert_to_quads(face_threshold=0.698132, shape_threshold=0.698132, uvs=False) bpy.ops.mesh.select_all(action='DESELECT') bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Tris to Quads failed:", str(e)) print("Continuing") # Function to apply auto-smooth to all objects def apply_auto_smooth(): try: selected_objects = bpy.context.selected_objects blender_version = bpy.app.version if selected_objects: for obj in selected_objects: if obj.type == 'MESH': bpy.context.view_layer.objects.active = obj # Check Blender version to determine shading method if blender_version >= (3, 6, 9): if "shade_smooth_by_angle" in dir(bpy.ops.object): # Use shade_smooth_by_angle if available bpy.ops.object.shade_smooth_by_angle(angle=0.523599, keep_sharp_edges=True) else: # Fallback to shade_smooth if shade_smooth_by_angle isn't available bpy.ops.object.shade_smooth(use_auto_smooth=True) else: # Use shade_smooth for older versions bpy.ops.object.shade_smooth(use_auto_smooth=True) else: # Deselect all objects to ensure autosmooth is applied to all visible objects bpy.ops.object.select_all(action='DESELECT') # Get all visible mesh objects in the scene visible_mesh_objects = [ obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get() ] if not visible_mesh_objects: print("No visible mesh objects in the scene.") return # Exit if no visible mesh objects are found # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_mesh_objects: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_mesh_objects) bpy.context.view_layer.objects.active = active_object if blender_version >= (3, 6, 9): if "shade_smooth_by_angle" in dir(bpy.ops.object): # Use shade_smooth_by_angle if available bpy.ops.object.shade_smooth_by_angle(angle=0.523599, keep_sharp_edges=True) else: # Fallback to shade_smooth if shade_smooth_by_angle isn't available bpy.ops.object.shade_smooth(use_auto_smooth=True) else: # Use shade_smooth for older versions bpy.ops.object.shade_smooth(use_auto_smooth=True) bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Autosmooth failed:", str(e)) print("Continuing") # Function to delete custom split normals data for all objects def delete_custom_split_normals(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: if obj.type == 'MESH': bpy.context.view_layer.objects.active = obj bpy.ops.mesh.customdata_custom_splitnormals_clear() else: # Dictionary to keep track of processed multi-user objects processed_multi_user_objects = {} # Deselect all objects to ensure autosmooth is applied to all visible objects bpy.ops.object.select_all(action='DESELECT') for obj in bpy.context.visible_objects: if obj.type == 'MESH': if obj.data.users > 1: multi_user_type = obj.data.name # Use the data name as a unique identifier # Check if this multi-user type has been processed if multi_user_type not in processed_multi_user_objects: # Set this object as the active one bpy.context.view_layer.objects.active = obj # Clear custom split normals bpy.ops.mesh.customdata_custom_splitnormals_clear() # Mark this type as processed processed_multi_user_objects[multi_user_type] = obj else: # For non-multi-user objects, apply the logic directly bpy.context.view_layer.objects.active = obj bpy.ops.mesh.customdata_custom_splitnormals_clear() # Deselect all objects at the end bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Autosmooth failed:", str(e)) print("Continuing") # Function to merge close vertices def merge_close_vertices(): try: threshold_distance = 0.001 selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: if obj.type == 'MESH': # Select the object and make it active bpy.context.view_layer.objects.active = obj bpy.ops.object.mode_set(mode='EDIT') # Get the mesh data of the object mesh = bmesh.from_edit_mesh(obj.data) # Ensure we're in vertex select mode bpy.ops.mesh.select_mode(type="VERT") # Merge close vertices bmesh.ops.remove_doubles(mesh, verts=mesh.verts, dist=threshold_distance) # Update mesh data bmesh.update_edit_mesh(obj.data) # Switch back to object mode bpy.ops.object.mode_set(mode='OBJECT') else: # Get all visible mesh objects in the scene visible_mesh_objects = [ obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get() ] if not visible_mesh_objects: print("No visible mesh objects in the scene.") return # Exit if no visible mesh objects are found # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_mesh_objects: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_mesh_objects) bpy.context.view_layer.objects.active = active_object # Perform operations on all selected objects bpy.ops.object.mode_set(mode='EDIT') # Select all vertices bpy.ops.mesh.select_all(action='SELECT') # Remove doubles bpy.ops.mesh.remove_doubles(threshold=threshold_distance) # Switch back to object mode bpy.ops.object.mode_set(mode='OBJECT') # Deselect all objects bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Merge close vertices failed:", str(e)) print("Continuing") # Function to Use Limited Dissolve def limited_dissolve(): try: selected_objects = bpy.context.selected_objects if selected_objects: bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='FACE') bpy.ops.mesh.dissolve_limited(angle_limit=0.0872665) #Adjust angle bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') else: # Dictionary to keep track of processed multi-user objects processed_multi_user_objects = {} # Deselect all objects to ensure autosmooth is applied to all visible objects bpy.ops.object.select_all(action='DESELECT') for obj in bpy.context.visible_objects: if obj.type == 'MESH': if obj.data.users > 1: multi_user_type = obj.data.name # Use the data name as a unique identifier # Check if this multi-user type has been processed if multi_user_type not in processed_multi_user_objects: # Set this object as the active one bpy.context.view_layer.objects.active = obj obj.select_set(True) # Select only the current object # Switch to EDIT mode bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='FACE') bpy.ops.mesh.dissolve_limited(angle_limit=0.0872665) # Adjust the angle limit as needed # Switch back to OBJECT mode bpy.ops.object.mode_set(mode='OBJECT') # Deselect the object and mark this type as processed obj.select_set(False) processed_multi_user_objects[multi_user_type] = obj else: bpy.context.view_layer.objects.active = obj obj.select_set(True) # Select only the current object bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='FACE') bpy.ops.mesh.dissolve_limited(angle_limit=0.0872665) # Adjust the angle limit as needed bpy.ops.object.mode_set(mode='OBJECT') obj.select_set(False) # Deselect all objects at the end bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Limited dissolve failed:", str(e)) print("Continuing") # Function to remove unused datablock def remove_unused_datablocks(): # Remove orphaned data (materials, textures, meshes, etc.) bpy.ops.outliner.orphans_purge(do_local_ids=True, do_linked_ids=True, do_recursive=True) # Function to reparent def reparent(empty_objects): def select_recursive(obj, selected_objects): for child_obj in obj.children: if child_obj.type == 'MESH': selected_objects.append(child_obj) if child_obj.children: select_recursive(child_obj, selected_objects) try: empty_found = False for specific_part in ['Outside', 'Interior', 'Architecture']: # Clear existing selection bpy.ops.object.select_all(action='DESELECT') # Check if the empty object contains the specified part matching_objects = [obj for obj in bpy.data.objects if obj.type == 'EMPTY' and specific_part in obj.name] if matching_objects: empty_found = True for empty_object in matching_objects: selected_objects = [] select_recursive(empty_object, selected_objects) # # Select the mesh objects # for obj in selected_objects: # obj.select_set(True) # Reparent the mesh objects back under the empty object with CLEAR_KEEP_TRANSFORM # bpy.ops.object.parent_clear(type='CLEAR_KEEP_TRANSFORM') for obj in selected_objects: obj.parent = empty_object # Clear existing selection bpy.ops.object.select_all(action='DESELECT') # Select hierarchy under the empty object bpy.context.view_layer.objects.active = empty_object bpy.ops.object.select_hierarchy(direction='CHILD', extend=True) # Filter selected objects to keep only empty objects under the empty object empty_objects_to_delete = [ obj for obj in bpy.context.selected_objects if obj.type == 'EMPTY' and obj.parent == empty_object ] # Delete the identified empty objects bpy.ops.object.select_all(action='DESELECT') for obj in empty_objects_to_delete: obj.select_set(True) bpy.ops.object.delete(use_global=False) # # Delete all empty objects not under an empty parent called 'Main' # main_empty = bpy.data.objects.get('Main') # if main_empty: # # Gather all empty objects not parented to 'Main' and not named 'Main' # empty_objects_to_delete = [ # obj for obj in bpy.data.objects if obj.type == 'EMPTY' and obj != main_empty and obj.parent != main_empty # ] # # Delete these empty objects # bpy.ops.object.select_all(action='DESELECT') # for obj in empty_objects_to_delete: # obj.select_set(True) # bpy.ops.object.delete(use_global=False) # Execute the specified code if none of the empty objects is found if not empty_found: bpy.ops.object.select_all(action='SELECT') bpy.ops.object.parent_clear(type='CLEAR_KEEP_TRANSFORM') bpy.ops.object.select_all(action='DESELECT') bpy.ops.object.select_by_type(extend=False, type='EMPTY') bpy.ops.object.delete(use_global=False) bpy.ops.object.select_all(action='DESELECT') # Create Main empty object bpy.ops.object.empty_add(type='PLAIN_AXES', align='WORLD', location=(0, 0, 0), scale=(1, 1, 1)) main_obj = bpy.context.object main_obj.name = "Main" # Create Floor 0 empty object and parent it to Main bpy.ops.object.empty_add(type='PLAIN_AXES', align='WORLD', location=(0, 0, 0), scale=(1, 1, 1)) floor_0_obj = bpy.context.object floor_0_obj.name = "Floor 0" floor_0_obj.parent = main_obj # List of child objects to be created under Floor 0 child_names = ["Floor 0 Architecture", "Floor 0 Interior", "Floor 0 Outside"] # Dictionary to store references to the created objects child_objects = {} # Create child objects and parent them to Floor 0 for name in child_names: bpy.ops.object.empty_add(type='PLAIN_AXES', align='WORLD', location=(0, 0, 0), scale=(1, 1, 1)) child_obj = bpy.context.object child_obj.name = name child_obj.parent = floor_0_obj child_objects[name] = child_obj # Store reference to the object # Select all visible mesh objects in the scene bpy.ops.object.select_all(action='DESELECT') for obj in bpy.context.scene.objects: if obj.type == 'MESH' and obj.visible_get(): obj.select_set(True) # Parent all selected mesh objects to "Floor 0 Architecture" floor_0_arch_obj = child_objects["Floor 0 Architecture"] bpy.context.view_layer.objects.active = floor_0_arch_obj # Set "Floor 0 Architecture" as active object bpy.ops.object.parent_set(type='OBJECT', keep_transform=True) # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Push undo step bpy.ops.ed.undo_push() except Exception as e: print("Error:", str(e)) print("Continuing") # Function to recalculate normals def recalculate_normals(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: bpy.context.view_layer.objects.active = obj bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.normals_make_consistent(inside=False) bpy.ops.object.mode_set(mode='OBJECT') else: # Get all visible mesh objects in the scene visible_mesh_objects = [ obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get() ] if not visible_mesh_objects: print("No visible mesh objects in the scene.") return # Exit if no visible mesh objects are found # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_mesh_objects: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_mesh_objects) bpy.context.view_layer.objects.active = active_object # Switch to edit mode bpy.ops.object.mode_set(mode='EDIT') # Select all vertices bpy.ops.mesh.select_all(action='SELECT') # Recalculate normals bpy.ops.mesh.normals_make_consistent(inside=False) # Deselect all objects bpy.ops.mesh.select_all(action='DESELECT') # Switch back to object mode bpy.ops.object.mode_set(mode='OBJECT') # Deselect all objects bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Recalculate normals failed:", str(e)) print("Continuing") # Function to open a Addon documentation def open_url(url): webbrowser.open(url) # Function to open Fix scan Rotation def rotate_model_operator(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: if obj.type == "MESH" and obj.data.users == 1: bpy.context.view_layer.objects.active = obj bpy.ops.object.select_all(action='DESELECT') obj.select_set(True) try: bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') bpy.ops.object.transform_apply(location=False, rotation=False, scale=True) except RuntimeError as e: if "Cannot apply to a multi user" in str(e): print(f"Skipping multi-user object: {obj.name}") continue else: raise e # Count the number of face normals facing in each direction normal_counts = [0, 0, 0] # X, Y, Z data = obj.data for polygon in data.polygons: normal = polygon.normal if normal.x > 0: normal_counts[0] += 1 if normal.y > 0: normal_counts[1] += 1 if normal.z > 0: normal_counts[2] += 1 # Find the index of the direction with the highest normal count highest_normal_direction = normal_counts.index(max(normal_counts)) # Calculate the rotation angle based on the highest normal direction rotation_angle = math.radians(-90 * highest_normal_direction) # Rotate the object to align the highest normal direction with the Z-up direction obj.rotation_euler = (0, 0, rotation_angle) bpy.ops.object.select_all(action='DESELECT') else: for obj in bpy.context.visible_objects: if obj.type == "MESH" and obj.data.users == 1: bpy.context.view_layer.objects.active = obj bpy.ops.object.select_all(action='DESELECT') obj.select_set(True) try: bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') bpy.ops.object.transform_apply(location=False, rotation=False, scale=True) except RuntimeError as e: if "Cannot apply to a multi user" in str(e): print(f"Skipping multi-user object: {obj.name}") continue else: raise e # Count the number of face normals facing in each direction normal_counts = [0, 0, 0] # X, Y, Z data = obj.data for polygon in data.polygons: normal = polygon.normal if normal.x > 0: normal_counts[0] += 1 if normal.y > 0: normal_counts[1] += 1 if normal.z > 0: normal_counts[2] += 1 # Find the index of the direction with the highest normal count highest_normal_direction = normal_counts.index(max(normal_counts)) # Calculate the rotation angle based on the highest normal direction rotation_angle = math.radians(-90 * highest_normal_direction) # Rotate the object to align the highest normal direction with the Z-up direction obj.rotation_euler = (0, 0, rotation_angle) bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("rotate_model_operator() failed:", str(e)) print("Continuing") # Function to reset origin on all objects in scene def reset_origin(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') bpy.ops.object.select_all(action='DESELECT') bpy.ops.ed.undo_push else: # Deselect all objects at the start (only once) bpy.ops.object.select_all(action='DESELECT') # Dictionary to track objects by their mesh data mesh_data_to_obj = {} # List to store objects that should have their origin reset objects_to_reset = [] # Loop through all mesh objects to organize them by mesh data for obj in bpy.data.objects: if obj.type == 'MESH' and obj.visible_get() and not obj.animation_data and (not obj.data.shape_keys or not obj.data.shape_keys.animation_data) and not obj.constraints: mesh_data = obj.data # Track objects sharing the same mesh data if mesh_data not in mesh_data_to_obj: mesh_data_to_obj[mesh_data] = [] mesh_data_to_obj[mesh_data].append(obj) # Now, loop through the mesh data dictionary for obj_list in mesh_data_to_obj.values(): # Check if any object in the list has animation data has_animation_data = any(other_obj.animation_data for other_obj in obj_list) # If no object in the list has animation data, add them to the reset list if not has_animation_data: objects_to_reset.extend(obj_list) # If we found objects that need their origin reset if objects_to_reset: # Manually set the active object and reset origin without re-selecting everything bpy.context.view_layer.objects.active = objects_to_reset[0] # Perform the origin set operation directly on the objects for obj in objects_to_reset: obj.select_set(True) # Set the selection status # Apply the origin_set operation to the selected objects bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') # Deselect all objects at the end (once) bpy.ops.object.select_all(action='DESELECT') # Push an undo step to the stack bpy.ops.ed.undo_push() except Exception as e: print("Reset origin failed:", str(e)) print("Continuing") # Function to reset scene to original state def reset_to_original_state(): # Store the initial state of the scene when the script is run for the first time if not hasattr(bpy.types.Scene, 'initial_objects'): bpy.types.Scene.initial_objects = None bpy.types.Scene.initial_cameras = None bpy.types.Scene.initial_lights = None bpy.types.Scene.initial_materials = None bpy.types.Scene.initial_objects = bpy.data.objects[:] bpy.types.Scene.initial_cameras = bpy.data.cameras[:] bpy.types.Scene.initial_lights = bpy.data.lights[:] bpy.types.Scene.initial_materials = bpy.data.materials[:] # Delete all objects, cameras, and lights that were added after importing the model for obj in bpy.data.objects: if obj not in bpy.context.scene.initial_objects: bpy.data.objects.remove(obj) for camera in bpy.data.cameras: if camera not in bpy.context.scene.initial_cameras: bpy.data.cameras.remove(camera) for light in bpy.data.lights: if light not in bpy.context.scene.initial_lights: bpy.data.lights.remove(light) # Function to remove duplicated materials def merge_similar_materials(): try: def are_values_similar(v1, v2, tolerance): return abs(v1 - v2) <= tolerance def get_shader_nodes(material): """Returns the BSDF nodes (Principled, Diffuse, Transparent, Translucent) for a given material.""" bsdf_nodes = {"principled": None, "diffuse": None, "transparent": None, "translucent": None} if material and material.use_nodes: for node in material.node_tree.nodes: if node.type == 'BSDF_PRINCIPLED': bsdf_nodes["principled"] = node elif node.type == 'BSDF_DIFFUSE': bsdf_nodes["diffuse"] = node elif node.type == 'BSDF_TRANSPARENT': bsdf_nodes["transparent"] = node elif node.type == 'BSDF_TRANSLUCENT': bsdf_nodes["translucent"] = node return bsdf_nodes def get_or_create_default_material(): """Checks if DefaultMaterial exists; if not, creates one and returns it.""" mat_name = "DefaultMaterial" # Check if material already exists existing_material = bpy.data.materials.get(mat_name) if existing_material: return existing_material # Create default material if it doesn't exist mat = bpy.data.materials.new(name=mat_name) mat.use_nodes = True bsdf_node = next((node for node in mat.node_tree.nodes if node.type == 'BSDF_PRINCIPLED'), None) if not bsdf_node: return None # Set default material properties bsdf_node.inputs['Base Color'].default_value = (0.8, 0.8, 0.8, 1.0) # Light gray bsdf_node.inputs['Metallic'].default_value = 0.0 bsdf_node.inputs['Roughness'].default_value = 0.5 bsdf_node.inputs['Alpha'].default_value = 1.0 return mat # Define tolerance values for each attribute diffuse_tolerance = 0.1 # Adjust as needed for diffuse color comparison metallic_tolerance = 0.1 # Adjust as needed for metallic comparison roughness_tolerance = 0.1 # Adjust as needed for roughness comparison alpha_tolerance = 0.1 # Adjust as needed for alpha comparison color_tolerance = 0.1 # Used for Diffuse & Translucent Color comparison # Create dictionaries to store material data (Script will compare values based on dictionary categories) material_data = {} # Set to track merged materials (to avoid re-merging) merged_materials = set() # Get a list of all visible objects in the current scene visible_objects = [obj for obj in bpy.context.scene.objects if obj.visible_get()] # Get or create the default material default_material = get_or_create_default_material() # Iterate over objects for obj in visible_objects: if obj.type == 'MESH': if obj.data.color_attributes: continue if not obj.material_slots: obj.data.materials.append(default_material) for slot in obj.material_slots: material = slot.material if not material or material.node_tree is None: continue if material in merged_materials: continue # Check if the material has a texture added to the TEX_IMAGE node has_texture = False for node in material.node_tree.nodes: if node.type == 'TEX_IMAGE' and node.image: has_texture = True break # Skip materials with a texture in the TEX_IMAGE node if has_texture: continue # Get shader nodes nodes = get_shader_nodes(material) # Extract properties base_color = (0, 0, 0, 1) metallic = 0.0 roughness = 0.5 alpha = 1.0 diffuse_color = (0, 0, 0, 1) translucent_color = (0, 0, 0, 1) has_transparent = nodes["transparent"] is not None if nodes["principled"]: bsdf_node = nodes["principled"] base_color = bsdf_node.inputs['Base Color'].default_value metallic = bsdf_node.inputs['Metallic'].default_value roughness = bsdf_node.inputs['Roughness'].default_value alpha = bsdf_node.inputs['Alpha'].default_value if nodes["diffuse"]: diffuse_color = nodes["diffuse"].inputs['Color'].default_value if nodes["translucent"]: translucent_color = nodes["translucent"].inputs['Color'].default_value key = ( tuple(base_color[:3]), # RGB only metallic, roughness, alpha, tuple(diffuse_color[:3]), # Diffuse color tuple(translucent_color[:3]), # Translucent color has_transparent # Boolean ) merged_material = None for existing_key, materials in material_data.items(): if materials[0][0] == material: continue base_color_match = all( are_values_similar(v1, v2, diffuse_tolerance) for v1, v2 in zip(base_color[:3], existing_key[0]) ) metallic_match = are_values_similar(metallic, existing_key[1], metallic_tolerance) roughness_match = are_values_similar(roughness, existing_key[2], roughness_tolerance) alpha_match = are_values_similar(alpha, existing_key[3], alpha_tolerance) diffuse_match = all( are_values_similar(v1, v2, color_tolerance) for v1, v2 in zip(diffuse_color[:3], existing_key[4]) ) translucent_match = all( are_values_similar(v1, v2, color_tolerance) for v1, v2 in zip(translucent_color[:3], existing_key[5]) ) transparent_match = existing_key[6] == has_transparent if ( base_color_match and metallic_match and roughness_match and alpha_match and diffuse_match and translucent_match and transparent_match ): merged_material = materials[0][0] break if merged_material is None: material_data[key] = [(material, base_color)] else: for obj in visible_objects: for slot in obj.material_slots: if slot.material == material: slot.material = merged_material merged_materials.add(material) # Remove unused materials and fix suffixes for material in bpy.data.materials: material.name = re.sub(r'\.\d+$', '', material.name) if not any(slot.material == material for obj in visible_objects for slot in obj.material_slots): bpy.data.materials.remove(material) # Iterate over all materials in the current Blender file for material in bpy.data.materials: # Ensure the material uses nodes if material.use_nodes: # Get the node tree of the material node_tree = material.node_tree # Find the relevant BSDF nodes bsdf_nodes = [] for node in node_tree.nodes: if node.type in {'BSDF_PRINCIPLED', 'BSDF_TRANSPARENT', 'BSDF_TRANSLUCENT'}: bsdf_nodes.append(node) # Process each relevant BSDF node for bsdf_node in bsdf_nodes: # Copy the Viewport Display color to the Base Color viewport_color = material.diffuse_color[:3] # Get the RGB part of the diffuse color if bsdf_node.type == 'BSDF_PRINCIPLED': base_color = bsdf_node.inputs['Base Color'].default_value[:3] # Get the RGB part of the Base Color input_name = 'Base Color' elif bsdf_node.type == 'BSDF_TRANSPARENT' or bsdf_node.type == 'BSDF_TRANSLUCENT': base_color = bsdf_node.inputs['Color'].default_value[:3] # Get the RGB part of the Color input_name = 'Color' # Skip if the Base Color is already the same as the Viewport Display color if viewport_color != base_color: bsdf_node.inputs[input_name].default_value = (*viewport_color, 1.0) # Set Color with full alpha except Exception as e: print("Merge similar materials failed:", str(e)) print("Continuing") # Function to make instanced objects from similar meshes in scene with treshold def compare_objects(obj1, obj2, threshold_distance): try: mesh_data1, mesh_data2 = obj1.data, obj2.data # Compare geometry for v1, v2 in zip(mesh_data1.vertices, mesh_data2.vertices): if (v1.co - v2.co).length > threshold_distance: return False # Compare materials (ignoring order of slots) materials1 = {slot.material for slot in obj1.material_slots if slot.material} materials2 = {slot.material for slot in obj2.material_slots if slot.material} if materials1 != materials2: return False return True except Exception as e: print("Compare_meshes failed:", str(e)) print("Continuing") # Function to separate objects by loose parts and delete faces with Z-dimension equal to 0 def object_separator(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: if obj.type == 'MESH' and obj.data.users == 1: bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='VERT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.remove_doubles(threshold=0.001) bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='FACE') bpy.ops.mesh.separate(type='LOOSE') bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') else: bpy.ops.object.select_all(action='DESELECT') bpy.ops.object.mode_set(mode='OBJECT') for obj in bpy.context.visible_objects: if obj.type == "MESH" and obj.data.users == 1: obj.select_set(True) bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.separate(type='LOOSE') bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Object instancer failed:", str(e)) print("You are trying to separate objects with shared geometry or objects without loose parts") # Function to delete bloat def delete_bloat(): try: def collect_children(parent): """ Recursively collect all child objects of a given parent. """ children = set() for child in parent.children: children.add(child) children.update(collect_children(child)) return children # Collect all objects in the scene all_objects = bpy.context.scene.objects # Initialize sets to collect objects for different operations # objects_to_delete_height = set() objects_to_delete_type = set() protected_objects = set() # Find the 'Main' object and its descendants main_obj = bpy.context.scene.objects.get('Main') if main_obj: # Collect children of 'Main' object children = collect_children(main_obj) if children: protected_objects.add(main_obj) protected_objects.update(children) for obj in all_objects: # Check if the object is a multiuser instance if obj.users > 1: continue # Skip multiuser objects # # Collect objects for deletion if Z-height is close to 0 # if obj.type == 'MESH' and abs(obj.dimensions.z) < 0.001: # objects_to_delete_height.add(obj) # Collect objects for deletion by type object_types_to_delete = {'CAMERA', 'LIGHT', 'CURVE'} if obj in protected_objects: continue # Skip protected objects if obj.type in object_types_to_delete or (obj.type == 'MESH' and obj.data is None): objects_to_delete_type.add(obj) # Check for Empty objects, delete 'Main' only if it has no children elif obj.type == 'EMPTY' and (obj.name != 'Main' or (obj.name == 'Main' and not obj.children)): objects_to_delete_type.add(obj) # # Delete selected objects by height # bpy.ops.object.select_all(action='DESELECT') # for obj in objects_to_delete_height: # obj.select_set(True) # bpy.ops.object.delete() # Delete objects by type for obj in objects_to_delete_type: bpy.data.objects.remove(obj, do_unlink=True) # Delete cameras for camera in bpy.context.scene.collection.objects: if camera.type == 'CAMERA': bpy.data.objects.remove(camera, do_unlink=True) # Delete lights for light in bpy.context.scene.collection.objects: if light.type == 'LIGHT': bpy.data.objects.remove(light, do_unlink=True) # # Delete Images # for image in bpy.context.scene.collection.objects: # if image.type == 'IMAGE': # bpy.data.objects.remove(image, do_unlink=True) # Delete Curves for curve in bpy.context.scene.collection.objects: if curve.type == 'CURVE': bpy.data.objects.remove(curve, do_unlink=True) # Call reparent function since it will delete unnecessary empty objects # try: # bpy.ops.object.reparent() # # except Exception as e: # print("Reparent function inside Delete bloat failed:", str(e)) # print("Continuing") # bpy.ops.object.select_all(action='SELECT') # bpy.ops.object.parent_clear(type='CLEAR_KEEP_TRANSFORM') # bpy.ops.object.select_all(action='DESELECT') # bpy.ops.object.empty_add(type='PLAIN_AXES', align='WORLD', location=(0, 0, 0), scale=(1, 1, 1)) # bpy.ops.object.select_grouped(type='TYPE') # bpy.ops.object.delete(use_global=False) # bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Delete bloat failed:", str(e)) print("Continuing") # Function to reset model scaling to 0 (Ignores instanced objects) def reset_scale_to_1(): try: selected_objects = bpy.context.selected_objects if selected_objects: # Collect non-empty, non-instanced objects objects_to_reset_scale = [obj for obj in selected_objects if obj.type != 'EMPTY' and not obj.is_instancer and obj.data.users == 1] # Set the active object to the first in the list if objects_to_reset_scale: bpy.context.view_layer.objects.active = objects_to_reset_scale[0] # Apply scale to all objects in batches batch_size = 10 # Adjust this value based on your scene's performance for i in range(0, len(objects_to_reset_scale), batch_size): batch = objects_to_reset_scale[i:i + batch_size] # Select objects in the batch for obj in batch: obj.select_set(True) # Apply scale bpy.ops.object.transform_apply(location=False, rotation=False, scale=True, properties=True, isolate_users=False) # Deselect objects in the batch for obj in batch: obj.select_set(False) # bpy.ops.object.select_all(action='SELECT') # bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') # Deselect all objects bpy.ops.object.select_all(action='DESELECT') else: # Deselect all objects first to ensure we're starting clean bpy.ops.object.select_all(action='DESELECT') # Collect non-empty, non-instanced objects without animations, shapekey animations, or constraints objects_to_reset_scale = [obj for obj in bpy.context.visible_objects if obj.type == 'MESH' and not obj.is_instancer and obj.data.users == 1 and not obj.animation_data and (not obj.data.shape_keys or not obj.data.shape_keys.animation_data) and not obj.constraints] # Set the active object to the first in the list if objects_to_reset_scale: bpy.context.view_layer.objects.active = objects_to_reset_scale[0] # Apply scale to all objects in batches batch_size = 10 # Adjust this value based on your scene's performance for i in range(0, len(objects_to_reset_scale), batch_size): batch = objects_to_reset_scale[i:i + batch_size] # Select objects in the batch for obj in batch: obj.select_set(True) # Apply scale transformation bpy.ops.object.transform_apply(location=False, rotation=False, scale=True, properties=True, isolate_users=False) # Deselect objects in the batch for obj in batch: obj.select_set(False) # Final deselection of all objects bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Reset scale failed:", str(e)) print("Continuing") # Function to delete any bones or armature in the scene def delete_armatures_and_bones(): try: selected_objects = bpy.context.selected_objects if selected_objects: for obj in selected_objects: if obj.type == 'ARMATURE': # Detach all children of the armature children = [child for child in obj.children if child.type != 'ARMATURE'] for child in children: # If armature has a parent, set it as the parent for its children if obj.parent: child.matrix_parent_inverse = child.matrix_world.inverted() child.parent = obj.parent else: child.matrix_parent_inverse = child.matrix_world.inverted() child.parent = None # Delete armature object bpy.data.objects.remove(obj) else: # No selected objects; iterate through objects in the scene armatures_to_delete = [] for obj in bpy.context.scene.objects: if obj.type == 'ARMATURE' and obj.visible_get(): armatures_to_delete.append(obj) # Detach children of each armature for armature in armatures_to_delete: children = [child for child in armature.children if child.type != 'ARMATURE'] for child in children: # If armature has a parent, set it as the parent for its children if armature.parent: child.matrix_parent_inverse = child.matrix_world.inverted() child.parent = armature.parent else: child.matrix_parent_inverse = child.matrix_world.inverted() child.parent = None # Delete selected armatures bpy.ops.object.select_all(action='DESELECT') for obj in armatures_to_delete: obj.select_set(True) bpy.ops.object.delete() # Clear selection bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Delete armature and bones failed:", str(e)) print("Continuing") # Function to delete any animation data from objects in scene def delete_animations(): try: selected_objects = bpy.context.selected_objects if selected_objects: # Clear animation data for selected objects bpy.ops.anim.keyframe_clear_v3d() bpy.ops.object.select_all(action='DESELECT') else: # Get all visible mesh objects in the scene visible_objects_with_animation = [ obj for obj in bpy.context.scene.objects if obj.visible_get() and obj.animation_data ] if not visible_objects_with_animation: print("No visible objects with animation.") return # Exit if no visible mesh objects are found # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_objects_with_animation: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_objects_with_animation) bpy.context.view_layer.objects.active = active_object # Clear animation data for selected objects bpy.ops.anim.keyframe_clear_v3d() bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Delete animations failed:", str(e)) print("Continuing") # Function to fix LOD namings in all objects in the scene def lod_rename(): try: # Get a list of all objects in the scene objects = bpy.data.objects # Iterate through all objects for obj in objects: # Check if the object name contains "LOD" followed by a number import re match = re.search(r'LOD\d+', obj.name) if match: # Extract the matched text lod_text = match.group() # Remove the matched text from the original name new_name = obj.name.replace(lod_text, '') # Append an underscore and the matched text to the end of the name new_name += '_' + lod_text # Rename the object with the modified name obj.name = new_name except Exception as e: print("LOD_rename failed:", str(e)) print("Continuing") # Function to delete duplicated objects in the scene def select_and_delete_duplicates(threshold=0.1, exclude_text="LOD"): try: # bpy.ops.object.select_by_type(type='MESH') # bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') bpy.ops.object.select_all(action='DESELECT') # Dictionary to store mesh objects based on their location and shape mesh_object_data = {} # Function to calculate the distance between two objects def distance_threshold(obj1, obj2): distance = math.sqrt( (obj1[0] - obj2[0]) ** 2 + (obj1[1] - obj2[1]) ** 2 + (obj1[2] - obj2[2]) ** 2 ) return distance <= threshold # Iterate through all mesh objects in the scene for obj in bpy.data.objects: # Check if the object is a mesh, is visible in the scene, and does not contain the exclude_text in its name if obj.type == 'MESH' and obj.visible_get() and exclude_text not in obj.name: # Create a tuple with the location and vertices (shape) rounded to 5 decimal places data_key = ( round(obj.location.x, 5), round(obj.location.y, 5), round(obj.location.z, 5), tuple(round(v.co.x, 5) for v in obj.data.vertices), tuple(round(v.co.y, 5) for v in obj.data.vertices), tuple(round(v.co.z, 5) for v in obj.data.vertices) ) # Check if there are existing objects within the distance threshold duplicate_found = False for existing_key in mesh_object_data: if distance_threshold(data_key, existing_key): duplicate_found = True mesh_object_data[existing_key].append(obj) break # If no duplicates are found within the threshold, add the object to the dictionary if not duplicate_found: mesh_object_data[data_key] = [obj] # Iterate through the dictionary and delete duplicates, leaving one in each group for key, mesh_objects in mesh_object_data.items(): if len(mesh_objects) > 1: for obj in mesh_objects[1:]: bpy.data.objects.remove(obj, do_unlink=True) except Exception as e: print("Delete Duplicated meshes failed:", str(e)) print("Continuing") # Function to reduce vertex count of objects in the scene def decimator_v5(): try: def adjust_ratio(face_count, dimensions): ratio = 1.0 if sum(1 for dim in dimensions if dim < 0.3) >= 2: ratio = 0.2 elif face_count >= 100000: ratio = 0.3 elif face_count >= 50000: ratio = 0.4 elif face_count >= 10000: ratio = 0.5 elif face_count >= 5000: ratio = 0.6 elif face_count >= 2000: ratio = 0.7 elif face_count >= 800: ratio = 0.8 return ratio # Dictionary to store multi-user objects multi_user_objects = {} # Get all visible mesh objects visible_meshes = [obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get()] # Iterate over visible mesh objects for obj in visible_meshes: # Check if object has less than 800 faces if len(obj.data.polygons) < 1000: continue # Check if object is multi-user if obj.data.users > 1: # Check if object type already exists in dictionary if obj.data.name in multi_user_objects: continue # Skip this object if another instance of the same type has been processed multi_user_objects[obj.data.name] = obj # Add object to dictionary obj = multi_user_objects[obj.data.name] # Use the original object # Select the object bpy.context.view_layer.objects.active = obj bpy.ops.object.mode_set(mode='EDIT') if not bpy.context.scene.tool_settings.use_transform_correct_face_attributes: bpy.context.scene.tool_settings.use_transform_correct_face_attributes = True if not bpy.context.scene.tool_settings.use_transform_correct_keep_connected: bpy.context.scene.tool_settings.use_transform_correct_keep_connected = True bpy.ops.mesh.select_all(action='SELECT') # Calculate face count and dimensions face_count = len(obj.data.polygons) dimensions = obj.dimensions # Adjust ratio based on face count and dimensions ratio = adjust_ratio(face_count, dimensions) # Apply decimate operator bpy.ops.mesh.decimate(ratio=ratio) # Switch back to object mode bpy.ops.object.mode_set(mode='OBJECT') except Exception as e: print("Decimator v5 failed:", str(e)) print("Continuing") # Function to pack external data into blend file def pack_external_data_into_blend_file(): try: bpy.ops.file.autopack_toggle() except Exception as e: print("Pack external data failed:", str(e)) print("Continuing") # Function to cut meshes into individual floors def floor_cutter(delete_interval=5): # Temporary delete instanced meshes def necessary_model_preparations(): bpy.ops.object.select_by_type(type='MESH') bpy.ops.object.make_single_user(object=True, obdata=True, material=False, animation=False, obdata_animation=False) bpy.ops.object.parent_clear(type='CLEAR_KEEP_TRANSFORM') bpy.ops.object.select_all(action='DESELECT') def delete_leftovers(): # Get all mesh objects with no polygons objects_to_delete = [obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and len(obj.data.polygons) == 0] # Select the objects to delete for obj in objects_to_delete: obj.select_set(True) # Delete the selected objects bpy.ops.object.delete() bpy.ops.object.select_all(action='DESELECT') def perform_bisect_operations(objects, heights): # Create a dictionary to store processed objects for each height processed_objects_dict = {height: set() for height in heights} # Counter to keep track of processed objects processed_count = 0 # Iterate over each height for height in heights: print(f"\nRunning bisect operation at height: {height}") # Iterate over each object for obj in objects: print("Processing object:", obj.name if obj else None) # Check if the object still exists if obj is None or obj.name not in bpy.context.scene.objects: print("Object no longer exists. Skipping...") continue # Skip the object if it's not a mesh, it's hidden, it has already been processed for this height, it has no polygons, or its z-dimension is zero if obj.type != 'MESH' or obj.hide_get() or obj.name in processed_objects_dict[height] or len( obj.data.vertices) == 0 or obj.dimensions.z == 0: print("Skipping object:", obj.name) continue # Add the object to the set of processed objects for this height processed_objects_dict[height].add(obj.name) print("Creating copy of object:", obj.name) # Create a unique copy of the object copied_obj = obj.copy() copied_obj.data = obj.data.copy() # Ensure the copied object has its own mesh data bpy.context.collection.objects.link(copied_obj) # Add the copied object to the set of processed objects for this height processed_objects_dict[height].add(copied_obj.name) # Perform bisect operation on the original object with clear_inner=True print("Bisecting original object:", obj.name) bpy.context.view_layer.objects.active = obj obj.select_set(True) bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.bisect(plane_co=(0, 0, height), plane_no=(0, 0, 1), clear_inner=True, clear_outer=False, use_fill=True) bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') # Select the copied object print("Selecting copied object:", copied_obj.name) copied_obj.select_set(True) bpy.context.view_layer.objects.active = copied_obj # Perform bisect operation on the copied object with clear_outer=True print("Bisecting copied object:", copied_obj.name) bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bpy.ops.mesh.bisect(plane_co=(0, 0, height), plane_no=(0, 0, 1), clear_inner=False, clear_outer=True, use_fill=True) bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') # Increment the processed objects count processed_count += 1 # Check if it's time to delete leftovers if processed_count % delete_interval == 0: print("Deleting leftovers...") delete_leftovers() # Get height values from objects in "Floor planes" collection floor_planes_collection = bpy.data.collections.get("Floor planes") if not floor_planes_collection: print("Error: Collection 'Floor planes' not found.") else: heights = [obj.location.z for obj in floor_planes_collection.objects if obj.type == 'MESH'] # Get all objects in the scene objects = bpy.context.scene.objects necessary_model_preparations() perform_bisect_operations(objects, heights) def after_cleanup(): print("Cleaning again") # Loop through each object for obj in objects: # Check if the object is a mesh and has no polygons if obj.type == 'MESH' and len(obj.data.polygons) == 0: # Select the object obj.select_set(True) else: obj.select_set(False) # Delete the selected objects bpy.ops.object.delete() bpy.ops.object.select_all(action='DESELECT') after_cleanup() def origin_correction(): print("origin correction") bpy.ops.object.select_all(action='DESELECT') bpy.ops.object.reset_origin() origin_correction() def parent_mesh_objects_based_on_height(): # Define the collection name where floor planes are stored collection_name = "Floor planes" # Get the collection collection = bpy.data.collections.get(collection_name) if not collection: print(f"Collection '{collection_name}' not found") return # Extract height values from the z location of objects in the collection floor_data = [] for obj in collection.objects: if obj.type == 'MESH': # Assuming floor planes are mesh objects floor_data.append((obj.name, obj.location[2])) # Sort the floor data by height floor_data.sort(key=lambda x: x[1]) # Create main parent object to hold all empty floor parents bpy.ops.object.empty_add(location=(0, 0, 0)) main_parent = bpy.context.active_object main_parent.name = "Main" main_parent.empty_display_size = 1.0 # Create empty parent objects for "Floor Architecture", "Interior", and "Others" and store their names floor_names = [] floor_ranges = [] prev_val = None for i, (floor_name, height) in enumerate(floor_data): floor_name_with_index = f"{floor_name} / ZHeight {height:.2f}" floor_names.append(floor_name_with_index) if prev_val is not None: floor_ranges.append((prev_val, height)) prev_val = height # Create empty object for "Floor Architecture" under main parent bpy.ops.object.empty_add(location=(0, 0, 0)) floor_parent = bpy.context.active_object floor_parent.name = floor_name_with_index floor_parent.empty_display_size = 0.5 floor_parent.parent = main_parent # Create empty objects for "Architecture", "Interior", and "Others" under each "Floor Architecture" for category in ["Architecture", "Interior", "Others"]: bpy.ops.object.empty_add(location=(0, 0, 0)) category_parent = bpy.context.active_object category_parent.name = f"{floor_name_with_index} {category}" category_parent.empty_display_size = 0.3 category_parent.parent = floor_parent # Get z location of visible mesh objects in the scene visible_mesh_objects_z = {} for obj in bpy.context.scene.objects: if obj.type == 'MESH' and obj.visible_get(): visible_mesh_objects_z[obj.name] = obj.location[2] # Parent mesh objects to corresponding "Architecture" empty parent based on ranges for obj_name, z_loc in visible_mesh_objects_z.items(): parent_assigned = False # Handle objects lower than the first specified value if z_loc < floor_ranges[0][0]: first_floor_name = floor_names[0] first_architecture_parent_name = f"{first_floor_name} Architecture" if first_architecture_parent_name in bpy.data.objects: first_architecture_parent = bpy.data.objects[first_architecture_parent_name] obj = bpy.data.objects[obj_name] obj.parent = first_architecture_parent parent_assigned = True # Handle objects within the specified ranges for i, (start, end) in enumerate(floor_ranges): if start <= z_loc < end: floor_name = floor_names[i] architecture_parent_name = f"{floor_name} Architecture" if architecture_parent_name in bpy.data.objects: architecture_parent = bpy.data.objects[architecture_parent_name] obj = bpy.data.objects[obj_name] obj.parent = architecture_parent parent_assigned = True break # If no appropriate range found, assign to the last floor if not parent_assigned and floor_ranges: last_floor_name = floor_names[-1] last_architecture_parent_name = f"{last_floor_name} Architecture" if last_architecture_parent_name in bpy.data.objects: last_architecture_parent = bpy.data.objects[last_architecture_parent_name] obj = bpy.data.objects[obj_name] obj.parent = last_architecture_parent parent_mesh_objects_based_on_height() # Remove mesh objects in collection "Floor planes" def remove_floor_planes(): collection = bpy.data.collections.get("Floor planes") if collection: for obj in collection.objects: if obj.type == 'MESH': bpy.data.objects.remove(obj, do_unlink=True) remove_floor_planes() # Remove increment suffix from empty objects' names def remove_increment_suffix(): for obj in bpy.data.objects: if obj.type == 'EMPTY': original_name = obj.name # Check if name ends with increment suffix dot_index = original_name.rfind('.') if dot_index != -1 and original_name[dot_index+1:].isdigit(): # Remove increment suffix new_name = original_name[:dot_index] obj.name = new_name # remove_increment_suffix() # Function to dynamically update Default Floor slider inside Floor cutter def update_default_floor(self, context): scene = context.scene max_floors = scene.number_of_floors # Ensure default_floor does not exceed number_of_floors if scene.default_floor > max_floors: scene.default_floor = max_floors # Function to dynamically update Nubmer of Floors slider inside Floor cutter def update_number_of_floors(self, context): update_default_floor(self, context) # Function to display heatmap by polygon number def display_heatmap_by_polygons(): # Precompute color mapping color_mapping = { (0, 800): (0.0, 1.0, 0.0, 1.0), # Good (801, 5000): (1.0, 0.5, 0.0, 1.0), # Mid (5001, float('inf')): (1.0, 0.0, 0.0, 1.0) # Bad } # Batch face color assignment processed_meshes = set() for obj in bpy.context.visible_objects: if obj.type == 'MESH' and obj.data.polygons: mesh_data = obj.data # Skip if this mesh data has already been processed if mesh_data in processed_meshes: continue num_polygons = len(mesh_data.polygons) # Determine the color based on the number of polygons using color_mapping color = next((mapped_color for (min_polygons, max_polygons), mapped_color in color_mapping.items() if min_polygons <= num_polygons <= max_polygons), None) if color is not None: # Ensure the mesh has a vertex color layer named "Heatmap" color_layer = mesh_data.vertex_colors.get("Heatmap") if color_layer is None: color_layer = mesh_data.vertex_colors.new(name="Heatmap") # Set "Heatmap" as the active vertex color layer mesh_data.vertex_colors.active = color_layer # Set the color for all faces of the object color_data = [color] * len(mesh_data.loops) # Use batch update with `foreach_set` color_layer.data.foreach_set("color", [component for loop in mesh_data.loops for component in color]) # Mark this mesh data as processed processed_meshes.add(mesh_data) # Set the shading color type to 'VERTEX' in the active 3D Viewport for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': space.shading.color_type = 'VERTEX' break break # Function to visualize backface culling def create_face_color_manager(): # Store the original colors in the closure original_face_front_color = None original_face_back_color = None def set_face_colors(face_front_rgba, face_back_rgba): nonlocal original_face_front_color, original_face_back_color # Retrieve the current active theme active_theme = bpy.context.preferences.themes.items()[0][1] # Access the ThemeView3D settings theme_view3d = active_theme.view_3d # Store the original colors if not already stored if original_face_front_color is None: original_face_front_color = theme_view3d.face_front[:] if original_face_back_color is None: original_face_back_color = theme_view3d.face_back[:] # Modify the face_front and face_back properties with provided RGBA values theme_view3d.face_front = face_front_rgba theme_view3d.face_back = face_back_rgba for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': if not space.overlay.show_face_orientation: space.overlay.show_face_orientation = True break def restore_face_colors(): nonlocal original_face_front_color, original_face_back_color # Retrieve the current active theme active_theme = bpy.context.preferences.themes.items()[0][1] # Access the ThemeView3D settings theme_view3d = active_theme.view_3d # Restore the original colors if they were stored if original_face_front_color is not None: theme_view3d.face_front = original_face_front_color original_face_front_color = None # Reset to allow setting new one if original_face_back_color is not None: theme_view3d.face_back = original_face_back_color original_face_back_color = None # Reset to allow setting new one for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': if space.overlay.show_face_orientation: space.overlay.show_face_orientation = False break return set_face_colors, restore_face_colors # Function to clear heatmap data def clear_heatmap_data(): attribute_name = 'Heatmap' # Loop through each object in the scene for obj in bpy.data.objects: if obj.type == 'MESH': # Access the mesh data mesh_data = obj.data # Check if the vertex color layer with the specified name exists if attribute_name in mesh_data.color_attributes: # Remove the vertex color layer directly mesh_data.color_attributes.remove(mesh_data.color_attributes[attribute_name]) # Set the shading color type to 'MATERIAL' in all 3D Viewports for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': space.shading.color_type = 'MATERIAL' break break # Function to visualize instanced meshes (Multi user objects) def display_heatmap_by_instances(): # Define the colors to be used for multiuser and non-multiuser objects multiuser_color = (0.0, 1.0, 0.0, 1.0) # Green non_multiuser_color = (1.0, 0.0, 0.0, 1.0) # Red # Track processed mesh data blocks to handle each only once processed_meshes = set() # Iterate over all objects in the scene for obj in bpy.context.visible_objects: if obj.type == 'MESH' and obj.data.polygons: mesh_data = obj.data # Skip if this mesh data has already been processed if mesh_data in processed_meshes: continue # Determine the color based on whether the object is multiuser or not color = multiuser_color if mesh_data.users > 1 else non_multiuser_color # Ensure the mesh has a vertex color layer named "Heatmap" color_layer = mesh_data.vertex_colors.get("Heatmap") if color_layer is None: color_layer = mesh_data.vertex_colors.new(name="Heatmap") # Set "Heatmap" as the active vertex color layer mesh_data.vertex_colors.active = color_layer # Set the color for all faces of the object # Use a list comprehension for faster color data setup color_data = [color] * len(mesh_data.loops) # Use batch update with `foreach_set` color_layer.data.foreach_set("color", [comp for loop in mesh_data.loops for comp in color]) # Mark this mesh data as processed processed_meshes.add(mesh_data) # Set the shading color type to 'VERTEX' in the active 3D Viewport for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': space.shading.color_type = 'VERTEX' space.overlay.show_face_orientation = False break break # Function to visualize objects by their mesh density def display_heatmap_by_density(): orange_threshold = 0.2 red_threshold = 0.1 # Define color mapping green_color = (0.0, 1.0, 0.0, 1.0) # Green orange_color = (1.0, 0.5, 0.0, 1.0) # Orange red_color = (1.0, 0.0, 0.0, 1.0) # Red # Batch face color assignment processed_meshes = set() for obj in bpy.context.visible_objects: if obj.type == 'MESH' and obj.data.polygons: mesh_data = obj.data # Skip if this mesh data has already been processed if mesh_data in processed_meshes: continue # Ensure the mesh has a vertex color layer named "Heatmap" color_layer = mesh_data.vertex_colors.get("Heatmap") if color_layer is None: color_layer = mesh_data.vertex_colors.new(name="Heatmap") # Set "Heatmap" as the active vertex color layer mesh_data.vertex_colors.active = color_layer # Get the transformation matrix of the object obj_matrix = obj.matrix_world # Function to check if vertices are closely packed def vertices_are_close(face, threshold): vertices = [obj_matrix @ mesh_data.vertices[vertex].co for vertex in face.vertices] for i, v1 in enumerate(vertices): for v2 in vertices[i+1:]: distance = (v1 - v2).length if distance > threshold: return False return True # Set the color for faces based on vertex density for poly in mesh_data.polygons: if vertices_are_close(poly, red_threshold): color = red_color elif vertices_are_close(poly, orange_threshold): color = orange_color else: color = green_color for loop_index in poly.loop_indices: color_layer.data[loop_index].color = color # Mark this mesh data as processed processed_meshes.add(mesh_data) # Set the shading color type to 'VERTEX' in the active 3D Viewport for area in bpy.context.screen.areas: if area.type == 'VIEW_3D': for space in area.spaces: if space.type == 'VIEW_3D': space.shading.color_type = 'VERTEX' break break # Function to bake animations to keyframes def bake_animations(): blender_version = bpy.app.version # Get the frame start and frame end from the current scene scene = bpy.context.scene frame_start = scene.frame_start frame_end = scene.frame_end # Deselect all objects first bpy.ops.object.select_all(action='DESELECT') # Collect objects that need to be selected in one iteration objects_to_select = set() # Using a set to avoid duplicates for obj in bpy.data.objects: # Check if the object is visible if obj.visible_get(): # Add object if it has animation data if obj.animation_data is not None: objects_to_select.add(obj) # If the object is an armature, also select the armature itself if obj.type == 'ARMATURE' and obj.animation_data is not None: objects_to_select.add(obj) # Add all objects parented to this armature or its bones for child_obj in bpy.data.objects: if child_obj.parent == obj or (child_obj.parent_type == 'BONE' and child_obj.parent == obj): objects_to_select.add(child_obj) # If the object's parent has animation data, select the object if obj.parent is not None and obj.parent.animation_data is not None: objects_to_select.add(obj) # Select all the collected objects for obj in objects_to_select: obj.select_set(True) if blender_version >= (4, 1, 0): # Use the frame_start and frame_end from the scene bpy.ops.nla.bake( frame_start=frame_start, frame_end=frame_end, step=1, only_selected=True, visual_keying=True, clear_constraints=True, clear_parents=False, use_current_action=False, clean_curves=False, bake_types={'OBJECT'}, # Bake only OBJECT animation (no pose bones) channel_types={'BBONE', 'LOCATION', 'PROPS', 'ROTATION', 'SCALE'} ) else: # Use the frame_start and frame_end from the scene bpy.ops.nla.bake( frame_start=frame_start, frame_end=frame_end, step=1, only_selected=True, visual_keying=True, clear_constraints=True, clear_parents=False, use_current_action=False, clean_curves=False, bake_types={'OBJECT'}, # Bake only OBJECT animation (no pose bones) ) bpy.ops.object.select_all(action='DESELECT') # Function to delete non visible geometry via camera array and raycasting def delete_non_visible_geometry(): previous_resolution_x = None previous_resolution_y = None previous_camera_type = None def create_vertex_groups(): # Create a new vertex group with the same name as the object vertex_group = obj.vertex_groups.new(name=obj.name) # Access the bmesh data of the object bm = bmesh.new() bm.from_mesh(obj.data) # Add all vertices to the vertex group for v in bm.verts: vertex_group.add([v.index], 1.0, 'ADD') # Clean up the bmesh bm.free() def delete_vertex_groups(): # Collect the vertex groups that match the object's name groups_to_delete = [vg for vg in obj.vertex_groups if vg.name == obj.name] # Remove the collected vertex groups for group in groups_to_delete: obj.vertex_groups.remove(group) def set_origin_to_cursor(): # Select all objects with linked data bpy.ops.object.select_linked(type='OBDATA') # Set the origin of all selected objects to the 3D cursor bpy.ops.object.origin_set(type='ORIGIN_CURSOR', center='MEDIAN') # Optionally, you can deselect all objects after the operation bpy.ops.object.select_all(action='DESELECT') def set_origin_to_geometry(): # Select all objects with linked data bpy.ops.object.select_linked(type='OBDATA') # Set the origin of all selected objects to the 3D cursor bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY', center='MEDIAN') # Optionally, you can deselect all objects after the operation bpy.ops.object.select_all(action='DESELECT') def get_or_create_collection(name): """Helper function to get or create a collection.""" collection = bpy.data.collections.get(name) if collection is None: collection = bpy.data.collections.new(name) bpy.context.scene.collection.children.link(collection) return collection def store_current_data(): nonlocal previous_resolution_x, previous_resolution_y, previous_camera_type scene = bpy.context.scene # Get active camera data if scene.camera: camera = scene.camera.data # Store camera type if camera exists previous_camera_type = camera.type # Store resolution previous_resolution_x = scene.render.resolution_x previous_resolution_y = scene.render.resolution_y def set_scene_resolution(): scene = bpy.context.scene # Set new render resolution to 800x800 scene.render.resolution_x = 800 scene.render.resolution_y = 800 def restore_previous_data(): nonlocal previous_resolution_x, previous_resolution_y, previous_camera_type # Check if previous resolution data exists if previous_resolution_x is not None and previous_resolution_y is not None: scene = bpy.context.scene # Restore the resolution scene.render.resolution_x = previous_resolution_x scene.render.resolution_y = previous_resolution_y # Check if previous camera type data exists if previous_camera_type is not None: # Get active camera data if scene.camera: camera = scene.camera.data camera.type = previous_camera_type def create_camera(name, location, obj_bbox_size, scale_multiplier): """Create and return a camera with orthographic scale set dynamically and adjusted by a multiplier.""" # Create the camera object camera_data = bpy.data.cameras.new(name) camera = bpy.data.objects.new(name, camera_data) # Set the camera's location camera.location = location # Set the camera to orthographic mode camera.data.type = 'ORTHO' # Compute the base orthographic scale based on the object's bounding box size base_ortho_scale = obj_bbox_size * 0.75 # Base scale adjustment # Apply the multiplier to the base orthographic scale camera.data.ortho_scale = base_ortho_scale * scale_multiplier return camera def create_cameras_around_object(num_cameras=8, scale=15, collection_name="BOPCameraArrayCollection", scale_multiplier=2.0, close_camera_factor=0.5): # Check if there is an active object if bpy.context.active_object is None: raise ValueError("No active object found. Please select an object.") # Get the active object obj = bpy.context.active_object # Ensure the object is a MESH if obj.type != 'MESH': raise ValueError("Active object is not a MESH. Please select a MESH object.") # Get the object's bounding box dimensions in world space bbox_corners = [obj.matrix_world @ mathutils.Vector(corner) for corner in obj.bound_box] bbox_min = mathutils.Vector((min([v.x for v in bbox_corners]), min([v.y for v in bbox_corners]), min([v.z for v in bbox_corners]))) bbox_max = mathutils.Vector((max([v.x for v in bbox_corners]), max([v.y for v in bbox_corners]), max([v.z for v in bbox_corners]))) # Calculate the center of the bounding box in world coordinates bbox_center = (bbox_max + bbox_min) / 2 # Calculate the bounding box size (diagonal length) bbox_size = (bbox_max - bbox_min).length # Set the radius as a function of the bounding box size radius = bbox_size * 1.5 # Adjust multiplier to control how far cameras are placed # # Remove existing cameras # cameras = [obj for obj in bpy.data.objects if obj.type == 'CAMERA'] # for cam in cameras: # bpy.data.objects.remove(cam, do_unlink=True) # Create a collection for the cameras camera_collection = get_or_create_collection(collection_name) ### Create the 90-degree cameras (horizontal ring, no change in Z) for i in range(num_cameras): angle = (2 * math.pi / num_cameras) * i # Distribute cameras evenly in a circle x = math.cos(angle) * radius y = math.sin(angle) * radius z = bbox_center.z # Use the Z of the bounding box center in world coordinates # Convert to world coordinates relative to the bounding box center camera_location = bbox_center + mathutils.Vector((x, y, 0)) # Z stays at bbox_center.z # Create and position the camera with dynamic orthographic scale and multiplier camera_name = f"Camera_90deg_{i+1}" camera = create_camera(camera_name, camera_location, bbox_size, scale_multiplier) vertex_group_name = obj.name # Use 'Track To' constraint to make the camera always look at the object track_to = camera.constraints.new(type='TRACK_TO') track_to.target = obj track_to.subtarget = vertex_group_name track_to.track_axis = 'TRACK_NEGATIVE_Z' # Track along negative Z axis track_to.up_axis = 'UP_Y' # Keep Y axis as the up direction # Link the camera to the collection camera_collection.objects.link(camera) ### Create the 60-degree cameras (closer to the object and slightly elevated) elevation_scaling_factor = 0.75 # Factor to lower the elevation of 60-degree cameras closer_radius = radius * close_camera_factor # Bring the 60-degree cameras closer for i in range(num_cameras): angle = (2 * math.pi / num_cameras) * i # Distribute cameras evenly in a circle x = math.cos(angle) * closer_radius # Use the closer radius y = math.sin(angle) * closer_radius # Use the closer radius z = bbox_center.z + closer_radius * math.tan(math.radians(50)) * elevation_scaling_factor # Reduced Z elevation # Convert to world coordinates relative to the bounding box center camera_location = bbox_center + mathutils.Vector((x, y, z - bbox_center.z)) # Adjust Z based on bbox_center # Create and position the camera with dynamic orthographic scale and multiplier camera_name = f"Camera_60deg_{i+1}" camera = create_camera(camera_name, camera_location, bbox_size, scale_multiplier) vertex_group_name = obj.name # Use 'Track To' constraint to make the camera always look at the object track_to = camera.constraints.new(type='TRACK_TO') track_to.target = obj track_to.subtarget = vertex_group_name track_to.track_axis = 'TRACK_NEGATIVE_Z' # Track along negative Z axis track_to.up_axis = 'UP_Y' # Keep Y axis as the up direction # Link the camera to the collection camera_collection.objects.link(camera) ### Create and position the top-down camera (no angle here, directly above) top_down_camera_name = "Camera_TopDown" top_down_camera_location = bbox_center + mathutils.Vector((0, 0, radius)) # Set height based on radius top_down_camera = create_camera(top_down_camera_name, top_down_camera_location, bbox_size, scale_multiplier) vertex_group_name = obj.name # Set the top-down camera to look directly at the object using the 'Track To' constraint track_to_top_down = top_down_camera.constraints.new(type='TRACK_TO') track_to_top_down.target = obj track_to_top_down.subtarget = vertex_group_name track_to_top_down.track_axis = 'TRACK_NEGATIVE_Z' track_to_top_down.up_axis = 'UP_Y' # Link the top-down camera to the collection camera_collection.objects.link(top_down_camera) # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Reselect and activate the original object obj.select_set(True) bpy.context.view_layer.objects.active = obj def get_camera_rays(scene, obj): """Generate rays from all cameras to multiple points on each face of the object with adaptive sampling.""" matrix_world = obj.matrix_world rays = [] bm = bmesh.new() bm.from_mesh(obj.data) bm.faces.ensure_lookup_table() # Directly access the 'BOPCameraArrayCollection' camera_collection = scene.collection.children['BOPCameraArrayCollection'] # Get all cameras in the collection cameras = [obj for obj in camera_collection.objects if obj.type == 'CAMERA'] # Define thresholds for small, medium, and large face areas small_face_threshold = 0.5 # Faces smaller than this will be considered "small" large_face_threshold = 1.0 # Faces larger than this will be considered "large" for face in bm.faces: # Calculate face center in world space face_center = matrix_world @ face.calc_center_median() face_verts = [matrix_world @ vert.co for vert in face.verts] # Calculate face area face_area = face.calc_area() # Initialize points to cast rays at (start with the center) face_points = [face_center] # Adaptive sampling based on face area if face_area > large_face_threshold: # Large face: Use center, vertices, and midpoints midpoints = [] for i in range(len(face_verts)): next_i = (i + 1) % len(face_verts) midpoint = (face_verts[i] + face_verts[next_i]) / 2 midpoints.append(midpoint) face_points += face_verts + midpoints # Add vertices and midpoints elif face_area > small_face_threshold: # Medium face: Use center and vertices face_points += face_verts # Add vertices # Small faces only use the center (already included) for camera in cameras: for point in face_points: # Calculate the ray direction from the camera to the face point ray_direction = (point - camera.location).normalized() rays.append((camera.location, ray_direction, face.index)) bm.free() return rays def batch_ray_casts(rays, bvh_tree, obj_matrix_world): """Process ray casts using BVH tree.""" visible_faces = set() for ray_origin, ray_direction, face_index in rays: # Transform ray origin and direction to local space of the object local_ray_origin = obj_matrix_world.inverted() @ ray_origin local_ray_direction = obj_matrix_world.inverted().to_3x3() @ ray_direction # Cast ray in the BVH tree hit_location, hit_normal, hit_index, hit_distance = bvh_tree.ray_cast(local_ray_origin, local_ray_direction) if hit_index == face_index: visible_faces.add(face_index) return visible_faces def delete_non_visible_geometry(obj, threshold=0.9, collection_name="BOPCameraArrayCollection"): """Delete geometry that is not visible from any camera using BVH tree.""" # Switch to object mode bpy.ops.object.mode_set(mode='OBJECT') # Start BMesh processing bm = bmesh.new() bm.from_mesh(obj.data) bm.faces.ensure_lookup_table() # Build BVH Tree bvh = bvhtree.BVHTree.FromBMesh(bm) # Generate camera rays for all cameras camera_rays = get_camera_rays(bpy.context.scene, obj) # Check visibility of faces obj_matrix_world = obj.matrix_world visible_faces = batch_ray_casts(camera_rays, bvh, obj_matrix_world) faces_to_delete = [] for face in bm.faces: if face.index not in visible_faces: faces_to_delete.append(face) # Apply threshold to faces_to_delete # num_faces_to_delete = int(len(faces_to_delete) * threshold) # faces_to_delete = faces_to_delete[:num_faces_to_delete] # Delete faces if faces_to_delete: bmesh.ops.delete(bm, geom=faces_to_delete, context='FACES') bm.to_mesh(obj.data) bm.free() if collection_name in bpy.data.collections: # Get the collection collection = bpy.data.collections[collection_name] # Iterate through the objects in the collection for obj in collection.objects: # Check if the object is a camera if obj.type == 'CAMERA': # Delete the camera bpy.data.objects.remove(obj, do_unlink=True) # After all cameras are deleted, remove the collection bpy.data.collections.remove(collection) store_current_data() set_scene_resolution() # This will keep track of processed multi user object (Its important to keep this, otherwise blender will most likely crash) processed_meshes = set() selected_objects = bpy.context.selected_objects if selected_objects: # Iterate through objects in the scene for obj in selected_objects: # Check if the object is a mesh and if it's visible in the scene if obj.type == 'MESH' and not obj.modifiers and obj.visible_get(): # Check if this mesh data has already been processed if obj.data in processed_meshes: # print(f"Skipped {obj.name}: Already processed another object with the same mesh data") continue # Check the number of polygons (faces) num_faces = len(obj.data.polygons) if num_faces > 4: # Mark this mesh data as processed processed_meshes.add(obj.data) # Set the object as active bpy.context.view_layer.objects.active = obj # Perform operations create_vertex_groups() # set_origin_to_geometry() create_cameras_around_object() # set_origin_to_cursor() delete_non_visible_geometry(obj) # set_origin_to_geometry() delete_vertex_groups() obj.data.update() # Deselect the object obj.select_set(False) # else: # print(f"Skipped {obj.name}: Only {num_faces} polygons") # else: # print(f"Skipped {obj.name}: Not a visible mesh object") else: # Iterate through objects in the scene for obj in bpy.context.scene.objects: # Check if the object is a mesh and if it's visible in the scene if obj.type == 'MESH' and not obj.modifiers and obj.visible_get(): # Check if this mesh data has already been processed if obj.data in processed_meshes: # print(f"Skipped {obj.name}: Already processed another object with the same mesh data") continue # Check the number of polygons (faces) num_faces = len(obj.data.polygons) if num_faces > 3500: # Mark this mesh data as processed processed_meshes.add(obj.data) # Set the object as active bpy.context.view_layer.objects.active = obj # Perform operations create_vertex_groups() # set_origin_to_geometry() create_cameras_around_object() # set_origin_to_cursor() delete_non_visible_geometry(obj) # set_origin_to_geometry() delete_vertex_groups() obj.data.update() # Deselect the object obj.select_set(False) # else: # print(f"Skipped {obj.name}: Only {num_faces} polygons") # else: # print(f"Skipped {obj.name}: Not a visible mesh object") restore_previous_data() #----------------------------Operators----------------------------# # Operator to execute check updates class CheckForUpdateOperator(bpy.types.Operator): """Check updates""" bl_idname = "wm.check_for_update" bl_label = "BOP - Check for Update" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): check_for_update() print("Operator: check_for_update() executed") return {'FINISHED'} # Operator to execute the delete downward faces function class OBJECT_OT_delete_downward_faces(bpy.types.Operator): """It will delete all mesh faces which normals are facing down (Z-). Function also ignore objects with active modifiers. (If wrong faces are deleted, try to use Reset Scale to 1 function at first""" bl_idname = "object.delete_downward_faces" bl_label = "BOP - Delete Downward Faces" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_downward_faces() print("Operator: delete_downward_faces(threshold) executed") return {'FINISHED'} # Operator to convert from tris to quads class OBJECT_OT_tris_to_quads(bpy.types.Operator): """It will convert Triangulated mesh into the Quads for a Unity""" bl_idname = "object.tris_to_quads" bl_label = "BOP - Tris to Quads" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): tris_to_quads() print("Operator: delete_downward_faces(threshold) executed") return {'FINISHED'} # Operator to execute the auto-smooth function class OBJECT_OT_auto_smooth_all(bpy.types.Operator): """It will smooth out mesh normals to default value. Usually it’s 30°""" bl_idname = "object.auto_smooth_all" bl_label = "BOP - Auto Smooth All" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): apply_auto_smooth() print("Operator: apply_auto_smooth() executed") return {'FINISHED'} # Operator to delete custom split normals data class OBJECT_OT_delete_custom_normals(bpy.types.Operator): """Custom Split Normals are a way to tweak/fake shading by pointing them towards other directions than default, auto-computed ones. By deleting them 3D model switch to default one provided by Blender""" bl_idname = "object.delete_custom_normals" bl_label = "BOP - Delete Custom Normals" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_custom_split_normals() print("Operator: delete_custom_split_normals() executed") return {'FINISHED'} # Operator to execute the merge close vertices function class OBJECT_OT_merge_close_vertices(bpy.types.Operator): """It will merge connected or unconnected vertices which positions are exact or very close (Threshold is 0.001)""" bl_idname = "object.merge_close_vertices" bl_label = "BOP - Merge Close Vertices" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): merge_close_vertices() print("Operator: merge_close_vertices() executed") return {'FINISHED'} # Operator to execute limited dissolve function class OBJECT_OT_limited_dissolve(bpy.types.Operator): """It will simplify your mesh by dissolving vertices and edges separating flat regions. Reduces detail on planar faces and linear edges""" bl_idname = "object.limited_dissolve" bl_label = "BOP - Limited Dissolve" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): limited_dissolve() print("Operator: limited_dissolve() executed") return {'FINISHED'} # Operator to remove unused datablocks class OBJECT_OT_remove_unused_datablocks(bpy.types.Operator): """It will make sure that all non necessary or unused data are deleted so exported model will be clean (Applies on whole scene)""" bl_idname = "object.remove_unused_datablocks" bl_label = "BOP - Remove Unused Datablocks" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): remove_unused_datablocks() print("Operator: remove_unused_datablocks() executed") return {'FINISHED'} # Operator to reparent mesh for Unity class OBJECT_OT_reparent(bpy.types.Operator): """It will put all objects inside scene under the empty parent object in order to function properly in Unity (Applies on all objects) Also it will keep hierarchy for multi-storey buildings""" bl_idname = "object.reparent" bl_label = "BOP - Reparent" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): reparent(['Outside', 'Interior', 'Architecture', 'Floor']) print("Operator: reparent(['Outside', 'Interior', 'Architecture', 'Floor']) executed") return {'FINISHED'} # Operator to recalculate normals class OBJECT_OT_recalculate_normals(bpy.types.Operator): """It will recalculate the normals of selected faces so that they point outside the volume that the face belongs to""" bl_idname = "object.recalculate_normals" bl_label = "BOP - Recalculate Mesh Normals" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): recalculate_normals() print("Operator: recalculate_normals() executed") return {'FINISHED'} # Operator to execute the open documentation URL function class OBJECT_OT_open_url_operator(bpy.types.Operator): """It will open addon documentations in default web browser""" bl_idname = "object.open_url_operator" bl_label = "BOP - Open URL" bl_options = {'REGISTER', 'UNDO'} url: bpy.props.StringProperty(name="URL", default="https://twinzo.atlassian.net/wiki/spaces/PUBD/pages/104398849/Blender+Optimization+package+BOP") def execute(self, context): open_url(self.url) print("Operator: open_url(self.url) executed") return {'FINISHED'} # Operator to execute the rotate model function class OBJECT_OT_rotate_model_to_highest_normal_direction(bpy.types.Operator): """It will rotate 3D model in such a way, that side with higher average number of normals facing corresponding direction will be rotated to facing up (Z+). Function is intended to use on Scanned meshes!""" bl_idname = "object.rotate_model_operator" bl_label = "BOP - Rotate Scan" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): rotate_model_operator() print("Operator: rotate_model_operator() executed") return {'FINISHED'} # Operator to execute the reset origin function class OBJECT_OT_reset_origin(bpy.types.Operator): """It will set objects origin point to their relative median position""" bl_idname = "object.reset_origin" bl_label = "BOP - Reset Origin" bl_options = set() def execute(self, context): reset_origin() bpy.ops.ed.undo_push(message='BOP - Reset Origin') print("Operator: reset_origin() executed") return {'FINISHED'} # Operator to execute Reset scene to original state class OBJECT_OT_reset_to_original_state(bpy.types.Operator): """It will reset everythig to the original state, USE IT WITH CAUTION!""" bl_idname = "object.reset" bl_label = "BOP - Reset to original state" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): reset_to_original_state() print("Operator: reset_to_original_state() executed") return {'FINISHED'} # Operator to execute remove duplicated materials function class OBJECT_OT_merge_similar_materials(bpy.types.Operator): """It will merge similar materials based on their values and colors (Applies on all objects in project)""" bl_idname = "object.merge_similar_materials" bl_label = "BOP - Remove duplicated materials" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): merge_similar_materials() print("Operator: merge_similar_materials() executed") return {'FINISHED'} # Operator to execute create instanced objects function class OBJECT_OT_create_instances(bpy.types.Operator): """It will make identic or very similar objects as a instanced one. It is best to use it after Reset Origin function. Applies on all objects in scene!""" bl_label = "BOP - Create Instances" bl_idname = "object.create_instances" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): try: # Get all visible mesh objects in the scene visible_mesh_objects = [ obj for obj in bpy.context.scene.objects if obj.type == 'MESH' and obj.visible_get() ] if not visible_mesh_objects: print("No visible mesh objects in the scene.") return {'CANCELLED'} # Deselect all objects bpy.ops.object.select_all(action='DESELECT') # Select all visible mesh objects for obj in visible_mesh_objects: obj.select_set(True) # Make one random mesh object active active_object = random.choice(visible_mesh_objects) bpy.context.view_layer.objects.active = active_object bpy.ops.object.select_all(action='DESELECT') bpy.ops.object.select_all(action='SELECT') bpy.ops.object.mode_set(mode='OBJECT') # obj = bpy.context.view_layer.objects.active # if not obj or obj.type != 'MESH': # mesh_objects = [obj for obj in bpy.context.visible_objects if obj.type == 'MESH' and len(obj.data.vertices) != 0] # if mesh_objects: # obj = mesh_objects[0] # bpy.context.view_layer.objects.active = obj # else: # return {'CANCELLED'} bpy.ops.object.mode_set(mode='EDIT') bpy.ops.mesh.select_all(action='SELECT') bm = bmesh.from_edit_mesh(obj.data) vertices_sorted = sorted(bm.verts, key=lambda v: v.co.z) vertex_mapping = {old_index: new_index for new_index, old_index in enumerate(v.index for v in vertices_sorted)} bpy.ops.mesh.select_all(action='DESELECT') for v in bm.verts: v.index = vertex_mapping[v.index] bmesh.update_edit_mesh(obj.data) bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.select_all(action='DESELECT') threshold_distance = 0.03 object_data = [obj for obj in bpy.context.visible_objects if obj.type == 'MESH' and len(obj.data.vertices) != 0] object_groups = {} for obj1 in object_data: added_to_group = False for group, group_objects in object_groups.items(): obj2 = group_objects[0] if compare_objects(obj1, obj2, threshold_distance): group_objects.append(obj1) added_to_group = True break if not added_to_group: object_groups[obj1] = [obj1] for group_objects in object_groups.values(): if len(group_objects) > 1: original_obj = group_objects[0] for obj in group_objects[1:]: obj.data = original_obj.data except Exception as e: print("Object instancer failed:", str(e)) print("Continuing") print("Operator: object.create_instances() executed") return {'FINISHED'} # Operator to execute object separator function class OBJECT_OT_object_separator(bpy.types.Operator): """It will separate objects by loose parts (Its best to use it before Object Instancer and script will also ignore Instanced meshes!)""" bl_idname = "object.object_separator" bl_label = "BOP - Object Separator" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): object_separator() print("Operator: object_separator() executed") return {'FINISHED'} # Operator to execute delete bloat function class OBJECT_OT_delete_bloat(bpy.types.Operator): """It will delete unnecessary types of objects like (Empty, Lights, Objects without mesh data). (Its best to use it before Object Instancer and Reparent function!)""" bl_idname = "object.delete_bloat" bl_label = "BOP - Delete bloat" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_bloat() print("Operator: delete_bloat() executed") return {'FINISHED'} # Operator to execute reset scale of objects to 1 function class OBJECT_OT_reset_scale_to_1(bpy.types.Operator): """It will reset object scale values in scene to 1 (Ignores empty and instances)""" bl_idname = "object.reset_scale_to_1" bl_label = "BOP - Reset scale to 1" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): reset_scale_to_1() print("Operator: reset_scale_to_1() executed") return {'FINISHED'} # Operator to execute delete armatures and bones function class OBJECT_OT_delete_armatures_and_bones(bpy.types.Operator): """It will delete visible armature or bones in scene (Nested objects will be preserved)""" bl_idname = "object.delete_armatures_and_bones" bl_label = "BOP - Delete armature or bones" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_armatures_and_bones() print("Operator: delete_armatures_and_bones() executed") return {'FINISHED'} # Operator to execute delete animations function class OBJECT_OT_delete_animations(bpy.types.Operator): """It will delete animations on visible objects in scene""" bl_idname = "object.delete_animations" bl_label = "BOP - Delete animations in scene" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_animations() print("Operator: delete_animations() executed") return {'FINISHED'} # Operator to execute delete animations function class OBJECT_OT_lod_rename(bpy.types.Operator): """It will adjust LOD names in order to work with Unity""" bl_idname = "object.lod_rename" bl_label = "BOP - LOD_Rename" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): lod_rename() print("Operator: lod_rename() executed") return {'FINISHED'} # Operator to execute delete duplicated meshes function class OBJECT_OT_delete_duplicated_meshes(bpy.types.Operator): """It will delete duplicated meshes which are roughly at the same location and same shape""" bl_idname = "object.delete_duplicated_meshes" bl_label = "BOP - Delete_Duplicated_meshes" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): select_and_delete_duplicates() print("Operator:select_and_delete_duplicates() executed") return {'FINISHED'} # Operator to execute export model function class OBJECT_OT_export_model(bpy.types.Operator): """It will export selected models as FBX file""" bl_idname = "object.export_model" bl_label = "BOP - Export model" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): try: # Check if the specified export path is valid export_path = context.scene.export_path # Add ".fbx" suffix if not already present if not export_path.lower().endswith(".fbx"): export_path += "BOP.fbx" if not os.path.isabs(export_path): self.report({'ERROR'}, "Invalid export path") return {'CANCELLED'} # Check if there are already selected objects selected_objects = bpy.context.selected_objects if selected_objects: # Export FBX with custom settings using the specified export path bpy.ops.export_scene.fbx( filepath=export_path, use_selection=True, global_scale=1.0, apply_unit_scale=True, apply_scale_options='FBX_SCALE_UNITS', bake_space_transform=False, use_space_transform=True, colors_type='SRGB', object_types={'MESH', 'EMPTY', 'ARMATURE', 'OTHER'}, use_mesh_modifiers=True, mesh_smooth_type='OFF', use_mesh_edges=False, use_tspace=False, use_custom_props=False, add_leaf_bones=False, bake_anim=True, bake_anim_use_all_actions=False, bake_anim_use_nla_strips=False, bake_anim_use_all_bones=True, bake_anim_force_startend_keying=True, bake_anim_step=1, bake_anim_simplify_factor=1, path_mode='COPY', embed_textures=True ) # Clear selection after export bpy.ops.object.select_all(action='DESELECT') else: if not selected_objects: # If no objects are selected, select all visible meshes in the scene for obj in bpy.context.visible_objects: if obj.type == 'MESH': obj.select_set(True) # Export FBX with custom settings using the specified export path bpy.ops.export_scene.fbx( filepath=export_path, use_selection=True, global_scale=1.0, apply_unit_scale=True, apply_scale_options='FBX_SCALE_UNITS', bake_space_transform=False, use_space_transform=True, colors_type='SRGB', object_types={'MESH', 'EMPTY', 'ARMATURE', 'OTHER'}, use_mesh_modifiers=True, mesh_smooth_type='OFF', use_mesh_edges=False, use_tspace=False, use_custom_props=False, add_leaf_bones=False, bake_anim=True, bake_anim_use_all_actions=False, bake_anim_use_nla_strips=False, bake_anim_use_all_bones=True, bake_anim_force_startend_keying=True, bake_anim_step=1, bake_anim_simplify_factor=1, path_mode='COPY', embed_textures=True ) # Clear selection after export bpy.ops.object.select_all(action='DESELECT') except Exception as e: print("Exporting failed:", str(e)) print("Continuing") print("Operator: object.export_model() executed") return {'FINISHED'} # Operator to execute Analyze classical scene function class SCENE_OT_classical_calculate_statistics(bpy.types.Operator): """It will analyze scene and print collected statistics bellow""" bl_idname = "classical_scene.calculate_statistics" bl_label = "BOP - Calculate Statistics" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): # Get the active scene scene = bpy.context.scene # Clear the existing unsuitability messages context.scene.statistics.unsuitability_messages.clear() # Initialize variables to count polygons, objects, and the farthest object poly_count = 0 object_count = 0 material_count = len(bpy.data.materials) closest_object_name = "" closest_distance = float('inf') total_scale = 0 shared_geometry_count = 0 has_bones_or_armatures = False has_animation = False # Initialize variables for additional checks empty_parents_count = 0 camera_count = 0 light_count = 0 image_count = 0 curve_count = 0 texture_count = 0 materials_mapped = False # Initialize as False # Initialize variables to store total dimensions exceeded_x = None exceeded_y = None has_mesh_objects = False lod_detected = False lod_level = None # Multi story building variables building_hierarchy = False multi_storey_building = False floor_count = 0 floor_names = [] osm_detected = { "OSM Detected": False, ".osm Collection Detected": False, "way_profiles Collection Detected": False, } # dimension size by pressing N and click on Item panel to show Transform values threshold = 5000 # Create a dictionary to track unique mesh data names and their counts unique_mesh_data = {} # Create a list of possible suffixes from ".001" to ".100" possible_suffixes = [f".{str(i).zfill(3)}" for i in range(1, 101)] # Initialize a set to store material names material_names = set() # Iterate through all objects in the scene for obj in scene.objects: if obj.type == 'MESH': has_mesh_objects = True # If the object is a mesh, add its polygon count to the total poly_count += len(obj.data.polygons) # Check if this mesh data is already encountered mesh_data_name = obj.data.name if mesh_data_name in unique_mesh_data: # This is a shared geometry object shared_geometry_count += 1 else: # This is a unique geometry object, add it to the dictionary unique_mesh_data[mesh_data_name] = True # Increment the object count object_count += 1 # Check if the object has materials assigned if obj.material_slots: materials_mapped = True # Set to True if any material is assigned to any object # Calculate the distance from the world origin (0,0,0) in meters distance_meters = obj.location.length # Blender Units, which are assumed to be in meters # Check if this object is farther than the previously farthest if distance_meters < closest_distance: closest_distance = distance_meters closest_object_name = obj.name # Calculate the scale for this object scale = obj.scale total_scale += (scale.x + scale.y + scale.z) / 3 # Average the scale components # Additional checks for empty parents, cameras, lights, loose geometry, and small Z-height if obj.type == 'EMPTY': #and not obj.children: empty_parents_count += 1 elif obj.type == 'CAMERA': camera_count += 1 elif obj.type == 'LIGHT': light_count += 1 elif obj.type == 'IMAGE': image_count += 1 elif obj.type == 'CURVE': curve_count += 1 # Check if the object is of type 'ARMATURE' or if it has bones if obj.type == 'ARMATURE' or (obj.type == 'MESH' and obj.find_armature()): has_bones_or_armatures = True # Check for animation in the scene if scene.animation_data: has_animation = True if obj.type == 'MESH': # Get the dimensions of the object x, y, _ = obj.dimensions # Discard the Z dimension # Check if the X or Y dimension exceeds the threshold if x > threshold: exceeded_x = (obj.name, x) if y > threshold: exceeded_y = (obj.name, y) # Check if there is a LOD object and which level (Checking by name) for i in range(10): lod_name = "LOD{}".format(i) if lod_name in obj.name: lod_detected = True # Split the name to extract the LOD number lod_number_str = obj.name.split("LOD")[1] try: # Try to convert the extracted portion to an integer lod_number = int(lod_number_str) # Compare the LOD level if lod_level is None or lod_number > lod_level: lod_level = lod_number except ValueError: # Ignore cases where the extracted portion is not a valid integer continue # Check for the ".osm_building" or ".osm_roads" condition among selected objects if any(".osm_building" in obj.name or ".osm_roads" in obj.name for obj in bpy.context.scene.objects): osm_detected["OSM Detected"] = True # Iterate through all materials in the scene and count texture nodes with images (Better to keep outside first loop) for material in bpy.data.materials: if material.use_nodes and material.node_tree: for node in material.node_tree.nodes: if isinstance(node, bpy.types.ShaderNodeTexImage) and node.image is not None: texture_count += 1 # Iterate through all collections in the scene to check for partial collection names for collection in bpy.context.scene.collection.children: if ".osm" in collection.name: osm_detected[".osm Collection Detected"] = True if "way_profiles" in collection.name: osm_detected["way_profiles Collection Detected"] = True # Determine if any of the conditions were met and store True or False in osm_detected osm_detected = any(value for value in osm_detected.values()) # Calculate the average scale only if there are mesh objects average_scale = total_scale / object_count if has_mesh_objects else 0 # Calculate the percentage of objects with shared geometry percentage_shared = (shared_geometry_count / object_count) * 100 if has_mesh_objects else 0 # Function to recursively search for the empty parent hierarchy def search_empty_hierarchy(obj): nonlocal floor_count, floor_names, building_hierarchy, multi_storey_building # Check if the object is an empty or a mesh if obj.type == 'EMPTY' or obj.type == 'MESH': # Check if the empty's name matches the hierarchy structure if obj.name.startswith("Main") and len(obj.children) > 0: # Check if there's at least one child object starting with "Floor" floor_children = [child for child in obj.children if child.name.startswith("Floor")] if floor_children: # Increment floor count floor_count += len(floor_children) # Record floor names floor_names.extend([child.name for child in floor_children]) # Check if there is more than one floor child if len(floor_children) > 1: multi_storey_building = True # Check if at least one of the required children is present for any floor required_children = ["Architecture", "Interior", "Outside"] for floor_child in floor_children: children_names = [child.name for child in floor_child.children] if any(required_child in child_name for required_child in required_children for child_name in children_names): # Hierarchy found, set global variable to True building_hierarchy = True return # Recursively search through children for child in obj.children: search_empty_hierarchy(child) # Start search from the root of the scene's objects for obj in bpy.data.objects: search_empty_hierarchy(obj) if building_hierarchy: break # Set the calculated values to the properties context.scene.statistics.poly_count = poly_count context.scene.statistics.object_count = object_count context.scene.statistics.material_count = material_count context.scene.statistics.closest_object_name = closest_object_name context.scene.statistics.closest_distance = closest_distance context.scene.statistics.total_scale = total_scale context.scene.statistics.shared_geometry_count = shared_geometry_count context.scene.statistics.has_bones_or_armatures = has_bones_or_armatures context.scene.statistics.has_animation = has_animation context.scene.statistics.empty_parents_count = empty_parents_count context.scene.statistics.camera_count = camera_count context.scene.statistics.light_count = light_count context.scene.statistics.image_count = image_count context.scene.statistics.curve_count = curve_count context.scene.statistics.texture_count = texture_count context.scene.statistics.exceeded_x = exceeded_x context.scene.statistics.exceeded_y = exceeded_y context.scene.statistics.average_scale = average_scale context.scene.statistics.percentage_shared = percentage_shared context.scene.statistics.building_hierarchy = building_hierarchy context.scene.statistics.multi_storey_building = multi_storey_building context.scene.statistics.floor_count = floor_count # Initialize a list to store unsuitability messages unsuitability_messages = [] # Check suitability conditions and append appropriate messages. Only these condition described bellow can result into reject model if poly_count > 1250000: unsuitability_messages.append("Too many polygons ({} polygons, should be less or equal to 1250000 (1.25M)).".format(poly_count)) #if poly_count < 100: # unsuitability_messages.append("There are ({} polygons), model is considered as a bloat.".format(poly_count)) if object_count > 6000: unsuitability_messages.append("Too many objects ({} objects, should be less or equal to 6000).".format(object_count)) #if object_count < 1: # unsuitability_messages.append("({} mesh objects detected), model is considered as a bloat.".format(object_count)) if closest_distance > 5000: unsuitability_messages.append("Farthest object is too far from the origin ({} meters, should be less or equal to 5000 meters).".format(farthest_distance)) if average_scale < 0.01: unsuitability_messages.append("Average scale is too small ({:.2f}, should be bigger or equal to 0.1).".format(average_scale)) if material_count > 200: unsuitability_messages.append("Too many materials ({} materials), should be less or equal 200.".format(len(bpy.data.materials))) if not materials_mapped: unsuitability_messages.append("Materials are not mapped to any objects in the scene.") if exceeded_x or exceeded_y: unsuitability_messages.append("One of your models exceeded total limit size (5KM). Please check your conversion units or make sure that your model is exported in meters") if percentage_shared < 40: unsuitability_messages.append("There are less than 40% objects with shared geometry") if (camera_count > 0 or light_count > 0 or curve_count > 0): unsuitability_messages.append("Please delete unnecessary objects like camera, light or curve") if empty_parents_count > 100: unsuitability_messages.append("Please minimize amount of empty objects in scene") if has_bones_or_armatures == True: unsuitability_messages.append("Please delete any armatures or bones from scene") if has_animation == True: unsuitability_messages.append("Animations are not currently supported") if building_hierarchy == False: unsuitability_messages.append("Your model is missing required Hierarchy. Please use Reparent function or check Documentation") for message in unsuitability_messages: new_item = context.scene.statistics.unsuitability_messages.add() new_item.message = message print("Operator: scene.calculate_statistics() executed") return {'FINISHED'} # Operator to execute Analyze scanned scene function class SCENE_OT_scanned_calculate_statistics(bpy.types.Operator): """It will analyze scene and print collected statistics bellow""" bl_idname = "scanned_scene.calculate_statistics" bl_label = "BOP - Calculate Statistics" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): # Get the active scene scene = bpy.context.scene # Clear the existing unsuitability messages context.scene.statistics.unsuitability_messages.clear() # Initialize variables to count polygons, objects, materials, textures, and the farthest object poly_count = 0 object_count = 0 material_count = len(bpy.data.materials) texture_count = 0 closest_object_name = "" closest_distance = float('inf') total_scale = 0 # Initialize variables for additional checks empty_parents_count = 0 camera_count = 0 light_count = 0 image_count = 0 small_z_height_count = 0 materials_with_missing_textures = [] materials_mapped = False # Initialize as False # Initialize variables to store treshold exceeded dimensions exceeded_x = None exceeded_y = None has_mesh_objects = False lod_detected = False lod_level = None # Multi story building variables building_hierarchy = False multi_storey_building = False floor_count = 0 floor_names = [] osm_detected = { "OSM Detected": False, ".osm Collection Detected": False, "way_profiles Collection Detected": False, } # Specify the dimension threshold in Blender units (assuming Blender's default unit is meters), You can verify # dimension size by pressing N and click on Item panel to show Transform values threshold = 5000 # Create a dictionary to track unique mesh data names and their counts unique_mesh_data = {} # Initialize a set to store material names material_names = set() # Iterate through all objects in the scene for obj in scene.objects: if obj.type == 'MESH': has_mesh_objects = True # If the object is a mesh, add its polygon count to the total poly_count += len(obj.data.polygons) # Increment the object count object_count += 1 # Check if the object has materials assigned if obj.material_slots: materials_mapped = True # Set to True if any material is assigned to any object # Check if this object is farther than the previously farthest if obj.type == 'MESH': # Calculate the global position of the object global_pos = obj.matrix_world.translation # Calculate the distance from the world origin (0,0,0) in meters distance_meters = global_pos.length # Blender Units, which are assumed to be in meters # Check if this object is closer than the previously closest if distance_meters < closest_distance: closest_distance = distance_meters closest_object_name = obj.name # Calculate the scale for this object scale = obj.scale total_scale += (scale.x + scale.y + scale.z) / 3 # Average the scale components # Additional checks for empty parents, cameras, lights, loose geometry, and small Z-height if obj.type == 'EMPTY' and not obj.children: empty_parents_count += 1 elif obj.type == 'CAMERA': camera_count += 1 elif obj.type == 'LIGHT': light_count += 1 elif obj.type == 'IMAGE': image_count += 1 elif obj.type == 'MESH': z_dimension = obj.dimensions.z if z_dimension < 0.001: small_z_height_count += 1 if obj.type == 'MESH': # Get the dimensions of the object x, y, _ = obj.dimensions # Discard the Z dimension # Check if the X or Y dimension exceeds the threshold if x > threshold: exceeded_x = (obj.name, x) if y > threshold: exceeded_y = (obj.name, y) # Check if there is a LOD object and which level (Checking by name) for i in range(10): lod_name = "LOD{}".format(i) if lod_name in obj.name: lod_detected = True # Split the name to extract the LOD number lod_number_str = obj.name.split("LOD")[1] try: # Try to convert the extracted portion to an integer lod_number = int(lod_number_str) # Compare the LOD level if lod_level is None or lod_number > lod_level: lod_level = lod_number except ValueError: # Ignore cases where the extracted portion is not a valid integer continue # Check for the ".osm_building" or ".osm_roads" condition among selected objects if any(".osm_building" in obj.name or ".osm_roads" in obj.name for obj in bpy.context.scene.objects): osm_detected["OSM Detected"] = True # Iterate through all materials in the scene and count texture nodes with images (Better to keep outside first loop) for material in bpy.data.materials: if material.use_nodes and material.node_tree: for node in material.node_tree.nodes: if isinstance(node, bpy.types.ShaderNodeTexImage) and node.image is not None: texture_count += 1 if hasattr(node, 'image') and (not node.image or not os.path.exists(node.image.filepath)): materials_with_missing_textures.append(material.name) break # Iterate through all collections in the scene to check for partial collection names for collection in bpy.context.scene.collection.children: if ".osm" in collection.name: osm_detected[".osm Collection Detected"] = True if "way_profiles" in collection.name: osm_detected["way_profiles Collection Detected"] = True # Determine if any of the conditions were met and store True or False in osm_detected osm_detected = any(value for value in osm_detected.values()) # Calculate the average scale for all objects in the scene average_scale = total_scale / object_count if has_mesh_objects else 0 def search_empty_hierarchy(obj): nonlocal floor_count, floor_names, building_hierarchy, multi_storey_building # Check if the object is an empty or a mesh if obj.type == 'EMPTY' or obj.type == 'MESH': # Check if the empty's name matches the hierarchy structure if obj.name.startswith("Main") and len(obj.children) > 0: # Check if there's at least one child object starting with "Floor" floor_children = [child for child in obj.children if child.name.startswith("Floor")] if floor_children: # Increment floor count floor_count += len(floor_children) # Record floor names floor_names.extend([child.name for child in floor_children]) # Check if there is more than one floor child if len(floor_children) > 1: multi_storey_building = True # Check if at least one of the required children is present for any floor required_children = ["Architecture", "Interior", "Outside"] for floor_child in floor_children: children_names = [child.name for child in floor_child.children] if any(required_child in child_name for required_child in required_children for child_name in children_names): # Hierarchy found, set global variable to True building_hierarchy = True return # Recursively search through children for child in obj.children: search_empty_hierarchy(child) # Start search from the root of the scene's objects for obj in bpy.data.objects: search_empty_hierarchy(obj) if building_hierarchy: break # Set the calculated values to the properties context.scene.statistics.poly_count = poly_count context.scene.statistics.object_count = object_count context.scene.statistics.material_count = material_count context.scene.statistics.closest_object_name = closest_object_name context.scene.statistics.closest_distance = closest_distance context.scene.statistics.total_scale = total_scale context.scene.statistics.empty_parents_count = empty_parents_count context.scene.statistics.camera_count = camera_count context.scene.statistics.light_count = light_count context.scene.statistics.image_count = image_count context.scene.statistics.texture_count = texture_count context.scene.statistics.exceeded_x = exceeded_x context.scene.statistics.exceeded_y = exceeded_y context.scene.statistics.average_scale = average_scale context.scene.statistics.building_hierarchy = building_hierarchy context.scene.statistics.multi_storey_building = multi_storey_building context.scene.statistics.floor_count = floor_count # Initialize a list to store unsuitability messages unsuitability_messages = [] # Check suitability conditions and append appropriate messages if poly_count > 2500000: unsuitability_messages.append("(Too many polygons ({} polygons), should be les or equal 2500000.".format(poly_count)) #if poly_count < 100: # unsuitability_messages.append("There are ({} polygons), model is considered as a bloat.".format(poly_count)) if object_count > 100: unsuitability_messages.append("Too many objects ({} objects), should be less or equal 100.".format(object_count)) #if object_count < 1: # unsuitability_messages.append("({} mesh objects detected), model is considered as a bloat.".format(object_count)) if material_count > 100: unsuitability_messages.append("Too many materials ({} materials), should be less or equal 100.".format(material_count)) if not materials_mapped: unsuitability_messages.append("Materials are not mapped to any objects in the scene.") if texture_count > 100: unsuitability_messages.append("Too many textures ({} textures), should be less or equal 100.".format(texture_count)) #if texture_count == 0: # unsuitability_messages.append("There are ({} textures), should be more than that.".format(texture_count)) if closest_distance > 5000: unsuitability_messages.append("Closest object is too far from the origin ({} meters, should be less or equal to 5000 meters).".format(closest_distance)) if average_scale < 0.01: unsuitability_messages.append("Average scale is too small ({:.2f}), should be bigger or equal 0.1.".format(average_scale)) if exceeded_x or exceeded_y: unsuitability_messages.append("One of your models exceeded total limit size (5KM). Please check your conversion units or make sure that your model is exported in meters") if building_hierarchy == False: unsuitability_messages.append("Your model is missing required Hierarchy. Please use Reparent function or check Documentation") for message in unsuitability_messages: new_item = context.scene.statistics.unsuitability_messages.add() new_item.message = message print("Operator: scanned_scene.calculate_statistics() executed") return {'FINISHED'} # Operator to execute display heatmap by polygon function class OBJECT_OT_polygon_heatmap(bpy.types.Operator): """It will assign colors to objects based on number of polygons""" bl_idname = "object.polygon_heatmap" bl_label = "HM - Visualize by polygons" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): display_heatmap_by_polygons() print("Operator: object.polygon_heatmap() executed") return {'FINISHED'} # Operator to execute clear heatmap data function class OBJECT_OT_clear_heatmap_data(bpy.types.Operator): """It will clear heatmap data""" bl_idname = "object.polygon_heatmap_clear" bl_label = "HM - Clear heatmap data" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): clear_heatmap_data() restore_face_colors() print("Operator: object.polygon_heatmap_clear() executed") return {'FINISHED'} # Operator to execute display heatmap by face orientation function class OBJECT_OT_face_orientation_heatmap(bpy.types.Operator): """It will assign colors to objects based on face orientation (Backface culling)""" bl_idname = "object.face_orientation_heatmap" bl_label = "HM - Visualize by face orientation" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): set_face_colors((0.0, 1.0, 0.0, 0.737), (1.0, 0.0, 0.0, 0.737)) print("Operator: object.face_orientation_heatmap() executed") return {'FINISHED'} # Operator to execute display heatmap by instances function class OBJECT_OT_display_heatmap_by_instances(bpy.types.Operator): """It will assing colors to multi user objects (Instanced meshes) and non multi user objects (Not Instanced meshes)""" bl_idname = "object.heatmap_by_instances" bl_label = "HM - Visualize by instances" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): display_heatmap_by_instances() print("Operator: object.heatmap_by_instances() executed") return {'FINISHED'} # Operator to execute display heatmap by mesh density function class OBJECT_OT_display_heatmap_by_meshdensity(bpy.types.Operator): """It will assign colors to objects based on their mesh density In more complex scenes, this function can take a some time""" bl_idname = "object.heatmap_by_meshdensity" bl_label = "HM - Visualize by meshdensity" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): display_heatmap_by_density() print("Operator: object.heatmap_by_meshdensity() executed") return {'FINISHED'} # Operator to execute decimator v4 function class OBJECT_OT_decimator_v5(bpy.types.Operator): """It will reduce vertex count and try to keep shape as much as possible (Applies on all visible objects!)""" bl_idname = "object.decimator_v5" bl_label = "BOP - Reduce_vertex_count" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): decimator_v5() print("Operator: decimator_v5() executed") return {'FINISHED'} # Operator to execute pack external data into blend file function class OBJECT_OT_pack_external_data_into_blend_file(bpy.types.Operator): """It will enable Automatic packing of external data such as textures and others into .blend file. Default state is False. Since its toggle, if you press it twice, it will disable function. Additionally you can check it in File - External data - Automatically Pack Resources""" bl_idname = "object.pack_external_data" bl_label = "BOP - Pack external data into blend file" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): pack_external_data_into_blend_file() print("Operator: pack_external_data_into_blend_file() executed") return {'FINISHED'} # Operator to execute CreatePlanes function class OBJECT_OT_CreatePlanes(bpy.types.Operator): """It will create Floor planes which should be placed at each floor starting height. Floor planes also indicates for script where to cut model into individual floors. Floor planes are specifically named based your input and they are located inside Floor planes collection folder""" bl_idname = "object.create_planes_operator" bl_label = "Create Planes" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): scene = context.scene # Check if "Floor planes" collection exists, if not, create it if "Floor planes" not in bpy.data.collections: floor_planes_collection = bpy.data.collections.new("Floor planes") bpy.context.scene.collection.children.link(floor_planes_collection) else: floor_planes_collection = bpy.data.collections["Floor planes"] # Create a material for the planes mat = bpy.data.materials.new(name="FloorMaterial") mat.use_nodes = False mat.diffuse_color = (1, 1, 1, 0.2) # Set the color to white mat.use_fake_user = True # Ensure material is not deleted new_planes = [] # Create the specified number of planes and place them along the Z-axis for i in range(scene.number_of_floors): bpy.ops.mesh.primitive_plane_add(size=40, enter_editmode=False, align='WORLD', location=(0, 0, i * 2)) plane = bpy.context.active_object plane.name = f"Floor Temp {i}" # Assign the material to the plane plane.data.materials.append(mat) # Link the new plane to the "Floor planes" collection floor_planes_collection.objects.link(plane) # Unlink the plane from the main collection to keep it only in "Floor planes" bpy.context.scene.collection.objects.unlink(plane) new_planes.append(plane) # Rename the planes with the correct prefix after all planes are created total_planes = len(floor_planes_collection.objects) prefix_length = 3 # We want a 3-digit prefix for index, plane in enumerate(new_planes): new_prefix = str(total_planes - 1 - index).zfill(prefix_length) plane.name = f"{new_prefix} / Floor {index}" def add_default_suffix_to_floor(): # Get the collection name and default floor number from the scene properties collection_name = "Floor planes" default_floor = bpy.context.scene.default_floor - 1 # Adjust for zero-based indexing # Get the collection containing the floor planes collection = bpy.data.collections.get(collection_name) if not collection: print(f"Collection '{collection_name}' not found.") return # Filter out mesh objects from the collection mesh_objects = [obj for obj in collection.objects if obj.type == 'MESH'] # Sort mesh objects based on their Z-coordinate (from bottom to top) mesh_objects.sort(key=lambda obj: obj.location.z) # Check if the default_floor is valid if default_floor >= len(mesh_objects): print("Invalid default floor number.") return # Get the object at the default floor index default_object = mesh_objects[default_floor] # Add "Default" as a suffix to the object name default_object.name += " Default" # print(f"Added 'Default' as a suffix to the object name: {default_object.name}") add_default_suffix_to_floor() return {'FINISHED'} # Operator to execute FloorCutter function class OBJECT_OT_FloorCutter(bpy.types.Operator): """It will start Floorcutting process. If you are not satisfied with results press CTRL+Z to revert changes""" bl_idname = "object.floor_cutter" bl_label = "BOP - Floor cutter" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): floor_cutter(delete_interval=5) print("Operator: floor_cutter() executed") return {'FINISHED'} # Operator to execute FloorCutter function class OBJECT_OT_BakeAnimationData(bpy.types.Operator): """It will Bake Animation data into objects itself. This will allow easier manipulation with objects""" bl_idname = "object.bake_animation_data" bl_label = "BOP - Bake Animation data" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): bake_animations() print("Operator: bake_animations() executed") return {'FINISHED'} # Operator to execute Delente non visible geometry function class OBJECT_OT_Delete_non_visible_geometry(bpy.types.Operator): """It will Delete non visible geometry via camera array and ray casting""" bl_idname = "object.delete_non_visible_geometry" bl_label = "BOP - Delete non visible geometry" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): delete_non_visible_geometry() print("Operator: delete_non_visible_geometry() executed") return {'FINISHED'} #----------------------------PROPERTY GROUPS----------------------------# # Property group for Analyze scene class UnsuitabilityMessageItem(bpy.types.PropertyGroup): message: bpy.props.StringProperty(name="Message") # Property group for Analyze scene # Convert variables into properties class SceneStatisticsProperties(bpy.types.PropertyGroup): poly_count: bpy.props.IntProperty(name="Polygon Count", default=0) object_count: bpy.props.IntProperty(name="Object Count", default=0) material_count: bpy.props.IntProperty(name="Material Count", default=0) closest_object_name: bpy.props.StringProperty(name="Closest Object Name", default="") closest_distance: bpy.props.FloatProperty(name="Closest Distance", default=0.0) total_scale: bpy.props.FloatProperty(name="Total Scale", default=0.0) shared_geometry_count: bpy.props.IntProperty(name="Shared Geometry Count", default=0) has_bones_or_armatures: bpy.props.BoolProperty(name="Has Bones or Armatures", default=False) has_animation: bpy.props.BoolProperty(name="Has Animation", default=False) empty_parents_count: bpy.props.IntProperty(name="Empty Parents Count", default=0) camera_count: bpy.props.IntProperty(name="Camera Count", default=0) light_count: bpy.props.IntProperty(name="Light Count", default=0) image_count: bpy.props.IntProperty(name="Image Count", default=0) curve_count: bpy.props.IntProperty(name="Curve Count", default=0) texture_count: bpy.props.IntProperty(name="Texture Count", default=0) exceeded_x: bpy.props.PointerProperty(type=bpy.types.Object) exceeded_y: bpy.props.PointerProperty(type=bpy.types.Object) average_scale: bpy.props.FloatProperty(name="Average Scale", default=0.0) percentage_shared: bpy.props.FloatProperty(name="Percentage Shared", default=0.0) building_hierarchy: bpy.props.BoolProperty(name="Model has Hierarchy", default = False) multi_storey_building: bpy.props.BoolProperty(name="Multi storey building", default = False) floor_count: bpy.props.IntProperty(name="Floor count", default=0) # Change the unsuitability_messages property to a collection unsuitability_messages: bpy.props.CollectionProperty(type=UnsuitabilityMessageItem) #----------------------------MULTICLASS OPERATORS----------------------------# # Multiclass operators will take existing operators described above and execute them in defined order under def execute section class ClassicModelOperator(bpy.types.Operator): """It will execute functions best for Classic models""" bl_idname = "object.custom_multi_class_operator1" bl_label = "BOP - Custom Multi-Class Operator" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): print("Your Classic Model is FIXED!") # Execute operators in this order (Classic 3D model) bpy.ops.object.bake_animation_data() # bpy.ops.object.delete_animations() # bpy.ops.object.delete_armatures_and_bones() bpy.ops.object.reset_origin() bpy.ops.object.reset_scale_to_1() bpy.ops.object.recalculate_normals() bpy.ops.object.auto_smooth_all() bpy.ops.object.delete_custom_normals() bpy.ops.object.tris_to_quads() bpy.ops.object.merge_close_vertices() bpy.ops.object.remove_unused_datablocks() bpy.ops.object.delete_bloat() bpy.ops.object.merge_similar_materials() bpy.ops.object.delete_downward_faces() bpy.ops.object.create_instances() bpy.ops.object.decimator_v5() bpy.ops.object.lod_rename() bpy.ops.object.reparent() return {'FINISHED'} class ScannedModelOperator(bpy.types.Operator): """It will execute functions best for Scanned models""" bl_idname = "object.custom_multi_class_operator2" bl_label = "BOP - Custom Multi-Class Operator" bl_options = {'REGISTER', 'UNDO'} def execute(self, context): print("Your Scanned Model is FIXED!") # Execute operators in this order (Scanned Model) bpy.ops.object.delete_animations() bpy.ops.object.delete_armatures_and_bones() bpy.ops.object.reset_origin() bpy.ops.object.reset_scale_to_1() bpy.ops.object.delete_custom_normals() bpy.ops.object.merge_close_vertices() bpy.ops.object.remove_unused_datablocks() bpy.ops.object.delete_bloat() bpy.ops.object.rotate_model_operator() bpy.ops.object.merge_similar_materials() bpy.ops.object.decimator_v5() bpy.ops.object.lod_rename() bpy.ops.object.reparent() return {'FINISHED'} #----------------------------PANEL with buttons in View_3D----------------------------# # Panel to show the Custom OperatorPanel with Classic and Scanned model functions class OperatorHolderPanel(bpy.types.Panel): bl_idname = "OBJECT_PT_custom_operator_panel" bl_label = "Twinzo" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' def draw(self, context): layout = self.layout layout.label(text="Click on one of the buttons") layout.label(text="to optimize your 3D model.") layout.label(text="For undo press CTRL+Z") # Add a separator (vertical spacing) layout.separator() # Add the custom operator button layout.operator("object.custom_multi_class_operator1", text="FIX my Classic Model") # Add the custom operator button2 layout.operator("object.custom_multi_class_operator2", text="FIX my Scanned Model") # Add a separator (vertical spacing) layout.separator() # Add the object reset button layout.operator("object.reset", text="RESET") # Panel to show Object Separator Functions class ObjectSeparator(bpy.types.Panel): """Panel with ObjectSeparator function""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Object Separator" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout layout.label(text="In order to separate objects") layout.label(text="you can either select") layout.label(text="desired object or let script") layout.label(text="to apply it on all objects.") layout.label(text="For undo press CTRL+Z") # Add a separator (vertical spacing) layout.operator("object.object_separator", text="Separate Objects") # Panel to show Object Instancer Functions class ObjectInstancer(bpy.types.Panel): """Panel with ObjectInstancer function""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Object Instancer" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout layout.label(text="Identic or very similar objects") layout.label(text="will share their geometry, which") layout.label(text="results in better performance") layout.label(text="Applies on all objects in scene!") # Add a separator (vertical spacing) layout.operator("object.create_instances", text="Create Instanced Objects") # Panel to show Individual Functions class IndividualFunctions(bpy.types.Panel): """Individual functions, which you can manually execute as you like""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Individual Functions" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout # SubPanel to show Delete/Cleanup section class DeleteCleanupSubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "DELETE / CLEAN UP" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='EVENT_X') def draw(self, context): layout = self.layout row = layout.row() layout.label(text="Delete unnecessary geometry") # Add the delete downward faces button layout.operator("object.delete_downward_faces", text="Delete Downward Faces", icon='EVENT_X') # Add the delete duplicated meshes button layout.operator("object.delete_duplicated_meshes", text="Delete Duplicated Meshes", icon='EVENT_X') # Add the delete bloat button layout.operator("object.delete_bloat", text="Delete Bloat", icon='EVENT_X') # Add the delete armatures and bones button layout.operator("object.delete_armatures_and_bones", text="Delete Armature and Bones", icon='EVENT_X') # Add the delete animations button layout.operator("object.delete_animations", text="Delete Animations", icon='EVENT_X') # Add the delete non visible geometry button layout.operator("object.delete_non_visible_geometry", text="Delete Non Visible Geometry", icon='EVENT_X') layout.label(text="Cleanup tools") # Add the remove unused datablocks button layout.operator("object.remove_unused_datablocks", text="Remove Unused Datablocks", icon='EVENT_X') # SubPanel to show MeshModify section class MeshModifySubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "MESH MODIFY" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='MODIFIER_DATA') def draw(self, context): layout = self.layout row = layout.row() # Add the merge close vertices button layout.operator("object.merge_close_vertices", text="Merge Close Vertices", icon='MODIFIER_DATA') # Add the tris to quads button layout.operator("object.tris_to_quads", text="Tris to Quads", icon='MODIFIER_DATA') # Add the limited dissolve button layout.operator("object.limited_dissolve", text="Use Limited Dissolve", icon='MODIFIER_DATA') # Add the fix scan rotation button layout.operator("object.rotate_model_operator", text="Fix Scan Rotation", icon='MODIFIER_DATA') # Add the reset origin button layout.operator("object.reset_origin", text="Reset Origin", icon='MODIFIER_DATA') # Add the reset scale button layout.operator("object.reset_scale_to_1", text="Reset Scale to 1", icon='MODIFIER_DATA') # SubPanel to show Shading section class ShadingSubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "SHADING" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='DISC') def draw(self, context): layout = self.layout row = layout.row() # Add the auto-smooth button layout.operator("object.auto_smooth_all", text="Auto Smooth", icon='DISC') # Add the recalculate normals button layout.operator("object.recalculate_normals", text="Recalculate Mesh Normals", icon='DISC') # Add the delete custom normals button layout.operator("object.delete_custom_normals", text="Delete Custom Split Normals", icon='DISC') # SubPanel to show Hierarchy section class HierarchySubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "HIERARCHY" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='OUTLINER') def draw(self, context): layout = self.layout row = layout.row() # Add the reparent all button layout.operator("object.reparent", text="Reparent", icon='OUTLINER') # SubPanel to show Material section class MaterialSubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "MATERIALS" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='SHADING_TEXTURE') def draw(self, context): layout = self.layout row = layout.row() # Add the remove duplicated materials button row.operator("object.merge_similar_materials", text="Merge Similar Materials", icon='SHADING_TEXTURE') # SubPanel to show Others section class AnimationsSubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "ANIMATIONS" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='ARMATURE_DATA') def draw(self, context): layout = self.layout # Add bake animation data button layout.operator("object.bake_animation_data", text="Bake Animation data", icon='ARMATURE_DATA') # SubPanel to show Others section class OthersSubPanel(bpy.types.Panel): bl_parent_id = "IndividualFunctions" bl_label = "OTHERS" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='COLLAPSEMENU') def draw(self, context): layout = self.layout # Add the adjust LODs button layout.operator("object.lod_rename", text="Adjust LOD names", icon='COLLAPSEMENU') # Add the pack external data button layout.operator("object.pack_external_data", text="Toggle Pack External data", icon='COLLAPSEMENU') # Panel to show the DocumantationPanel class DocumentationPanel(bpy.types.Panel): bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Documentation" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout # Add button for Operator to execute the open URL function layout.operator("object.open_url_operator", text="Need help ?") # Panel to show the UpdatePanel class UpdatePanel(bpy.types.Panel): bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Updater" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout # Add the object Check update button layout.operator("wm.check_for_update", text="Check update") # Panel to show Object Instancer Functions class Decimator_v5Panel(bpy.types.Panel): """Panel with Decimator v5 function""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Decimator v5" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout layout.label(text="Decimator v5 will try to reduce") layout.label(text="vertex count as much as possible") layout.label(text="Reducing vertex count is heavy process") layout.label(text="and can take a while to compute") layout.label(text="---------------------------------") layout.label(text="If you are not satisfied with results") layout.label(text="Press CTRL+Z to revert changes") layout.label(text="Applies on all visible objects!") # Add a blank separator layout.separator() # Add button for Operator to execute the Decimator_v5 function layout.operator("object.decimator_v5", text="Process") # Panel to show the Analyze scene function class VIEW3D_PT_scene_statistics(bpy.types.Panel): bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Analyze scene" bl_idname = "PT_Scene_Statistics" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout layout.label(text="Choose category based on your") layout.label(text="model type.") # Panel to hold classical scene statistics functions class VIEW3D_PT_classical_scene_statistics_holder(bpy.types.Panel): bl_parent_id = "PT_Scene_Statistics" bl_label = "CLASSICAL MESH" bl_idname = "PT_Classical_scene_statistics_holder" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='MESH_CUBE') def draw(self, context): layout = self.layout # Panel to show Heatmap functions class VIEW3D_PT_classical_scene_heatmap_holder(bpy.types.Panel): bl_parent_id = "PT_Classical_scene_statistics_holder" bl_label = " Heatmap" bl_idname = "PT_Classical_scene_heatmap_holder" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='MESH_CUBE') def draw(self, context): blender_version = bpy.app.version layout = self.layout layout.label(text="Graphical representation of data visualized") layout.label(text="in different colors") layout.label(text="Description:") row = layout.row() if blender_version >= (4, 4, 0): row.label(icon="STRIP_COLOR_04", text="Good") row.label(icon="STRIP_COLOR_02", text="Okay") row.label(icon="STRIP_COLOR_01", text="Not great") layout.label(icon="STRIP_COLOR_09", text="Cannot be visualized") else: row.label(icon="SEQUENCE_COLOR_04", text="Good") row.label(icon="SEQUENCE_COLOR_02", text="Okay") row.label(icon="SEQUENCE_COLOR_01", text="Not great") layout.label(icon="SEQUENCE_COLOR_09", text="Cannot be visualized") # Add a separator (vertical spacing) layout.separator() # Add the custom operator button layout.operator("object.polygon_heatmap", text="Heatmap by polygons", icon='MESH_CUBE') layout.operator("object.face_orientation_heatmap", text="Heatmap by face orientation", icon='MESH_CUBE') layout.operator("object.heatmap_by_instances", text="Heatmap by instances", icon='MESH_CUBE') layout.operator("object.heatmap_by_meshdensity", text="Heatmap by meshdensity", icon='MESH_CUBE') # Add a separator (vertical spacing) layout.separator() layout.operator("object.polygon_heatmap_clear", text="Clear Heatmap data", icon='MESH_CUBE') # Panel to show the Analyze classical scene function class VIEW3D_PT_classical_scene_statistics(bpy.types.Panel): bl_parent_id = "PT_Classical_scene_statistics_holder" bl_label = " Statistics chart" bl_idname = "PT_Classical_scene_Statistics" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='MESH_CUBE') def draw(self, context): layout = self.layout # Add a button layout.operator("classical_scene.calculate_statistics", text="Calculate Statistics", icon='MESH_CUBE') layout.label(text="Statistics:") # Add a box layout with text to display the result box = layout.box() box.label(text=f"Polygons: {context.scene.statistics.poly_count}") box.label(text=f"Total Mesh Objects: {context.scene.statistics.object_count}") box.label(text=f"Materials: {context.scene.statistics.material_count}") box.label(text=f"Closest Object: {context.scene.statistics.closest_object_name}") box.label(text=f"Closest Distance: {context.scene.statistics.closest_distance}") # box.label(text=f"Total Scale: {context.scene.statistics.total_scale}") box.label(text=f"Bones or Armatures: {context.scene.statistics.has_bones_or_armatures}") box.label(text=f"Animations: {context.scene.statistics.has_animation}") box.label(text=f"Empty Parents: {context.scene.statistics.empty_parents_count}") box.label(text=f"Cameras: {context.scene.statistics.camera_count}") box.label(text=f"Lights: {context.scene.statistics.light_count}") # box.label(text=f"Image Count: {context.scene.statistics.image_count}") box.label(text=f"Curves: {context.scene.statistics.curve_count}") box.label(text=f"Textures: {context.scene.statistics.texture_count}") box.label(text=f"Exceeded X: {context.scene.statistics.exceeded_x}") box.label(text=f"Exceeded Y: {context.scene.statistics.exceeded_y}") box.label(text=f"Average Scale: {context.scene.statistics.average_scale:.2f}") box.label(text=f"Shared Geometry: {context.scene.statistics.percentage_shared:.2f}%") box.label(text=f"Building Hierarchy: {context.scene.statistics.building_hierarchy}") box.label(text=f"Multi storey building: {context.scene.statistics.multi_storey_building}") box.label(text=f"Floor count: {context.scene.statistics.floor_count}") # Add a separator layout.separator() layout.label(text="Results") # Add a box layout for unsuitability messages unsuitability_box = layout.box() unsuitability_box.label(text="Potentional Issues:") for message_item in context.scene.statistics.unsuitability_messages: unsuitability_box.label(text=message_item.message) # Panel to show the Analyze scanned scene function class VIEW3D_PT_ScanLidar_based_scene_statistics(bpy.types.Panel): bl_parent_id = "PT_Scene_Statistics" bl_label = "SCAN / LIDAR MESH" bl_idname = "PT_ScanLidar_scene_Statistics" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw_header(self, context): layout = self.layout layout.label(icon='CAMERA_DATA') def draw(self, context): layout = self.layout # Add a button layout.operator("scanned_scene.calculate_statistics", text="Calculate Statistics", icon='CAMERA_DATA') layout.label(text="Statistics:") # Add a box layout with text to display the result box = layout.box() box.label(text=f"Polygons: {context.scene.statistics.poly_count}") box.label(text=f"Total Mesh Objects: {context.scene.statistics.object_count}") box.label(text=f"Materials: {context.scene.statistics.material_count}") box.label(text=f"Closest Object: {context.scene.statistics.closest_object_name}") box.label(text=f"Closest Distance: {context.scene.statistics.closest_distance}") # box.label(text=f"Total Scale: {context.scene.statistics.total_scale}") box.label(text=f"Empty Parents: {context.scene.statistics.empty_parents_count}") box.label(text=f"Cameras: {context.scene.statistics.camera_count}") box.label(text=f"Lights: {context.scene.statistics.light_count}") # box.label(text=f"Image Count: {context.scene.statistics.image_count}") box.label(text=f"Curves: {context.scene.statistics.curve_count}") box.label(text=f"Textures: {context.scene.statistics.texture_count}") box.label(text=f"Exceeded X: {context.scene.statistics.exceeded_x}") box.label(text=f"Exceeded Y: {context.scene.statistics.exceeded_y}") box.label(text=f"Average Scale: {context.scene.statistics.average_scale:.2f}") box.label(text=f"Building Hierarchy: {context.scene.statistics.building_hierarchy}") box.label(text=f"Multi storey building: {context.scene.statistics.multi_storey_building}") box.label(text=f"Floor count: {context.scene.statistics.floor_count}") # Add a separator layout.separator() layout.label(text="Results") # Add a box layout for unsuitability messages unsuitability_box = layout.box() unsuitability_box.label(text="Potentional Issues:") for message_item in context.scene.statistics.unsuitability_messages: unsuitability_box.label(text=message_item.message) # Panel to show Object Instancer Functions class ExporterPanel(bpy.types.Panel): """Panel with Export function""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Exporter" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout layout.label(text="Please make sure you have selected") layout.label(text="all the objects you want to export ") # Display the export path property layout.prop(context.scene, "export_path") # Add a separator (vertical spacing) layout.operator("object.export_model", text="Export") # Panel to show Floor cutter function class FloorcutterPanel(bpy.types.Panel): """Floor cutter""" bl_parent_id = "OBJECT_PT_custom_operator_panel" bl_label = "Floor cutter" bl_space_type = 'VIEW_3D' bl_region_type = 'UI' bl_category = 'Twinzo' bl_options = {'DEFAULT_CLOSED'} def draw(self, context): layout = self.layout scene = context.scene layout.label(text="Floor cutter will cut and redistribute objects into") layout.label(text="individual floors, based on values you set in") layout.label(text="If you are not satisfied with results") layout.label(text="Press CTRL+Z to revert changes") layout.label(text="---------------------------------") layout.label(text="Step 1: Set number of Floors") layout.label(text="and choose which floor you") layout.label(text="want to see as default in") layout.label(text="Digital Twin app, than press") layout.label(text="Create Planes button:") # Add a blank separator layout.separator() row = layout.row() row.prop(scene, "number_of_floors", text="Number of Floors:") row = layout.row() row.prop(scene, "default_floor", text="Default Floor:") row = layout.row() row.operator("object.create_planes_operator") # Add a blank separator layout.separator() layout.label(text="Step 2: Move every labeled plane") layout.label(text="on each floor starting height:") # Add a blank separator layout.separator() layout.label(text="Step 3: Press button bellow to start") layout.label(text="floor cutting process:") row = layout.row() row.operator("object.floor_cutter") #----------------------------Register classes----------------------------# # Register the addon def register(): bpy.types.Scene.export_path = bpy.props.StringProperty( name="Export Path", description="Path for exporting FBX file", default="Choose path", subtype='FILE_PATH' ) enable_developer_extras() global set_face_colors, restore_face_colors set_face_colors, restore_face_colors = create_face_color_manager() bpy.utils.register_class(CheckForUpdateOperator) bpy.utils.register_class(OBJECT_OT_delete_downward_faces) bpy.utils.register_class(OBJECT_OT_tris_to_quads) bpy.utils.register_class(OBJECT_OT_auto_smooth_all) bpy.utils.register_class(OBJECT_OT_delete_custom_normals) bpy.utils.register_class(OBJECT_OT_delete_duplicated_meshes) bpy.utils.register_class(OBJECT_OT_merge_close_vertices) bpy.utils.register_class(OBJECT_OT_limited_dissolve) bpy.utils.register_class(OBJECT_OT_remove_unused_datablocks) bpy.utils.register_class(OBJECT_OT_reparent) bpy.utils.register_class(OBJECT_OT_recalculate_normals) bpy.utils.register_class(OBJECT_OT_open_url_operator) bpy.utils.register_class(OBJECT_OT_rotate_model_to_highest_normal_direction) bpy.utils.register_class(OBJECT_OT_reset_origin) bpy.utils.register_class(OBJECT_OT_reset_scale_to_1) bpy.utils.register_class(OBJECT_OT_reset_to_original_state) bpy.utils.register_class(OBJECT_OT_merge_similar_materials) bpy.utils.register_class(OBJECT_OT_delete_bloat) bpy.utils.register_class(OBJECT_OT_create_instances) bpy.utils.register_class(OBJECT_OT_object_separator) bpy.utils.register_class(OBJECT_OT_delete_armatures_and_bones) bpy.utils.register_class(OBJECT_OT_delete_animations) bpy.utils.register_class(OBJECT_OT_lod_rename) bpy.utils.register_class(OBJECT_OT_export_model) bpy.utils.register_class(OBJECT_OT_decimator_v5) bpy.utils.register_class(OBJECT_OT_pack_external_data_into_blend_file) bpy.utils.register_class(OBJECT_OT_polygon_heatmap) bpy.utils.register_class(OBJECT_OT_clear_heatmap_data) bpy.utils.register_class(OBJECT_OT_face_orientation_heatmap) bpy.utils.register_class(OBJECT_OT_display_heatmap_by_instances) bpy.utils.register_class(OBJECT_OT_display_heatmap_by_meshdensity) bpy.utils.register_class(ClassicModelOperator) bpy.utils.register_class(ScannedModelOperator) bpy.utils.register_class(OperatorHolderPanel) bpy.utils.register_class(IndividualFunctions) bpy.utils.register_class(DeleteCleanupSubPanel) bpy.utils.register_class(MeshModifySubPanel) bpy.utils.register_class(ShadingSubPanel) bpy.utils.register_class(HierarchySubPanel) bpy.utils.register_class(MaterialSubPanel) bpy.utils.register_class(AnimationsSubPanel) bpy.utils.register_class(OthersSubPanel) bpy.utils.register_class(ObjectSeparator) bpy.utils.register_class(ObjectInstancer) bpy.utils.register_class(VIEW3D_PT_scene_statistics) bpy.utils.register_class(VIEW3D_PT_classical_scene_statistics_holder) bpy.utils.register_class(VIEW3D_PT_classical_scene_heatmap_holder) bpy.utils.register_class(VIEW3D_PT_classical_scene_statistics) bpy.utils.register_class(VIEW3D_PT_ScanLidar_based_scene_statistics) bpy.utils.register_class(Decimator_v5Panel) # bpy.utils.register_class(FloorcutterPanel) # bpy.utils.register_class(OBJECT_OT_CreatePlanes) bpy.types.Scene.number_of_floors = bpy.props.IntProperty(name="Number of Floors", default=1, min=1, max=100, update=update_number_of_floors) bpy.types.Scene.lowest_floor_number = bpy.props.IntProperty(name="Lowest Floor Number", default=0, min=0, max=99) bpy.types.Scene.default_floor = bpy.props.IntProperty(name="Default Floor", default=1, min=1, max=100, update=update_default_floor) bpy.utils.register_class(OBJECT_OT_FloorCutter) bpy.utils.register_class(OBJECT_OT_BakeAnimationData) bpy.utils.register_class(OBJECT_OT_Delete_non_visible_geometry) bpy.utils.register_class(ExporterPanel) bpy.utils.register_class(DocumentationPanel) bpy.utils.register_class(UpdatePanel) bpy.utils.register_class(UnsuitabilityMessageItem) bpy.utils.register_class(SceneStatisticsProperties) bpy.utils.register_class(SCENE_OT_classical_calculate_statistics) bpy.utils.register_class(SCENE_OT_scanned_calculate_statistics) bpy.types.Scene.statistics = bpy.props.PointerProperty(type=SceneStatisticsProperties) # Unregister the addon def unregister(): bpy.utils.unregister_class(CheckForUpdateOperator) bpy.utils.unregister_class(OBJECT_OT_delete_downward_faces) bpy.utils.unregister_class(OBJECT_OT_tris_to_quads) bpy.utils.unregister_class(OBJECT_OT_auto_smooth_all) bpy.utils.unregister_class(OBJECT_OT_delete_custom_normals) bpy.utils.unregister_class(OBJECT_OT_delete_duplicated_meshes) bpy.utils.unregister_class(OBJECT_OT_merge_close_vertices) bpy.utils.unregister_class(OBJECT_OT_limited_dissolve) bpy.utils.unregister_class(OBJECT_OT_remove_unused_datablocks) bpy.utils.unregister_class(OBJECT_OT_reparent) bpy.utils.unregister_class(OBJECT_OT_recalculate_normals) bpy.utils.unregister_class(OBJECT_OT_open_url_operator) bpy.utils.unregister_class(OBJECT_OT_rotate_model_to_highest_normal_direction) bpy.utils.unregister_class(OBJECT_OT_reset_origin) bpy.utils.unregister_class(OBJECT_OT_reset_scale_to_1) bpy.utils.unregister_class(OBJECT_OT_reset_to_original_state) bpy.utils.unregister_class(OBJECT_OT_merge_similar_materials) bpy.utils.unregister_class(OBJECT_OT_delete_bloat) bpy.utils.unregister_class(OBJECT_OT_create_instances) bpy.utils.unregister_class(OBJECT_OT_object_separator) bpy.utils.unregister_class(OBJECT_OT_delete_armatures_and_bones) bpy.utils.unregister_class(OBJECT_OT_delete_animations) bpy.utils.unregister_class(OBJECT_OT_lod_rename) bpy.utils.unregister_class(OBJECT_OT_export_model) bpy.utils.unregister_class(OBJECT_OT_decimator_v5) bpy.utils.unregister_class(OBJECT_OT_pack_external_data_into_blend_file) bpy.utils.unregister_class(OBJECT_OT_polygon_heatmap) bpy.utils.unregister_class(OBJECT_OT_clear_heatmap_data) bpy.utils.unregister_class(OBJECT_OT_face_orientation_heatmap) bpy.utils.unregister_class(OBJECT_OT_display_heatmap_by_instances) bpy.utils.unregister_class(OBJECT_OT_display_heatmap_by_meshdensity) bpy.utils.unregister_class(ClassicModelOperator) bpy.utils.unregister_class(ScannedModelOperator) bpy.utils.unregister_class(OperatorHolderPanel) bpy.utils.unregister_class(IndividualFunctions) bpy.utils.unregister_class(DeleteCleanupSubPanel) bpy.utils.unregister_class(MeshModifySubPanel) bpy.utils.unregister_class(ShadingSubPanel) bpy.utils.unregister_class(HierarchySubPanel) bpy.utils.unregister_class(MaterialSubPanel) bpy.utils.unregister_class(AnimationsSubPanel) bpy.utils.unregister_class(OthersSubPanel) bpy.utils.unregister_class(ObjectSeparator) bpy.utils.unregister_class(ObjectInstancer) bpy.utils.unregister_class(VIEW3D_PT_scene_statistics) bpy.utils.unregister_class(VIEW3D_PT_classical_scene_statistics_holder) bpy.utils.unregister_class(VIEW3D_PT_classical_scene_heatmap_holder) bpy.utils.unregister_class(VIEW3D_PT_classical_scene_statistics) bpy.utils.unregister_class(VIEW3D_PT_ScanLidar_based_scene_statistics) bpy.utils.unregister_class(Decimator_v5Panel) # bpy.utils.unregister_class(FloorcutterPanel) # bpy.utils.register_class(OBJECT_OT_CreatePlanes) # bpy.types.Scene.plane_count = bpy.props.IntProperty(name="Plane Count", default=1, min=1) bpy.utils.unregister_class(OBJECT_OT_FloorCutter) bpy.utils.unregister_class(OBJECT_OT_BakeAnimationData) bpy.utils.unregister_class(OBJECT_OT_Delete_non_visible_geometry) bpy.utils.unregister_class(ExporterPanel) bpy.utils.unregister_class(DocumentationPanel) bpy.utils.unregister_class(UpdatePanel) bpy.utils.unregister_class(UnsuitabilityMessageItem) bpy.utils.unregister_class(SceneStatisticsProperties) bpy.utils.unregister_class(SCENE_OT_classical_calculate_statistics) bpy.utils.unregister_class(SCENE_OT_scanned_calculate_statistics) del bpy.types.Scene.number_of_floors del bpy.types.Scene.lowest_floor_number del bpy.types.Scene.default_floor del bpy.types.Scene.statistics del bpy.types.Scene.export_path if __name__ == "__main__": register()

Related content

If you encounter any issues or need assistance with using this product, please do not hesitate to reach out for support. Our team is here to help you resolve any problems and answer any questions you may have.
To create a support ticket, visit our support portal at https://partner.twinzo.eu/helpdesk/customer-care-1