You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Blender-TexTools/op_align.py

123 lines
3.8 KiB
Python

import bpy
import bmesh
import operator
from mathutils import Vector
from collections import defaultdict
from math import pi
from . import utilities_uv
class op(bpy.types.Operator):
4 years ago
bl_idname = "uv.textools_align"
bl_label = "Align"
bl_description = "Align vertices, edges or shells"
bl_options = {'REGISTER', 'UNDO'}
4 years ago
direction: bpy.props.StringProperty(name="Direction", default="top")
4 years ago
@classmethod
def poll(cls, context):
if not bpy.context.active_object:
return False
4 years ago
# Only in Edit mode
4 years ago
if bpy.context.active_object.mode != 'EDIT':
return False
4 years ago
# Only in UV editor mode
4 years ago
if bpy.context.area.type != 'IMAGE_EDITOR':
return False
4 years ago
# Requires UV map
4 years ago
if not bpy.context.object.data.uv_layers:
4 years ago
# self.report({'WARNING'}, "Object must have more than one UV map")
return False
4 years ago
# Not in Synced mode
if bpy.context.scene.tool_settings.use_uv_select_sync:
return False
4 years ago
return True
4 years ago
def execute(self, context):
4 years ago
4 years ago
align(context, self.direction)
return {'FINISHED'}
def align(context, direction):
4 years ago
# Store selection
4 years ago
utilities_uv.selection_store()
if bpy.context.tool_settings.transform_pivot_point != 'CURSOR':
bpy.context.tool_settings.transform_pivot_point = 'CURSOR'
4 years ago
# B-Mesh
4 years ago
obj = bpy.context.active_object
4 years ago
bm = bmesh.from_edit_mesh(obj.data)
uv_layers = bm.loops.layers.uv.verify()
4 years ago
if len(obj.data.uv_layers) == 0:
print("There is no UV channel or UV data set")
return
# Collect BBox sizes
boundsAll = utilities_uv.getSelectionBBox()
mode = bpy.context.scene.tool_settings.uv_select_mode
if mode == 'FACE' or mode == 'ISLAND':
print("____ Align Islands")
4 years ago
# Collect UV islands
4 years ago
islands = utilities_uv.getSelectionIslands()
for island in islands:
4 years ago
4 years ago
bpy.ops.uv.select_all(action='DESELECT')
utilities_uv.set_selected_faces(island)
bounds = utilities_uv.getSelectionBBox()
# print("Island "+str(len(island))+"x faces, delta: "+str(delta.y))
if direction == "bottom":
4 years ago
delta = boundsAll['min'] - bounds['min']
4 years ago
bpy.ops.transform.translate(value=(0, delta.y, 0))
elif direction == "top":
delta = boundsAll['max'] - bounds['max']
bpy.ops.transform.translate(value=(0, delta.y, 0))
elif direction == "left":
4 years ago
delta = boundsAll['min'] - bounds['min']
4 years ago
bpy.ops.transform.translate(value=(delta.x, 0, 0))
elif direction == "right":
delta = boundsAll['max'] - bounds['max']
bpy.ops.transform.translate(value=(delta.x, 0, 0))
else:
print("Unkown direction: "+str(direction))
elif mode == 'EDGE' or mode == 'VERTEX':
print("____ Align Verts")
for f in bm.faces:
if f.select:
for l in f.loops:
luv = l[uv_layers]
if luv.select:
# print("Idx: "+str(luv.uv))
if direction == "top":
luv.uv[1] = boundsAll['max'].y
elif direction == "bottom":
luv.uv[1] = boundsAll['min'].y
elif direction == "left":
luv.uv[0] = boundsAll['min'].x
elif direction == "right":
luv.uv[0] = boundsAll['max'].x
bmesh.update_edit_mesh(obj.data)
4 years ago
# Restore selection
4 years ago
utilities_uv.selection_restore()
4 years ago
bpy.utils.register_class(op)