2020-07-12 21:04:38 +08:00
|
|
|
import bpy,bmesh
|
2020-10-07 15:20:51 +08:00
|
|
|
import mathutils
|
2020-10-06 23:57:21 +08:00
|
|
|
import bpy.types
|
2022-04-03 22:48:12 +08:00
|
|
|
from . import UTILS_functions
|
2020-07-12 21:04:38 +08:00
|
|
|
|
2020-09-02 23:04:21 +08:00
|
|
|
class BALLANCE_OT_rail_uv(bpy.types.Operator):
|
2020-07-20 10:13:18 +08:00
|
|
|
"""Create a UV for rail"""
|
|
|
|
bl_idname = "ballance.rail_uv"
|
|
|
|
bl_label = "Create Rail UV"
|
|
|
|
bl_options = {'UNDO'}
|
|
|
|
|
2020-10-06 23:57:21 +08:00
|
|
|
uv_type: bpy.props.EnumProperty(
|
|
|
|
name="Type",
|
|
|
|
description="Define how to create UV",
|
|
|
|
items=(
|
|
|
|
("POINT", "Point", "All UV will be created in a specific point"),
|
|
|
|
("UNIFORM", "Uniform", "All UV will be created within 1x1"),
|
2022-08-29 10:52:34 +08:00
|
|
|
("SCALE", "Scale", "Give a scale number to scale UV"),
|
|
|
|
("TT", "TT_ReflectionMapping", "The real internal process of Ballance rail")
|
2020-10-06 23:57:21 +08:00
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2020-10-19 22:12:27 +08:00
|
|
|
projection_axis: bpy.props.EnumProperty(
|
|
|
|
name="Projection axis",
|
|
|
|
description="Projection axis",
|
|
|
|
items=(
|
|
|
|
("X", "X axis", "X axis"),
|
|
|
|
("Y", "Y axis", "Y axis"),
|
|
|
|
("Z", "Z axis", "Z axis")
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2020-10-06 23:57:21 +08:00
|
|
|
uv_scale : bpy.props.FloatProperty(
|
|
|
|
name="Scale",
|
|
|
|
description="The scale of UV",
|
|
|
|
min=0.0,
|
|
|
|
default=1.0,
|
|
|
|
)
|
|
|
|
|
2020-07-20 10:13:18 +08:00
|
|
|
@classmethod
|
|
|
|
def poll(self, context):
|
2022-04-03 22:48:12 +08:00
|
|
|
return _check_rail_target()
|
2020-07-20 10:13:18 +08:00
|
|
|
|
2020-10-06 23:57:21 +08:00
|
|
|
def invoke(self, context, event):
|
|
|
|
wm = context.window_manager
|
|
|
|
return wm.invoke_props_dialog(self)
|
|
|
|
|
2020-07-20 10:13:18 +08:00
|
|
|
def execute(self, context):
|
2020-10-07 15:20:51 +08:00
|
|
|
if context.scene.BallanceBlenderPluginProperty.material_picker == None:
|
2022-04-03 22:48:12 +08:00
|
|
|
UTILS_functions.show_message_box(("No specific material", ), "Lost parameter", 'ERROR')
|
2020-10-07 15:20:51 +08:00
|
|
|
else:
|
2022-04-03 22:48:12 +08:00
|
|
|
_create_rail_uv(self.uv_type, context.scene.BallanceBlenderPluginProperty.material_picker, self.uv_scale, self.projection_axis)
|
2020-07-20 10:13:18 +08:00
|
|
|
return {'FINISHED'}
|
|
|
|
|
2020-10-06 23:57:21 +08:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
layout.prop(self, "uv_type")
|
|
|
|
layout.prop(context.scene.BallanceBlenderPluginProperty, "material_picker")
|
2022-08-29 10:52:34 +08:00
|
|
|
if self.uv_type == 'SCALE' or self.uv_type == 'UNIFORM':
|
2020-10-19 22:12:27 +08:00
|
|
|
layout.prop(self, "projection_axis")
|
2020-10-06 23:57:21 +08:00
|
|
|
if self.uv_type == 'SCALE':
|
|
|
|
layout.prop(self, "uv_scale")
|
|
|
|
|
2020-07-20 10:13:18 +08:00
|
|
|
# ====================== method
|
|
|
|
|
2022-04-03 22:48:12 +08:00
|
|
|
def _check_rail_target():
|
2020-07-20 10:13:18 +08:00
|
|
|
for obj in bpy.context.selected_objects:
|
|
|
|
if obj.type != 'MESH':
|
|
|
|
continue
|
|
|
|
if obj.mode != 'OBJECT':
|
|
|
|
continue
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2022-04-03 22:48:12 +08:00
|
|
|
def _get_distance(iterator):
|
2020-10-07 20:38:01 +08:00
|
|
|
is_first_min = True
|
|
|
|
is_first_max = True
|
|
|
|
max_value = 0.0
|
|
|
|
min_value = 0.0
|
|
|
|
|
|
|
|
for item in iterator:
|
|
|
|
if is_first_max:
|
|
|
|
is_first_max = False
|
|
|
|
max_value = item
|
|
|
|
else:
|
|
|
|
if item > max_value:
|
|
|
|
max_value = item
|
|
|
|
if is_first_min:
|
|
|
|
is_first_min = False
|
|
|
|
min_value = item
|
|
|
|
else:
|
|
|
|
if item < min_value:
|
|
|
|
min_value = item
|
|
|
|
|
|
|
|
return max_value - min_value
|
|
|
|
|
2022-04-03 22:48:12 +08:00
|
|
|
def _create_rail_uv(rail_type, material_pointer, scale_size, projection_axis):
|
2020-10-07 15:20:51 +08:00
|
|
|
objList = []
|
2020-07-20 10:13:18 +08:00
|
|
|
ignoredObj = []
|
2020-07-12 21:04:38 +08:00
|
|
|
for obj in bpy.context.selected_objects:
|
|
|
|
if obj.type != 'MESH':
|
2020-07-20 10:13:18 +08:00
|
|
|
ignoredObj.append(obj.name)
|
|
|
|
continue
|
|
|
|
if obj.mode != 'OBJECT':
|
|
|
|
ignoredObj.append(obj.name)
|
2020-07-12 21:04:38 +08:00
|
|
|
continue
|
2020-08-31 21:56:08 +08:00
|
|
|
if obj.data.uv_layers.active is None:
|
2020-09-02 23:04:21 +08:00
|
|
|
# create a empty uv for it.
|
|
|
|
obj.data.uv_layers.new(do_init=False)
|
2020-07-12 21:04:38 +08:00
|
|
|
|
2020-10-07 15:20:51 +08:00
|
|
|
objList.append(obj)
|
2020-07-12 21:04:38 +08:00
|
|
|
|
2020-10-07 15:20:51 +08:00
|
|
|
for obj in objList:
|
|
|
|
mesh = obj.data
|
|
|
|
|
|
|
|
# clean it material and set rail first
|
|
|
|
obj.data.materials.clear()
|
|
|
|
obj.data.materials.append(material_pointer)
|
|
|
|
|
2020-10-07 20:38:01 +08:00
|
|
|
# copy mesh vec for scale or uniform mode
|
|
|
|
vecList = mesh.vertices[:]
|
2020-10-07 15:20:51 +08:00
|
|
|
real_scale = 1.0
|
|
|
|
if rail_type == 'SCALE':
|
|
|
|
real_scale = scale_size
|
|
|
|
elif rail_type == 'UNIFORM':
|
|
|
|
# calc proper scale
|
2020-10-19 22:12:27 +08:00
|
|
|
if projection_axis == 'X':
|
|
|
|
maxLength = max(
|
2022-04-03 22:48:12 +08:00
|
|
|
_get_distance(vec.co[1] for vec in vecList),
|
|
|
|
_get_distance(vec.co[2] for vec in vecList)
|
2020-10-19 22:12:27 +08:00
|
|
|
)
|
|
|
|
elif projection_axis == 'Y':
|
|
|
|
maxLength = max(
|
2022-04-03 22:48:12 +08:00
|
|
|
_get_distance(vec.co[0] for vec in vecList),
|
|
|
|
_get_distance(vec.co[2] for vec in vecList)
|
2020-10-19 22:12:27 +08:00
|
|
|
)
|
|
|
|
elif projection_axis == 'Z':
|
|
|
|
maxLength = max(
|
2022-04-03 22:48:12 +08:00
|
|
|
_get_distance(vec.co[0] for vec in vecList),
|
|
|
|
_get_distance(vec.co[1] for vec in vecList)
|
2020-10-19 22:12:27 +08:00
|
|
|
)
|
2020-10-07 15:20:51 +08:00
|
|
|
real_scale = 1.0 / maxLength
|
|
|
|
|
2020-07-12 21:04:38 +08:00
|
|
|
uv_layer = mesh.uv_layers.active.data
|
|
|
|
for poly in mesh.polygons:
|
|
|
|
for loop_index in range(poly.loop_start, poly.loop_start + poly.loop_total):
|
2020-10-07 15:20:51 +08:00
|
|
|
# get correspond vec index
|
|
|
|
index = mesh.loops[loop_index].vertex_index
|
|
|
|
if rail_type == 'POINT':
|
|
|
|
# set to 1 point
|
|
|
|
uv_layer[loop_index].uv[0] = 0
|
|
|
|
uv_layer[loop_index].uv[1] = 1
|
2022-08-29 10:52:34 +08:00
|
|
|
elif rail_type == 'SCALE' or rail_type == 'UNIFORM':
|
2020-10-07 15:20:51 +08:00
|
|
|
# following xy -> uv scale
|
2020-10-19 22:12:27 +08:00
|
|
|
#
|
|
|
|
# use Z axis: X->U Y->V
|
|
|
|
# use X axis: Y->U Z->V
|
|
|
|
# use Y axis: X->U Z->V
|
|
|
|
if projection_axis == 'X':
|
|
|
|
uv_layer[loop_index].uv[0] = vecList[index].co[1] * real_scale
|
|
|
|
uv_layer[loop_index].uv[1] = vecList[index].co[2] * real_scale
|
|
|
|
elif projection_axis == 'Y':
|
|
|
|
uv_layer[loop_index].uv[0] = vecList[index].co[0] * real_scale
|
|
|
|
uv_layer[loop_index].uv[1] = vecList[index].co[2] * real_scale
|
|
|
|
elif projection_axis == 'Z':
|
|
|
|
uv_layer[loop_index].uv[0] = vecList[index].co[0] * real_scale
|
|
|
|
uv_layer[loop_index].uv[1] = vecList[index].co[1] * real_scale
|
2022-08-29 10:52:34 +08:00
|
|
|
elif rail_type == 'TT':
|
|
|
|
(uv_layer[loop_index].uv[0], uv_layer[loop_index].uv[1]) = _tt_reflection_mapping_compute(
|
|
|
|
vecList[index].co,
|
|
|
|
mesh.loops[loop_index].normal,
|
|
|
|
(0.0, 0.0, 0.0)
|
|
|
|
)
|
2020-07-12 21:04:38 +08:00
|
|
|
|
2020-07-20 10:13:18 +08:00
|
|
|
if len(ignoredObj) != 0:
|
2022-04-03 22:48:12 +08:00
|
|
|
UTILS_functions.show_message_box(
|
|
|
|
("Following objects are not processed due to they are not suit for this function now: ", ) + tuple(ignoredObj),
|
|
|
|
"Execution result", 'INFO'
|
|
|
|
)
|
2022-08-29 10:52:34 +08:00
|
|
|
|
|
|
|
def _tt_reflection_mapping_compute(_point, _n, _refobj):
|
|
|
|
# switch blender coord to virtools coord for convenient calc
|
|
|
|
point = mathutils.Vector((_point[0], _point[2], _point[1]))
|
|
|
|
n = mathutils.Vector((_n[0], _n[2], _n[1])).normalized()
|
|
|
|
refobj = mathutils.Vector((_refobj[0], _refobj[2], _refobj[1]))
|
|
|
|
|
|
|
|
p = (refobj - point).normalized()
|
|
|
|
b=(((2*(p*n))*n)-p).normalized()
|
|
|
|
|
|
|
|
# convert back to blender coord
|
|
|
|
return ((b.x + 1.0) / 2.0, -(b.z + 1.0) / 2.0)
|