Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

Operator to automatically bake lightmaps for selected objects. #281

Open
wants to merge 18 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
18 commits
Select commit Hold shift + click to select a range
4af0244
Operator to automatically bake lightmaps for selected objects.
GottfriedHofmann Apr 22, 2024
567b121
Use smart UV project instead of lightmap pack
GottfriedHofmann May 8, 2024
a6bad91
Use Smart UV Project instead of Lightmap Pack
GottfriedHofmann May 17, 2024
496b9f4
Merge branch 'Hubs-Foundation:master' into LightMapBakeOperator
GottfriedHofmann Jul 30, 2024
deda117
Fix liniting issues
GottfriedHofmann Jul 30, 2024
6a69cfe
Fix liniting issues
GottfriedHofmann Jul 30, 2024
c0c68ea
Update addons/io_hubs_addon/components/operators.py
GottfriedHofmann Sep 3, 2024
4f06808
Add custom name in consts.py for the Lightmap UV Layer
GottfriedHofmann Sep 3, 2024
021d1ba
Return to old settings (render engine, bake etc.) after bake is finis…
GottfriedHofmann Sep 3, 2024
d5cfa46
Remove check whether file is saved, not needed anymore with packed im…
GottfriedHofmann Sep 3, 2024
5ca8cd1
Add warning to readme that the second UV layer could be overwritten b…
GottfriedHofmann Sep 3, 2024
8582eda
Adjust name of lightmap uv layer to be the same as in the readme.
GottfriedHofmann Sep 3, 2024
3d03bea
Suggestion: Create object groups with the same material for unwrappin…
GottfriedHofmann Sep 4, 2024
020d77d
Fix linting issues
GottfriedHofmann Sep 4, 2024
7e455ef
Remove TODO
GottfriedHofmann Sep 4, 2024
cf994da
Several fixes thanks to @vincentfretin
GottfriedHofmann Oct 8, 2024
a91a5b4
Fix: Create selection sets materialwise to omit overwriting already b…
GottfriedHofmann Oct 8, 2024
9a3f6f4
Microfix: Remove TODO
GottfriedHofmann Oct 8, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,10 @@ Note that for use in Hubs, you currently **MUST** use the second UV set, as Thre

