1
0
mirror of https://github.com/drewcassidy/TexTools-Blender synced 2024-09-01 14:54:44 +00:00
Blender-TexTools/utilities_color.py

215 lines
6.0 KiB
Python
Raw Normal View History

2019-06-08 23:42:50 +00:00
import bpy
import bmesh
import operator
import time
from mathutils import Vector
from collections import defaultdict
from math import pi
from mathutils import Color
from . import settings
material_prefix = "TT_color_"
gamma = 2.2
def assign_slot(obj, index):
2019-12-18 20:53:16 +00:00
if index < len(obj.material_slots):
obj.material_slots[index].material = get_material(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Verify color
assign_color(index)
2019-06-08 23:42:50 +00:00
def safe_color(color):
2019-12-18 20:53:16 +00:00
if len(color) == 3:
if bpy.app.version > (2, 80, 0):
# Newer blender versions use RGBA
return (color[0], color[1], color[2], 1)
else:
return color
elif len(color) == 4:
if bpy.app.version > (2, 80, 0):
# Newer blender versions use RGBA
return color
else:
return (color[0], color[1], color[2])
return color
2019-06-08 23:42:50 +00:00
def assign_color(index):
2019-12-18 20:53:16 +00:00
material = get_material(index)
if material:
# material.use_nodes = False
2019-12-24 19:59:46 +00:00
2019-12-18 20:53:16 +00:00
rgb = get_color(index)
rgba = (rgb[0], rgb[1], rgb[2], 1)
2019-06-08 23:42:50 +00:00
2019-12-24 19:59:46 +00:00
if material.use_nodes and bpy.context.scene.render.engine == 'CYCLES' or material.use_nodes and bpy.context.scene.render.engine == 'BLENDER_EEVEE':
2019-12-18 20:53:16 +00:00
# Cycles material (Preferred for baking)
material.node_tree.nodes["Principled BSDF"].inputs[0].default_value = rgba
material.diffuse_color = rgba
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
elif not material.use_nodes and bpy.context.scene.render.engine == 'BLENDER_EEVEE':
# Legacy render engine, not suited for baking
material.diffuse_color = rgba
2019-06-08 23:42:50 +00:00
def get_material(index):
2019-12-18 20:53:16 +00:00
name = get_name(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Material already exists?
if name in bpy.data.materials:
2019-12-24 19:59:46 +00:00
material = bpy.data.materials[name]
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Check for incorrect matreials for current render engine
if not material:
replace_material(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
if not material.use_nodes and bpy.context.scene.render.engine == 'CYCLES':
replace_material(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
elif material.use_nodes and bpy.context.scene.render.engine == 'BLENDER_EEVEE':
replace_material(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
else:
2019-12-24 19:59:46 +00:00
return material
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
print("Could nt find {} , create a new one??".format(name))
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
material = create_material(index)
assign_color(index)
return material
2019-06-08 23:42:50 +00:00
# Replaace an existing material with a new one
# This is sometimes necessary after switching the render engine
def replace_material(index):
2019-12-18 20:53:16 +00:00
name = get_name(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
print("Replace material and create new")
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Check if material exists
if name in bpy.data.materials:
2019-12-24 19:59:46 +00:00
material = bpy.data.materials[name]
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Collect material slots we have to re-assign
slots = []
2019-12-24 19:59:46 +00:00
for obj in bpy.context.view_layer.objects:
2019-12-18 20:53:16 +00:00
for slot in obj.material_slots:
if slot.material == material:
slots.append(slot)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Get new material
material.user_clear()
bpy.data.materials.remove(material)
2019-12-24 19:59:46 +00:00
2019-12-18 20:53:16 +00:00
# Re-assign new material to all previous slots
material = create_material(index)
for slot in slots:
2019-12-24 19:59:46 +00:00
slot.material = material
2019-06-08 23:42:50 +00:00
def create_material(index):
2019-12-18 20:53:16 +00:00
name = get_name(index)
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Create new image instead
material = bpy.data.materials.new(name)
material.preview_render_type = 'FLAT'
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
if bpy.context.scene.render.engine == 'CYCLES':
# Cycles: prefer nodes as it simplifies baking
2019-12-24 19:59:46 +00:00
material.use_nodes = True
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
return material
2019-06-08 23:42:50 +00:00
def get_name(index):
2019-12-18 20:53:16 +00:00
return (material_prefix+"{:02d}").format(index)
2019-06-08 23:42:50 +00:00
def get_color(index):
2019-12-18 20:53:16 +00:00
if index < bpy.context.scene.texToolsSettings.color_ID_count:
return getattr(bpy.context.scene.texToolsSettings, "color_ID_color_{}".format(index))
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Default return (Black)
return (0, 0, 0)
2019-06-08 23:42:50 +00:00
def set_color(index, color):
2019-12-18 20:53:16 +00:00
if index < bpy.context.scene.texToolsSettings.color_ID_count:
2019-12-24 19:59:46 +00:00
setattr(bpy.context.scene.texToolsSettings,
"color_ID_color_{}".format(index), color)
2019-06-08 23:42:50 +00:00
def validate_face_colors(obj):
2019-12-18 20:53:16 +00:00
# Validate face colors and material slots
2019-12-24 19:59:46 +00:00
previous_mode = bpy.context.object.mode
2019-12-18 20:53:16 +00:00
count = bpy.context.scene.texToolsSettings.color_ID_count
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Verify enough material slots
if len(obj.material_slots) < count:
for i in range(count):
if len(obj.material_slots) < count:
bpy.ops.object.material_slot_add()
assign_slot(obj, len(obj.material_slots)-1)
else:
break
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# TODO: Check face.material_index
bpy.ops.object.mode_set(mode='EDIT')
2019-12-24 19:59:46 +00:00
bm = bmesh.from_edit_mesh(obj.data)
2019-12-18 20:53:16 +00:00
for face in bm.faces:
2019-12-24 19:59:46 +00:00
face.material_index %= count
2019-12-18 20:53:16 +00:00
obj.data.update()
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Remove material slots that are not used
if len(obj.material_slots) > count:
bpy.ops.object.mode_set(mode='OBJECT')
for i in range(len(obj.material_slots) - count):
if len(obj.material_slots) > count:
# Remove last
2019-12-24 19:59:46 +00:00
bpy.context.object.active_material_index = len(
obj.material_slots)-1
2019-12-18 20:53:16 +00:00
bpy.ops.object.material_slot_remove()
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
# Restore previous mode
bpy.ops.object.mode_set(mode=previous_mode)
2019-06-08 23:42:50 +00:00
def hex_to_color(hex):
2019-12-24 19:59:46 +00:00
2019-12-18 20:53:16 +00:00
hex = hex.strip('#')
lv = len(hex)
fin = list(int(hex[i:i + lv // 3], 16) for i in range(0, lv, lv // 3))
r = pow(fin[0] / 255, gamma)
g = pow(fin[1] / 255, gamma)
b = pow(fin[2] / 255, gamma)
fin.clear()
fin.append(r)
fin.append(g)
fin.append(b)
return tuple(fin)
2019-06-08 23:42:50 +00:00
def color_to_hex(color):
2019-12-18 20:53:16 +00:00
rgb = []
for i in range(3):
2019-12-24 19:59:46 +00:00
rgb.append(pow(color[i], 1.0/gamma))
2019-06-08 23:42:50 +00:00
2019-12-18 20:53:16 +00:00
r = int(rgb[0]*255)
g = int(rgb[1]*255)
b = int(rgb[2]*255)
2019-06-08 23:42:50 +00:00
2019-12-24 19:59:46 +00:00
return "#{:02X}{:02X}{:02X}".format(r, g, b)
2019-06-08 23:42:50 +00:00
def get_color_id(index, count):
2019-12-18 20:53:16 +00:00
# Get unique color
color = Color()
2019-12-24 19:59:46 +00:00
color.hsv = (index / (count)), 0.9, 1.0
return color