如何使用Blender脚本删除场景中较小的多个对象?

时间:2019-11-06 21:43:12

标签: python blender mesh

我正在使用Blender 2.8。我想将一个对象导入到由未连接的几部分组成的搅拌器中。所以我想将对象分割开,只导出最大的部分。

因此,可以说一个物体有3个碎片,一个大而两个小。我可以将这个对象变成三个对象,每个对象都包含其中一个片段。我想删除两个较小的对象,而只保留最大的一个。我在想也许以某种方式找到三个不同对象的表面积,并仅保留最大的面积,而删除所有其他对象?我在Blender很新。

bpy.ops.import_mesh.stl(filepath='path/of/file.stl')
bpy.ops.mesh.separate(type='LOOSE')
amount_of_pieces = len(context.selected_objects)

if amount_of_pieces > 1:
    highest_surface_area = 0

    #the rest is pseudocode
    for object in scene:
        if object.area > highest_surface_area:
            highest_surface_area = object.area
        else:
            bpy.ops.object.delete()

bpy.ops.export_mesh.stl(filepath='path/of/new/file.stl')

2 个答案:

答案 0 :(得分:1)

步骤将是:-

  • 导入文件
  • 分为多个对象
  • 为了安全起见,获取网格物体列表
  • 列出每个物体的表面积
  • 从区域列表中获取最大值
  • 删除不大的对象
  • 出口最大的
  • 清理

我们不需要使用bmesh来获得表面积,常规网格数据包括polygon.area

使用list comprehension,我们可以将大多数步骤都排成一行。

import bpy

# import and separate
file = (r'path/of/file.stl')
bpy.ops.import_mesh.stl(filepath= file)
bpy.ops.mesh.separate(type='LOOSE')

# list of mesh objects
mesh_objs = [o for o in bpy.context.scene.objects
                if o.type == 'MESH']

# dict with surface area of each object
obj_areas = {o:sum([f.area for f in o.data.polygons])
                for o in mesh_objs}

# which is biggest
big_obj = max(obj_areas, key=obj_areas.get)

# select and delete not biggest
[o.select_set(o is not big_obj) for o in mesh_objs]
bpy.ops.object.delete(use_global=False, confirm=False)

#export
bpy.ops.export_mesh.stl(filepath= 'path/of/new/file.stl')

# cleanup
bpy.ops.object.select_all(action='SELECT')
bpy.ops.object.delete(use_global=False, confirm=False)

答案 1 :(得分:0)

我能够编写一个适用于此的代码,但是它很长而且很混乱。如果有人可以给我一些清理建议,我将不胜感激。

import bpy
import os
import bmesh


context = bpy.context

file = (r'path\to\file.stl')

bpy.ops.import_mesh.stl(filepath= file)
fileName = os.path.basename(file)[:-4].capitalize()
bpy.ops.mesh.separate(type='LOOSE')
bpy.ops.object.select_all(action='SELECT')
piece = len(context.selected_objects)
bpy.ops.object.select_all(action='DESELECT')

high = 0
if piece > 1:
    bpy.data.objects[fileName].select_set(True)
    obj = bpy.context.active_object
    bm = bmesh.new()
    bm.from_mesh(obj.data)
    area = sum(f.calc_area() for f in bm.faces)
    high = area
    bm.free()
    bpy.ops.object.select_all(action='DESELECT')

    for x in range (1, piece):
        name = fileName + '.00' + str(x)
        object = bpy.data.objects[name]
        context.view_layer.objects.active = object
        bpy.data.objects[name].select_set(True)
        obj = bpy.context.active_object
        bm = bmesh.new()
        bm.from_mesh(obj.data)
        newArea = sum(f.calc_area() for f in bm.faces)
        bm.free()
        if newArea > high:
            high = newArea
            bpy.ops.object.select_all(action='DESELECT')
        else:
            bpy.ops.object.delete()
        bpy.ops.object.select_all(action='DESELECT')

    if area != high:
        bpy.data.objects[fileName].select_set(True)
        bpy.ops.object.delete()


bpy.ops.export_mesh.stl(filepath= 'path/to/export/file.stl')
bpy.ops.object.select_all(action='SELECT')
bpy.ops.object.delete(use_global=False, confirm=False)