![setting bake UV](https://user-images.githubusercontent.com/130735/83697782-b9e96b00-a5b4-11ea-986b-6690c69d8a3f.png)

# Automatically baking Lightmaps

To automatically create the node-setup needed to bake lightmaps and run baking on one step, select all objects you want to bake lightmaps for and got to `Object Properties > Hubs Lightmap Baker` and click on `Bake Lightmaps of selected objects`. **WARNING**: If a second UV layer is present on an object but it does not have a material with a `MOZ_lightmap` node, the UV layer will be overwritten!

# Exporting

This addon works in conjunction with the official glTF add-on, so exporting is done through it. Select "File > Export > glTF 2.0" and then ensure "Hubs Components" is enabled under "Extensions".
Expand Down
3 changes: 3 additions & 0 deletions addons/io_hubs_addon/components/consts.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
from math import pi

LIGHTMAP_LAYER_NAME = "LightmapUV"
LIGHTMAP_UV_ISLAND_MARGIN = 0.01

DISTANCE_MODELS = [("inverse", "Inverse drop off (inverse)",
"Volume will decrease inversely with distance"),
("linear", "Linear drop off (linear)",
Expand Down
206 changes: 205 additions & 1 deletion addons/io_hubs_addon/components/operators.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import bpy
from bpy.props import StringProperty, IntProperty, BoolProperty, CollectionProperty
from bpy.props import StringProperty, IntProperty, BoolProperty, CollectionProperty, FloatProperty
from bpy.types import Operator, PropertyGroup
from functools import reduce

Expand All @@ -11,6 +11,7 @@
from .gizmos import update_gizmos
from .utils import is_linked, redraw_component_ui
from ..icons import get_hubs_icons
from .consts import LIGHTMAP_LAYER_NAME, LIGHTMAP_UV_ISLAND_MARGIN
import os


Expand Down Expand Up @@ -670,6 +671,207 @@ def invoke(self, context, event):
return {'RUNNING_MODAL'}


class BakeLightmaps(Operator):
bl_idname = "object.bake_lightmaps"
bl_label = "Bake Lightmaps"
bl_description = "Bake lightmaps of selected objects using the Cycles render engine and pack them into the .blend."
bl_options = {'REGISTER', 'UNDO'}

default_intensity: FloatProperty(name="Lightmaps Intensity",
default=3.14,
description="Multiplier for hubs on how to interpret the brightness of the image. Set this to 1.0 if you have set up the lightmaps manually and use a non-HDR format like png or jpg.")
resolution: IntProperty(name="Lightmaps Resolution",
default=2048,
description="The pixel resolution of the resulting lightmap.")
samples: IntProperty(name="Max Samples",
default=1024,
description="The number of samples to use for baking. Higher values reduce noise but take longer.")

def create_uv_layouts(self, context, mesh_objs):
# set up UV layer structure. The first layer has to be UV0, the second one LIGHTMAP_LAYER_NAME for the lightmap.
for obj in mesh_objs:
obj_uv_layers = obj.data.uv_layers
# Check whether there are any UV layers and if not, create the two that are required.
if len(obj_uv_layers) == 0:
obj_uv_layers.new(name='UV0')
obj_uv_layers.new(name=LIGHTMAP_LAYER_NAME)

# In case there is only one UV layer create a second one named LIGHTMAP_LAYER_NAME for the lightmap.
if len(obj_uv_layers) == 1:
obj_uv_layers.new(name=LIGHTMAP_LAYER_NAME)
# Check if object has a second UV layer. If it is named LIGHTMAP_LAYER_NAME, assume it is used for the lightmap.
# Otherwise add a new UV layer LIGHTMAP_LAYER_NAME and place it second in the slot list.
elif obj_uv_layers[1].name != LIGHTMAP_LAYER_NAME:
print("The second UV layer in hubs should be named " + LIGHTMAP_LAYER_NAME + " and is reserved for the lightmap, all the layers >1 are ignored.")
obj_uv_layers.new(name=LIGHTMAP_LAYER_NAME)
# The new layer is the last in the list, swap it for position 1
obj_uv_layers[1], obj_uv_layers[-1] = obj_uv_layers[-1], obj_uv_layers[1]

# The layer for the lightmap needs to be the active one before lightmap packing
obj_uv_layers.active = obj_uv_layers[LIGHTMAP_LAYER_NAME]
# Set the object as selected in object mode
obj.select_set(True)

# run UV lightmap packing on all selected objects
bpy.ops.object.mode_set(mode='EDIT')
bpy.ops.mesh.select_all(action='SELECT')
# bpy.ops.uv.lightmap_pack()
bpy.ops.uv.smart_project(island_margin=LIGHTMAP_UV_ISLAND_MARGIN)
bpy.ops.object.mode_set(mode='OBJECT')
# Deselct the objects again to return without changing the scene
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
for obj in mesh_objs:
obj.select_set(False)
# Update the view layer so all parts take notice of the changed UV layout
bpy.context.view_layer.update()

return{'FINISHED'}
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved

def execute(self, context):
# Check selected objects
selected_objects = bpy.context.selected_objects

# filter mesh objects and others
mesh_objs, other_objs = [], []
for ob in selected_objects:
(mesh_objs if ob.type == 'MESH' else other_objs).append(ob)
# Remove all objects from selection so we can easily re-select subgroups later
ob.select_set(False)

# for ob in other_objs:
# Remove non-mesh objects from selection to ensure baking will work
# ob.select_set(False)

# Gather all materials on the selected objects
# materials = []
# Dictionary that stores which object has which materials so we can group them later
material_object_associations = {}
for obj in mesh_objs:
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
if len(obj.material_slots) >= 1:
# TODO: Make more efficient
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
for slot in obj.material_slots:
if slot.material is not None:
mat = slot.material
if mat not in material_object_associations:
# materials.append(mat)
material_object_associations[mat] = []
material_object_associations[mat].append(obj)
else:
# an object without materials should not be selected when running the bake operator
print("Object " + obj.name + " does not have material slots, removing from set")
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
obj.select_set(False)
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
mesh_objs.remove(obj)

print(material_object_associations.items())
# Set up the UV layer structure and auto-unwrap optimized for lightmaps
visited_objects = set()
for mat, obj_list in material_object_associations.items():
for ob in visited_objects:
if ob in obj_list:
obj_list.remove(ob)
self.create_uv_layouts(context, obj_list)
for ob in obj_list:
visited_objects.add(ob)

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This visited_objects logic seems wrong. I spend 1h thinking about it, I hope my example will be clear.

We agree that an object can have several materials, each material with their own image texture, the lightmap uvmap is shared for those materials on an object.

Example:
obj1 with alu and glass materials
obj2 with alu
obj3 with glass
obj4 with rock

We handle alu material first, create uvmap for obj1 obj2 on same image texture for alu.
Then we handle glass material, we don't recreate the uvmap for obj1 because of your visited_objects, we create the uvmap for obj3. The faces for obj1 and obj3 may end up overlapping on the same image texture for glass.

What we need to do is create the uvmap for obj1 obj2 obj3 with the same smart uv project.
For example with alu, glass and rock materials,
iterate over materials, iterating over alu objects, if an object has other materials, include all objects of those materials to do the uv smart project. Mark alu and glass as done. Continue iterating over materials, skip glass because it's done, then handle rock material.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You are totally right, it needs to work by material and not by object, should be adressed by a91a5b4


# Check for the required nodes and set them up if not present
lightmap_texture_nodes = []
for mat in material_object_associations.keys():
mat_nodes = mat.node_tree.nodes
lightmap_nodes = [node for node in mat_nodes if node.bl_idname == 'moz_lightmap.node']
if len(lightmap_nodes) > 1:
print("Too many lightmap nodes in node tree of material", mat.name)
elif len(lightmap_nodes) < 1:
lightmap_texture_nodes.append(self.setup_moz_lightmap_nodes(mat.node_tree))
else:
# TODO: Check wether all nodes are set up correctly, for now assume they are
lightmap_nodes[0].intensity = self.default_intensity
# the image texture node needs to be the active one for baking, it is connected to the lightmap node so get it from there
lightmap_texture_node = lightmap_nodes[0].inputs[0].links[0].from_node
mat.node_tree.nodes.active = lightmap_texture_node
lightmap_texture_nodes.append(lightmap_texture_node)

# Re-select all the objects that need baking before running the operator
for ob in mesh_objs:
ob.select_set(True)
# Baking has to happen in Cycles, it is not supported in EEVEE yet
render_engine_tmp = context.scene.render.engine
context.scene.render.engine = 'CYCLES'
samples_tmp = context.scene.cycles.samples
context.scene.cycles.samples = self.samples
# Baking needs to happen without the color pass because we only want the direct and indirect light contributions
bake_settings_before = context.scene.render.bake
bake_settings = context.scene.render.bake
bake_settings.use_pass_direct = True
bake_settings.use_pass_indirect = True
bake_settings.use_pass_color = False
# The should be small because otherwise it could overwrite UV islands
bake_settings.margin = 2
# Not sure whether this has any influence
bake_settings.image_settings.file_format = 'HDR'
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
context.scene.render.image_settings.file_format = 'HDR'
bpy.ops.object.bake(type='DIFFUSE')
# After baking is done, return everything back to normal
context.scene.cycles.samples = samples_tmp
context.scene.render.engine = render_engine_tmp
# Pack all newly created or updated images
for node in lightmap_texture_nodes:
file_path = bpy.path.abspath(f"{bpy.app.tempdir}/{node.image.name}.hdr")
# node.image.save_render(file_path)
node.image.filepath_raw = file_path
node.image.file_format = 'HDR'
node.image.save()
node.image.pack()
# Update the filepath so it unpacks nicely for the user.
# TODO: Mechanism taken from reflection_probe.py line 300-306, de-duplicate
new_filepath = f"//{node.image.name}.hdr"
node.image.packed_files[0].filepath = new_filepath
node.image.filepath_raw = new_filepath
node.image.filepath = new_filepath

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tested those four lines in my own script on blender 4.2.1, it doesn't seem to be needed if you want to unpack the textures later, the names are correct in the textures folder as far as I can tell, and actually those lines gave me an error during the execution saying the file didn't exist. I didn't test your PR though, so may not apply, but that's something to verify.


# Remove file from temporary directory to de-clutter the system. Especially on windows the temporary directory is rarely purged.
if os.path.exists(file_path):
os.remove(file_path)

# return to old settings
bake_settings = bake_settings_before
context.scene.cycles.samples = samples_tmp
context.scene.render.engine = render_engine_tmp

return {'FINISHED'}

def invoke(self, context, event):
# needed to get the dialoge with the intensity
GottfriedHofmann marked this conversation as resolved.
Show resolved Hide resolved
return context.window_manager.invoke_props_dialog(self)

def setup_moz_lightmap_nodes(self, node_tree):
''' Returns the lightmap texture node of the newly created setup '''
mat_nodes = node_tree.nodes
# This function gets called when no lightmap node is present
lightmap_node = mat_nodes.new(type="moz_lightmap.node")
lightmap_node.intensity = self.default_intensity

lightmap_texture_node = mat_nodes.new(type="ShaderNodeTexImage")
lightmap_texture_node.location[0] -= 300

img = bpy.data.images.new('LightMap', self.resolution, self.resolution, alpha=False, float_buffer=True)
lightmap_texture_node.image = img
if bpy.app.version < (4, 0, 0):
lightmap_texture_node.image.colorspace_settings.name = "Linear"
else:
lightmap_texture_node.image.colorspace_settings.name = "Linear Rec.709"

UVmap_node = mat_nodes.new(type="ShaderNodeUVMap")
UVmap_node.uv_map = "UV1"
UVmap_node.location[0] -= 500

node_tree.links.new(UVmap_node.outputs['UV'], lightmap_texture_node.inputs['Vector'])
node_tree.links.new(lightmap_texture_node.outputs['Color'], lightmap_node.inputs['Lightmap'])

# the image texture node needs to be the active one for baking
node_tree.nodes.active = lightmap_texture_node

return lightmap_texture_node


def register():
bpy.utils.register_class(AddHubsComponent)
bpy.utils.register_class(RemoveHubsComponent)
Expand All @@ -681,6 +883,7 @@ def register():
bpy.utils.register_class(ViewReportInInfoEditor)
bpy.utils.register_class(CopyHubsComponent)
bpy.utils.register_class(OpenImage)
bpy.utils.register_class(BakeLightmaps)
bpy.types.WindowManager.hubs_report_scroll_index = IntProperty(
default=0, min=0)
bpy.types.WindowManager.hubs_report_scroll_percentage = IntProperty(
Expand All @@ -700,6 +903,7 @@ def unregister():
bpy.utils.unregister_class(ViewReportInInfoEditor)
bpy.utils.unregister_class(CopyHubsComponent)
bpy.utils.unregister_class(OpenImage)
bpy.utils.unregister_class(BakeLightmaps)
del bpy.types.WindowManager.hubs_report_scroll_index
del bpy.types.WindowManager.hubs_report_scroll_percentage
del bpy.types.WindowManager.hubs_report_last_title
Expand Down
15 changes: 15 additions & 0 deletions addons/io_hubs_addon/components/ui.py
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,19 @@ def draw(self, context):
draw_components_list(self, context)


class HubsObjectLightmapPanel(bpy.types.Panel):
bl_label = "Hubs Lightmap Baker"
bl_idname = "OBJECT_PT_hubs_lightmap_baker"
bl_space_type = 'PROPERTIES'
bl_region_type = 'WINDOW'
bl_context = "object"

def draw(self, context):
layout = self.layout
row = layout.row()
row.operator("object.bake_lightmaps", text="Bake Lightmaps of selected objects")


class HUBS_PT_ToolsPanel(bpy.types.Panel):
bl_idname = "HUBS_PT_ToolsPanel"
bl_space_type = 'VIEW_3D'
Expand Down Expand Up @@ -230,6 +243,7 @@ def register():
bpy.utils.register_class(HubsBonePanel)
bpy.utils.register_class(TooltipLabel)
bpy.utils.register_class(HUBS_PT_ToolsPanel)
bpy.utils.register_class(HubsObjectLightmapPanel)

bpy.types.TOPBAR_MT_window.append(window_menu_addition)
bpy.types.VIEW3D_MT_object.append(object_menu_addition)
Expand All @@ -243,6 +257,7 @@ def unregister():
bpy.utils.unregister_class(HubsBonePanel)
bpy.utils.unregister_class(TooltipLabel)
bpy.utils.unregister_class(HUBS_PT_ToolsPanel)
bpy.utils.unregister_class(HubsObjectLightmapPanel)

bpy.types.TOPBAR_MT_window.remove(window_menu_addition)
bpy.types.VIEW3D_MT_object.remove(object_menu_addition)
Expand Down
Loading