128 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import bpy,mathutils
 | |
| from . import utils
 | |
| 
 | |
| class SuperAlignOperator(bpy.types.Operator):
 | |
|     """Align object with 3ds Max way"""
 | |
|     bl_idname = "ballance.super_align"
 | |
|     bl_label = "Super Align"
 | |
|     bl_options = {'UNDO'}
 | |
| 
 | |
|     align_x: bpy.props.BoolProperty(name="X position")
 | |
|     align_y: bpy.props.BoolProperty(name="Y position")
 | |
|     align_z: bpy.props.BoolProperty(name="Z position")
 | |
| 
 | |
|     current_references: bpy.props.EnumProperty(
 | |
|         name="Current",
 | |
|         items=(('MIN', "Min", ""),
 | |
|                 ('CENTER', "Center (bound box)", ""),
 | |
|                 ('POINT', "Center (axis)", ""),
 | |
|                 ('MAX', "Max", "")
 | |
|                 ),
 | |
|         )
 | |
| 
 | |
|     target_references: bpy.props.EnumProperty(
 | |
|         name="Target",
 | |
|         items=(('MIN', "Min", ""),
 | |
|                 ('CENTER', "Center (bound box)", ""),
 | |
|                 ('POINT', "Center (axis)", ""),
 | |
|                 ('MAX', "Max", "")
 | |
|                 ),
 | |
|         )
 | |
| 
 | |
|     @classmethod
 | |
|     def poll(self, context):
 | |
|         return check_align_target()
 | |
| 
 | |
|     def execute(self, context):
 | |
|         align_object(self.align_x, self.align_y, self.align_z, self.current_references, self.target_references)
 | |
|         return {'FINISHED'}
 | |
| 
 | |
|     def invoke(self, context, event):
 | |
|         wm = context.window_manager
 | |
|         return wm.invoke_props_dialog(self)
 | |
| 
 | |
|     def draw(self, context):
 | |
|         layout = self.layout
 | |
|         col = layout.column()
 | |
|         col.label(text="Align axis")
 | |
| 
 | |
|         row = col.row()
 | |
|         row.prop(self, "align_x")
 | |
|         row.prop(self, "align_y")
 | |
|         row.prop(self, "align_z")
 | |
| 
 | |
|         col.prop(self, "current_references")
 | |
|         col.prop(self, "target_references")
 | |
| 
 | |
| # ============================== method
 | |
| 
 | |
| def check_align_target():
 | |
|     if bpy.context.active_object is None:
 | |
|         return False
 | |
| 
 | |
|     selected = bpy.context.selected_objects[:]
 | |
|     length = len(selected)
 | |
|     if bpy.context.active_object in selected:
 | |
|         length -= 1
 | |
|     if length == 0:
 | |
|         return False
 | |
|     
 | |
|     return True
 | |
| 
 | |
| def align_object(use_x, use_y, use_z, currentMode, targetMode):
 | |
|     if not (use_x or use_y or use_z):
 | |
|         return
 | |
| 
 | |
|     # calc active object data
 | |
|     currentObj = bpy.context.active_object
 | |
|     currentObjBbox = [currentObj.matrix_world @ mathutils.Vector(corner) for corner in currentObj.bound_box]
 | |
|     currentObjRef = provideObjRefPoint(currentObj, currentObjBbox, currentMode)
 | |
| 
 | |
|     # calc target
 | |
|     targetObjList = bpy.context.selected_objects[:]
 | |
|     if currentObj in targetObjList:
 | |
|         targetObjList.remove(currentObj)
 | |
|         
 | |
|     # process each obj
 | |
|     for targetObj in targetObjList:
 | |
|         targetObjBbox = [targetObj.matrix_world @ mathutils.Vector(corner) for corner in targetObj.bound_box]
 | |
|         targetObjRef = provideObjRefPoint(targetObj, targetObjBbox, targetMode)
 | |
| 
 | |
|         if use_x:
 | |
|             targetObj.location.x += currentObjRef.x - targetObjRef.x
 | |
|         if use_y:
 | |
|             targetObj.location.y += currentObjRef.y - targetObjRef.y
 | |
|         if use_z:
 | |
|             targetObj.location.z += currentObjRef.z - targetObjRef.z
 | |
| 
 | |
| def provideObjRefPoint(obj, vecList, mode):
 | |
|     refPoint = mathutils.Vector((0, 0, 0))
 | |
| 
 | |
|     if (mode == 'MIN'):
 | |
|         refPoint.x = min([vec.x for vec in vecList])
 | |
|         refPoint.y = min([vec.y for vec in vecList])
 | |
|         refPoint.z = min([vec.z for vec in vecList])
 | |
|     elif (mode == 'MAX'):
 | |
|         refPoint.x = max([vec.x for vec in vecList])
 | |
|         refPoint.y = max([vec.y for vec in vecList])
 | |
|         refPoint.z = max([vec.z for vec in vecList])
 | |
|     elif (mode == 'CENTER'):
 | |
|         maxVecCache = mathutils.Vector((0, 0, 0))
 | |
|         minVecCache = mathutils.Vector((0, 0, 0))
 | |
| 
 | |
|         minVecCache.x = min([vec.x for vec in vecList])
 | |
|         minVecCache.y = min([vec.y for vec in vecList])
 | |
|         minVecCache.z = min([vec.z for vec in vecList])
 | |
|         maxVecCache.x = max([vec.x for vec in vecList])
 | |
|         maxVecCache.y = max([vec.y for vec in vecList])
 | |
|         maxVecCache.z = max([vec.z for vec in vecList])
 | |
| 
 | |
|         refPoint.x = (maxVecCache.x + minVecCache.x) / 2
 | |
|         refPoint.y = (maxVecCache.y + minVecCache.y) / 2
 | |
|         refPoint.z = (maxVecCache.z + minVecCache.z) / 2
 | |
|     else:
 | |
|         refPoint.x = obj.location.x
 | |
|         refPoint.y = obj.location.y
 | |
|         refPoint.z = obj.location.z
 | |
| 
 | |
|     return refPoint |