Одним из методов аналитической оценки меша является оценка его триангулированной геометрии. Например, подобная оценка проводится при поиске полигонов, которые перекрываются, как в самом меше, так и в его UV-развертке. Для того чтобы вернуть результат, полученный после анализа триангулированного меша обратно в исходный меш, нужно знать точное соответствие между исходным полигоном и полученными из него треугольниками.
Сам по себе оператор bmesh.ops.triangulate(), при помощи которого проще всего триангулировать меш, предоставляет определенный инструмент для поиска такого соответствия, однако его стоит немного доработать.
Для примера из текущего активного меша создадим bmesh и приведем в соответствие индексы для его точек, ребер и полигонов. Так же сделаем его копию, она нам пригодится немного позднее.
1 2 3 4 5 6 7 |
bm = bmesh.from_edit_mesh(bpy.context.object.data) bm.verts.ensure_lookup_table() bm.edges.ensure_lookup_table() bm.faces.ensure_lookup_table() bm_copy = bm.copy() |
Теперь мы можем триангулировать полученный bmesh.
1 2 3 4 5 6 |
t_mesh = bmesh.ops.triangulate( bm, faces=bm.faces, quad_method='BEAUTY', ngon_method='BEAUTY' ) |
Вся работа по триангуляции производится сразу в bmesh объекте, однако для дальнейших действий нам также нужен результат, возвращаемый функцией триангуляции.
Функция bmesh.ops.triangulate() возвращает словарь, состоящий в свою очередь из четырех словарей: “edges”, “faces”, “face_map” и “face_map_double”.
Из них нам наиболее полезен будет словарь “face_map”. Этот словарь имеет следующий формат: {треугольник: исходный полигон, треугольник: исходный полигон, …}.
1 2 3 |
print(t_mesh['face_map']) # {<BMFace(0x000002315485B118), index=447, totverts=3>: <BMFace(0x000002315485B118), index=447, totverts=3>, ...} |
Такой словарь дает нам возможность проводить соответствие треугольник-полигон со стороны полученного триангулированного меша, однако гораздо полезнее иметь возможность получать набор полученных треугольников отталкиваясь от исходного полигона.
Мы можем получить список вида ((треугольник, полигон), (треугольник, полигон), …) при помощи функции items().
1 2 3 |
print(t_mesh['face_map'].items()) # [(<BMFace(0x000002315EFC36B8), index=67, totverts=3>, <BMFace(0x000002315EFC36B8), index=67, totverts=3>), ...] |
Однако такой формат тоже не дает возможности быстро получить нужные данные.
Давайте инвертируем словарь “face_map”.
1 2 3 |
inv_map = {} for k, v in t_mesh['face_map'].items(): inv_map[v] = inv_map.get(v, []) + [k] |
И теперь мы уже получаем что-то похожее на то что нам нужно – через функцию items() мы получаем список вида ((исходный полигон, [треугольник, треугольник]), …).
1 2 3 |
print(inv_map.items()) # (<BMFace(0x0000023162F9ADB8), index=363, totverts=3>, [<BMFace(0x0000023162F9ADB8), index=363, totverts=3>, <BMFace(0x0000023154551258), index=839, totverts=3>], ...) |
Однако здесь есть еще одна особенность – если в исходном меше изначально присутствовали треугольные полигоны, они, что логично, не были триангулированы, и не попали в итоговый список. Мы можем проверить это просто сравнив длины полученного списка, списка всех полигонов исходного меша и списка треугольных полигонов в исходном меше.
1 2 3 4 5 6 7 8 |
print(len(bm_copy.faces)) # 500 print(len(inv_map)) # 468 print(len([face for face in bm_copy.faces if len(face.verts) <= 3])) # 32 |
Как мы видим, общее количество полигонов – 500 у нас может складываться из количества полигонов в инвертированной face_map – 468, и количества треугольников в исходном полигоне – 32.
Приведем инвертированную face_map к виду словаря, в котором ключами будут индексы исходных полигонов, а элементами – набор полученных из этого полигона треугольников.
1 2 3 |
face_map_idx = {key.index: value for key, value in inv_map.items()} # {498: [<BMFace(0x0000023162F9C5C0), index=498, totverts=3>, <BMFace(0x000002315F7CA2E0), index=966, totverts=3>], ...} |
И сделаем такой же словарь для недостающих треугольников из исходного меша. Как раз для этого мы сохраняли отдельную копию нетриангулированного bmesh в начале.
1 2 3 |
tris_idx = {face.index: face for face in bm_copy.faces if len(face.verts) <= 3} # {48: <BMFace(0x000002315F1B6210), index=48, totverts=3>, ...} |
Объединим оба словаря в один.
1 2 3 4 5 6 7 |
face_map = face_map_idx | tris_idx print(face_map) # {498: [<BMFace(0x0000023162F9C5C0), index=498, totverts=3>, <BMFace(0x000002315F7CA2E0), index=966, totverts=3>], ...} print(len(face_map)) # 500 |
В итоге мы получили полный словарь с ключами – индексами исходных полигонов, и элементами – треугольниками после триангуляции.
Для того чтобы быстро получить треугольники, которые были созданы из полигона с индексом, например, 322, нам нужно просто обратиться по этому индексу к нашему финальному словарю:
1 2 3 |
print(face_map[322]) # [<BMFace(0x000002315EFC5770), index=804, totverts=3>, <BMFace(0x000002315F7C91C0), index=322, totverts=3>] |
Если нам требуется сохранить триангулирование, запишем результат из bmesh обратно в исходный меш и удалим теперь уже не нужный bmesh.
1 2 |
bmesh.update_edit_mesh(bpy.context.object.data) bm.free() |