2019-06-08 23:42:50 +00:00
|
|
|
import bpy
|
|
|
|
import os
|
|
|
|
import bmesh
|
|
|
|
import math
|
|
|
|
import operator
|
|
|
|
|
|
|
|
from mathutils import Vector
|
|
|
|
from collections import defaultdict
|
2019-12-24 19:59:46 +00:00
|
|
|
from itertools import chain # 'flattens' collection of iterables
|
2019-06-08 23:42:50 +00:00
|
|
|
|
|
|
|
from . import utilities_uv
|
|
|
|
|
|
|
|
|
|
|
|
class op(bpy.types.Operator):
|
2019-12-18 20:53:16 +00:00
|
|
|
bl_idname = "uv.textools_island_align_world"
|
|
|
|
bl_label = "Align World"
|
|
|
|
bl_description = "Align selected UV islands to world / gravity directions"
|
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# is_global = bpy.props.BoolProperty(
|
|
|
|
# name = "Global Axis",
|
|
|
|
# description = "Global or local object axis alignment",
|
|
|
|
# default = False
|
|
|
|
# )
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# def draw(self, context):
|
|
|
|
# layout = self.layout
|
|
|
|
# layout.prop(self, "is_global")
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
if not bpy.context.active_object:
|
|
|
|
return False
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Only in Edit mode
|
2019-12-18 20:53:16 +00:00
|
|
|
if bpy.context.active_object.mode != 'EDIT':
|
|
|
|
return False
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Requires UV map
|
2019-12-18 20:53:16 +00:00
|
|
|
if not bpy.context.object.data.uv_layers:
|
|
|
|
return False
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
if bpy.context.scene.tool_settings.use_uv_select_sync:
|
|
|
|
return False
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Only in UV editor mode
|
2019-12-18 20:53:16 +00:00
|
|
|
if bpy.context.area.type != 'IMAGE_EDITOR':
|
|
|
|
return False
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
return True
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
def execute(self, context):
|
|
|
|
main(self)
|
|
|
|
return {'FINISHED'}
|
2019-06-08 23:42:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main(context):
|
2019-12-18 20:53:16 +00:00
|
|
|
print("\n________________________\nis_global")
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Store selection
|
2019-12-18 20:53:16 +00:00
|
|
|
utilities_uv.selection_store()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
bm = bmesh.from_edit_mesh(bpy.context.active_object.data)
|
|
|
|
uv_layers = bm.loops.layers.uv.verify()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Only in Face or Island mode
|
2019-12-18 20:53:16 +00:00
|
|
|
if bpy.context.scene.tool_settings.uv_select_mode is not 'FACE' or 'ISLAND':
|
|
|
|
bpy.context.scene.tool_settings.uv_select_mode = 'FACE'
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
obj = bpy.context.object
|
|
|
|
bm = bmesh.from_edit_mesh(bpy.context.active_object.data)
|
|
|
|
uv_layers = bm.loops.layers.uv.verify()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
islands = utilities_uv.getSelectionIslands()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
for faces in islands:
|
|
|
|
# Get average viewport normal of UV island
|
2019-12-24 19:59:46 +00:00
|
|
|
avg_normal = Vector((0, 0, 0))
|
2019-12-18 20:53:16 +00:00
|
|
|
for face in faces:
|
2019-12-24 19:59:46 +00:00
|
|
|
avg_normal += face.normal
|
|
|
|
avg_normal /= len(faces)
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# avg_normal = (obj.matrix_world*avg_normal).normalized()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# Which Side
|
|
|
|
x = 0
|
|
|
|
y = 1
|
|
|
|
z = 2
|
|
|
|
max_size = max(abs(avg_normal.x), abs(avg_normal.y), abs(avg_normal.z))
|
2019-12-24 19:59:46 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# Use multiple steps
|
|
|
|
for i in range(3):
|
|
|
|
if(abs(avg_normal.x) == max_size):
|
|
|
|
print("x normal")
|
2019-12-24 19:59:46 +00:00
|
|
|
align_island(obj, bm, uv_layers, faces, y,
|
|
|
|
z, avg_normal.x < 0, False)
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
elif(abs(avg_normal.y) == max_size):
|
|
|
|
print("y normal")
|
2019-12-24 19:59:46 +00:00
|
|
|
align_island(obj, bm, uv_layers, faces, x,
|
|
|
|
z, avg_normal.y > 0, False)
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
elif(abs(avg_normal.z) == max_size):
|
|
|
|
print("z normal")
|
2019-12-24 19:59:46 +00:00
|
|
|
align_island(obj, bm, uv_layers, faces, x,
|
|
|
|
y, False, avg_normal.z < 0)
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
print("align island: faces {}x n:{}, max:{}".format(
|
|
|
|
len(faces), avg_normal, max_size))
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# Restore selection
|
2019-12-18 20:53:16 +00:00
|
|
|
utilities_uv.selection_restore()
|
2019-06-08 23:42:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
def align_island(obj, bm, uv_layers, faces, x=0, y=1, flip_x=False, flip_y=False):
|
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# Find lowest and highest verts
|
2019-12-24 19:59:46 +00:00
|
|
|
minmax_val = [0, 0]
|
2019-12-18 20:53:16 +00:00
|
|
|
minmax_vert = [None, None]
|
|
|
|
|
|
|
|
axis_names = ['x', 'y', 'z']
|
2019-12-24 19:59:46 +00:00
|
|
|
print("Align shell {}x at {},{} flip {},{}".format(
|
|
|
|
len(faces), axis_names[x], axis_names[y], flip_x, flip_y))
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
# print(" Min #{} , Max #{} along '{}'".format(minmax_vert[0].index, minmax_vert[1].index, axis_names[y] ))
|
|
|
|
# print(" A1 {:.1f} , A2 {:.1f} along ".format(minmax_val[0], minmax_val[1] ))
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# Collect UV to Vert
|
|
|
|
vert_to_uv = utilities_uv.get_vert_to_uv(bm, uv_layers)
|
|
|
|
uv_to_vert = utilities_uv.get_uv_to_vert(bm, uv_layers)
|
|
|
|
|
|
|
|
processed_edges = []
|
|
|
|
edges = []
|
|
|
|
for face in faces:
|
|
|
|
for edge in face.edges:
|
|
|
|
if edge not in processed_edges:
|
|
|
|
processed_edges.append(edge)
|
2019-12-24 19:59:46 +00:00
|
|
|
delta = edge.verts[0].co - edge.verts[1].co
|
2019-12-18 20:53:16 +00:00
|
|
|
max_side = max(abs(delta.x), abs(delta.y), abs(delta.z))
|
|
|
|
|
|
|
|
# Check edges dominant in active axis
|
2019-12-24 19:59:46 +00:00
|
|
|
if(abs(delta[x]) == max_side or abs(delta[y]) == max_side):
|
|
|
|
# if( abs(delta[y]) == max_side):
|
2019-12-18 20:53:16 +00:00
|
|
|
edges.append(edge)
|
|
|
|
|
|
|
|
print("Edges {}x".format(len(edges)))
|
|
|
|
|
|
|
|
avg_angle = 0
|
|
|
|
for edge in edges:
|
2019-12-24 19:59:46 +00:00
|
|
|
uv0 = vert_to_uv[edge.verts[0]][0]
|
|
|
|
uv1 = vert_to_uv[edge.verts[1]][0]
|
2019-12-18 20:53:16 +00:00
|
|
|
delta_verts = Vector((
|
|
|
|
edge.verts[1].co[x] - edge.verts[0].co[x],
|
|
|
|
edge.verts[1].co[y] - edge.verts[0].co[y]
|
|
|
|
))
|
|
|
|
|
|
|
|
if flip_x:
|
|
|
|
delta_verts.x = -edge.verts[1].co[x] + edge.verts[0].co[x]
|
|
|
|
if flip_y:
|
|
|
|
delta_verts.y = -edge.verts[1].co[y] + edge.verts[0].co[y]
|
2019-12-24 19:59:46 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
# delta_verts.y = edge.verts[0].co[y] - edge.verts[1].co[y]
|
|
|
|
|
|
|
|
delta_uvs = Vector((
|
|
|
|
uv1.uv.x - uv0.uv.x,
|
|
|
|
uv1.uv.y - uv0.uv.y
|
|
|
|
))
|
|
|
|
a0 = math.atan2(delta_verts.y, delta_verts.x) - math.pi/2
|
|
|
|
a1 = math.atan2(delta_uvs.y, delta_uvs.x) - math.pi/2
|
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
a_delta = math.atan2(math.sin(a0-a1), math.cos(a0-a1))
|
2019-12-18 20:53:16 +00:00
|
|
|
# edge.verts[0].index, edge.verts[1].index
|
|
|
|
# print(" turn {:.1f} .. {:.1f} , {:.1f}".format(a_delta*180/math.pi, a0*180/math.pi,a1*180/math.pi))
|
2019-12-24 19:59:46 +00:00
|
|
|
avg_angle += a_delta
|
2019-12-18 20:53:16 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
avg_angle /= len(edges) # - math.pi/2
|
2019-12-18 20:53:16 +00:00
|
|
|
print("Turn {:.1f}".format(avg_angle * 180/math.pi))
|
2019-12-24 19:59:46 +00:00
|
|
|
|
2019-12-18 20:53:16 +00:00
|
|
|
bpy.ops.uv.select_all(action='DESELECT')
|
|
|
|
for face in faces:
|
|
|
|
for loop in face.loops:
|
|
|
|
loop[uv_layers].select = True
|
|
|
|
|
|
|
|
bpy.context.tool_settings.transform_pivot_point = 'MEDIAN_POINT'
|
|
|
|
bpy.ops.transform.rotate(value=avg_angle, orient_axis='Z')
|
|
|
|
# bpy.ops.transform.rotate(value=0.58191, axis=(-0, -0, -1), constraint_axis=(False, False, False), orient_type='GLOBAL', mirror=False, use_proportional_edit=False, proportional_edit_falloff='SPHERE', proportional_size=0.0267348)
|
|
|
|
|
|
|
|
# processed = []
|
|
|
|
|
|
|
|
'''
|
|
|
|
bpy.ops.uv.select_all(action='DESELECT')
|
|
|
|
for face in faces:
|
|
|
|
|
|
|
|
# Collect UV to Vert
|
|
|
|
for loop in face.loops:
|
|
|
|
loop[uv_layers].select = True
|
|
|
|
vert = loop.vert
|
|
|
|
uv = loop[uv_layers]
|
|
|
|
# vert_to_uv
|
|
|
|
if vert not in vert_to_uv:
|
|
|
|
vert_to_uv[vert] = [uv];
|
|
|
|
else:
|
|
|
|
vert_to_uv[vert].append(uv)
|
|
|
|
# uv_to_vert
|
|
|
|
if uv not in uv_to_vert:
|
|
|
|
uv_to_vert[ uv ] = vert;
|
|
|
|
|
|
|
|
|
|
|
|
for vert in face.verts:
|
|
|
|
if vert not in processed:
|
|
|
|
processed.append(vert)
|
|
|
|
|
|
|
|
vert_y = (vert.co)[y] #obj.matrix_world *
|
|
|
|
|
|
|
|
print("idx {} = {}".format(vert.index, vert_y))
|
|
|
|
|
|
|
|
if not minmax_vert[0] or not minmax_vert[1]:
|
|
|
|
minmax_vert[0] = vert
|
|
|
|
minmax_vert[1] = vert
|
|
|
|
minmax_val[0] = vert_y
|
|
|
|
minmax_val[1] = vert_y
|
|
|
|
continue
|
|
|
|
|
|
|
|
if vert_y < minmax_val[0]:
|
|
|
|
# Not yet defined or smaller
|
|
|
|
minmax_vert[0] = vert
|
|
|
|
minmax_val[0] = vert_y
|
|
|
|
|
|
|
|
elif vert_y > minmax_val[1]:
|
|
|
|
minmax_vert[1] = vert
|
|
|
|
minmax_val[1] = vert_y
|
|
|
|
|
|
|
|
|
|
|
|
if minmax_vert[0] and minmax_vert[1]:
|
|
|
|
axis_names = ['x', 'y', 'z']
|
|
|
|
print(" Min #{} , Max #{} along '{}'".format(minmax_vert[0].index, minmax_vert[1].index, axis_names[y] ))
|
|
|
|
# print(" A1 {:.1f} , A2 {:.1f} along ".format(minmax_val[0], minmax_val[1] ))
|
|
|
|
|
|
|
|
vert_A = minmax_vert[0]
|
|
|
|
vert_B = minmax_vert[1]
|
|
|
|
uv_A = vert_to_uv[vert_A][0]
|
|
|
|
uv_B = vert_to_uv[vert_B][0]
|
|
|
|
|
|
|
|
delta_verts = Vector((
|
|
|
|
vert_B.co[x] - vert_A.co[x],
|
|
|
|
vert_B.co[y] - vert_A.co[y]
|
|
|
|
))
|
|
|
|
|
|
|
|
delta_uvs = Vector((
|
|
|
|
uv_B.uv.x - uv_A.uv.x,
|
|
|
|
uv_B.uv.y - uv_A.uv.y,
|
|
|
|
|
|
|
|
))
|
|
|
|
# Get angles
|
|
|
|
angle_vert = math.atan2(delta_verts.y, delta_verts.x) - math.pi/2
|
|
|
|
angle_uv = math.atan2(delta_uvs.y, delta_uvs.x) - math.pi/2
|
|
|
|
|
|
|
|
angle_delta = math.atan2(math.sin(angle_vert-angle_uv), math.cos(angle_vert-angle_uv))
|
|
|
|
|
|
|
|
print(" Angles {:.2f} | {:.2f}".format(angle_vert*180/math.pi, angle_uv*180/math.pi))
|
|
|
|
print(" Angle Diff {:.2f}".format(angle_delta*180/math.pi))
|
|
|
|
|
|
|
|
bpy.context.tool_settings.transform_pivot_point = 'MEDIAN_POINT'
|
|
|
|
bpy.ops.transform.rotate(value=angle_delta, axis='Z')
|
|
|
|
# bpy.ops.transform.rotate(value=0.58191, axis=(-0, -0, -1), constraint_axis=(False, False, False), orient_type='GLOBAL', mirror=False, use_proportional_edit=False, proportional_edit_falloff='SPHERE', proportional_size=0.0267348)
|
|
|
|
|
|
|
|
|
|
|
|
# bpy.ops.mesh.select_all(action='DESELECT')
|
|
|
|
# vert_A.select = True
|
|
|
|
# vert_B.select = True
|
|
|
|
|
|
|
|
# return
|
|
|
|
'''
|
2019-06-08 23:42:50 +00:00
|
|
|
|
2019-12-24 19:59:46 +00:00
|
|
|
|
2019-06-08 23:42:50 +00:00
|
|
|
bpy.utils.register_class(op)
|