Script help: Parenting and keeping transforms.

Question Scripting

I have an idea for an add-on that I think would be very useful to my workflow, but this is my first interaction with python so I wanted to have an extra set of eyes to make sure I'm not doing anything blatantly wrong or inefficient. The goal is to eventually make it an extension, so I'm trying to make the code proper.

The first step is to add an empty in the same location as the selected object, rename it to match the object, and parent the object to it. I added for now the ability to choose a few different empty types and set the radius relative to the size of the object. 

I ran into all sorts of trouble with parenting, with the object inheriting the scale of the empty (which is why I went simpler to its radius instead), and keeping offset when the object was not in the world origin. My solution is the code below. It's all very annotated because I'm still getting used to python's syntax and it helps me remember what my intention was with each line, so I apologize if it's annoying.

I'm wondering if there's anything you would do differently to make it simpler or more straight-forward?

Let me know! And thank you!


import bpy

class EmptyAssistantOperator(bpy.types.Operator):
    """Add an empty at each selected object's origin and parent the object to it"""
    bl_idname = "object.empty_assist"
    bl_label = "Empty Assistant"
    bl_options = {'REGISTER', 'UNDO'}
    
    empty_type: bpy.props.EnumProperty(
        default='PLAIN_AXES',
        name="Empty Type",
        description="Choose the type of empty",
        items=[
        ('PLAIN_AXES', "Plain Axes", ""),
        ('CUBE', "Cube", ""),
        ('SPHERE', "Sphere", ""),
        ('CIRCLE', "Circle", "")
        ]
    )
    
    scale_factor: bpy.props.FloatProperty(
        name="Empty Scale",
        description="Multiplier for empty size relative to object",
        default=1.0,
        min=0.1,
        max=5.0
    )
    
    @classmethod
    def poll(cls, context):
        return context.selected_objects
    
    def invoke(self, context, event):
        # Opens a popup dialog for user input
        return context.window_manager.invoke_props_dialog(self)
    
    
    def execute(self, context):
        selected_objects = context.selected_objects
        
        for obj in selected_objects:
            # Get the current world location of the object
            world_location = obj.matrix_world.translation
            
            # Calculate object's bounding box dimensions
            bbox = (max(obj.dimensions)) * self.scale_factor
            
            # Create an empty at the object's world location
            bpy.ops.object.empty_add(type=self.empty_type, radius=bbox, location=world_location)
            empty = context.object # The newly created empty becomes the active object
            
            # Rename the empty to match the object's name
            empty.name = f"{obj.name}_Empty"
            
            # Parent the object to the empty
            obj.parent = empty
            
            # Update the object's location to maintain its world position (relative to the empty)
            obj.location = empty.matrix_world.inverted() @ world_location
        return {'FINISHED'}

def menu_func(self, context):
    self.layout.separator()
    self.layout.operator(EmptyAssistantOperator.bl_idname)

# Register and add to the "Object" menu
def register():
    bpy.utils.register_class(EmptyAssistantOperator)
    bpy.types.VIEW3D_MT_object.append(menu_func)

def unregister():
    bpy.utils.unregister_class(EmptyAssistantOperator)
    bpy.types.VIEW3D_MT_object.remove(menu_func)

if __name__ == "__main__":
    register()


1 love
Reply
  • Martin Bergwerf replied

    Hi Nathalia,

     "It's all very annotated because I'm still getting used to python's syntax and it helps me remember what my intention was with each line, so I apologize if it's annoying."

    I can't help you with the Code, but  a beginner mistake would be, to not write (enough) comments; it's not annoying, but good practice👍🏼

    2 loves
  • Dwayne Savage(dillenbata3) replied

    It looks good to me. You may want to use something like the following:

    for cls in classes:
            bpy.utils.register_class(cls)

    In your registering and unregistering If you plan on adding more classes. then you would just create the list in a variable called classes before the registration like:

    classes = (
        Class1name,
        Class2name,
        Class3name,
    )

    This makes it easier to add more or temporarily disable classes(Ctrl+/ to toggle comment on selected line). Also if you need to change the order that the classes are loaded you can just move them around easier. 

    2 loves
  • Nathi Tappan(nathitappan) replied

    Thanks Martin! I'll keep annotating it on, then. After staring at it for a couple of days they finally start making more sense, but the annotations sure do help to find things now that I'm moving back and forth for tweaking.

    Awesome! Thanks for the tip dillenbata3! I got to a point where I'm drawing the panel and menus so that will certainly be helpful!

    1 love