Add files via upload
This commit is contained in:
		| @@ -0,0 +1,477 @@ | ||||
| # ##### BEGIN GPL LICENSE BLOCK ##### | ||||
| # | ||||
| #  This program is free software; you can redistribute it and/or | ||||
| #  modify it under the terms of the GNU General Public License | ||||
| #  as published by the Free Software Foundation; either version 2 | ||||
| #  of the License, or (at your option) any later version. | ||||
| # | ||||
| #  This program is distributed in the hope that it will be useful, | ||||
| #  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| #  GNU General Public License for more details. | ||||
| # | ||||
| #  You should have received a copy of the GNU General Public License | ||||
| #  along with this program; if not, write to the Free Software Foundation, | ||||
| #  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
| # | ||||
| # ##### END GPL LICENSE BLOCK ##### | ||||
| # --------------------------- LATTICE ALONG SURFACE -------------------------- # | ||||
| # -------------------------------- version 0.3 ------------------------------- # | ||||
| #                                                                              # | ||||
| # Automatically generate and assign a lattice that follows the active surface. # | ||||
| #                                                                              # | ||||
| #                        (c)   Alessandro Zomparelli                           # | ||||
| #                                    (2017)                                    # | ||||
| #                                                                              # | ||||
| # http://www.co-de-it.com/                                                     # | ||||
| #                                                                              # | ||||
| # ############################################################################ # | ||||
|  | ||||
| import bpy | ||||
| import bmesh | ||||
| from bpy.types import Operator | ||||
| from bpy.props import (BoolProperty, StringProperty, FloatProperty) | ||||
| from mathutils import Vector | ||||
|  | ||||
| from .utils import * | ||||
|  | ||||
|  | ||||
| def not_in(element, grid): | ||||
|     output = True | ||||
|     for loop in grid: | ||||
|         if element in loop: | ||||
|             output = False | ||||
|             break | ||||
|     return output | ||||
|  | ||||
|  | ||||
| def grid_from_mesh(mesh, swap_uv): | ||||
|     bm = bmesh.new() | ||||
|     bm.from_mesh(mesh) | ||||
|     verts_grid = [] | ||||
|     edges_grid = [] | ||||
|     faces_grid = [] | ||||
|  | ||||
|     running_grid = True | ||||
|     while running_grid: | ||||
|         verts_loop = [] | ||||
|         edges_loop = [] | ||||
|         faces_loop = [] | ||||
|  | ||||
|         # storing first point | ||||
|         verts_candidates = [] | ||||
|         if len(faces_grid) == 0: | ||||
|             # for first loop check all vertices | ||||
|             verts_candidates = bm.verts | ||||
|         else: | ||||
|             # for other loops start form the vertices of the first face | ||||
|             # the last loop, skipping already used vertices | ||||
|             verts_candidates = [v for v in bm.faces[faces_grid[-1][0]].verts if not_in(v.index, verts_grid)] | ||||
|  | ||||
|         # check for last loop | ||||
|         is_last = False | ||||
|         for vert in verts_candidates: | ||||
|             if len(vert.link_faces) == 1:   # check if corner vertex | ||||
|                 vert.select = True | ||||
|                 verts_loop.append(vert.index) | ||||
|                 is_last = True | ||||
|                 break | ||||
|  | ||||
|         if not is_last: | ||||
|             for vert in verts_candidates: | ||||
|                 new_link_faces = [f for f in vert.link_faces if not_in(f.index, faces_grid)] | ||||
|                 if len(new_link_faces) < 2:  # check if corner vertex | ||||
|                     vert.select = True | ||||
|                     verts_loop.append(vert.index) | ||||
|                     break | ||||
|  | ||||
|         running_loop = len(verts_loop) > 0 | ||||
|  | ||||
|         while running_loop: | ||||
|             bm.verts.ensure_lookup_table() | ||||
|             id = verts_loop[-1] | ||||
|             link_edges = bm.verts[id].link_edges | ||||
|             # storing second point | ||||
|             if len(verts_loop) == 1:            # only one vertex stored in the loop | ||||
|                 if len(faces_grid) == 0:        # first loop # | ||||
|                     edge = link_edges[swap_uv]  # chose direction | ||||
|                     for vert in edge.verts: | ||||
|                         if vert.index != id: | ||||
|                             vert.select = True | ||||
|                             verts_loop.append(vert.index)                # new vertex | ||||
|                             edges_loop.append(edge.index)                # chosen edge | ||||
|                             faces_loop.append(edge.link_faces[0].index)  # only one face | ||||
|                             # edge.link_faces[0].select = True | ||||
|                 else:  # other loops # | ||||
|                     # start from the edges of the first face of the last loop | ||||
|                     for edge in bm.faces[faces_grid[-1][0]].edges: | ||||
|                         # chose an edge starting from the first vertex that is not returning back | ||||
|                         if bm.verts[verts_loop[0]] in edge.verts and \ | ||||
|                                 bm.verts[verts_grid[-1][0]] not in edge.verts: | ||||
|                             for vert in edge.verts: | ||||
|                                 if vert.index != id: | ||||
|                                     vert.select = True | ||||
|                                     verts_loop.append(vert.index) | ||||
|                             edges_loop.append(edge.index) | ||||
|  | ||||
|                             for face in edge.link_faces: | ||||
|                                 if not_in(face.index, faces_grid): | ||||
|                                     faces_loop.append(face.index) | ||||
|             # continuing the loop | ||||
|             else: | ||||
|                 for edge in link_edges: | ||||
|                     for vert in edge.verts: | ||||
|                         store_data = False | ||||
|                         if not_in(vert.index, verts_grid) and vert.index not in verts_loop: | ||||
|                             if len(faces_loop) > 0: | ||||
|                                 bm.faces.ensure_lookup_table() | ||||
|                                 if vert not in bm.faces[faces_loop[-1]].verts: | ||||
|                                     store_data = True | ||||
|                             else: | ||||
|                                 store_data = True | ||||
|                             if store_data: | ||||
|                                 vert.select = True | ||||
|                                 verts_loop.append(vert.index) | ||||
|                                 edges_loop.append(edge.index) | ||||
|                                 for face in edge.link_faces: | ||||
|                                     if not_in(face.index, faces_grid): | ||||
|                                         faces_loop.append(face.index) | ||||
|                                 break | ||||
|             # ending condition | ||||
|             if verts_loop[-1] == id or verts_loop[-1] == verts_loop[0]: | ||||
|                 running_loop = False | ||||
|  | ||||
|         verts_grid.append(verts_loop) | ||||
|         edges_grid.append(edges_loop) | ||||
|         faces_grid.append(faces_loop) | ||||
|  | ||||
|         if len(faces_loop) == 0: | ||||
|             running_grid = False | ||||
|  | ||||
|     return verts_grid, edges_grid, faces_grid | ||||
|  | ||||
|  | ||||
| class lattice_along_surface(Operator): | ||||
|     bl_idname = "object.lattice_along_surface" | ||||
|     bl_label = "Lattice along Surface" | ||||
|     bl_description = ("Automatically add a Lattice modifier to the selected " | ||||
|                       "object, adapting it to the active one.\nThe active " | ||||
|                       "object must be a rectangular grid compatible with the " | ||||
|                       "Lattice's topology") | ||||
|     bl_options = {'REGISTER', 'UNDO'} | ||||
|  | ||||
|     set_parent : BoolProperty( | ||||
|             name="Set Parent", | ||||
|             default=True, | ||||
|             description="Automatically set the Lattice as parent" | ||||
|             ) | ||||
|     flipNormals : BoolProperty( | ||||
|             name="Flip Normals", | ||||
|             default=False, | ||||
|             description="Flip normals direction" | ||||
|             ) | ||||
|     swapUV : BoolProperty( | ||||
|             name="Swap UV", | ||||
|             default=False, | ||||
|             description="Flip grid's U and V" | ||||
|             ) | ||||
|     flipU : BoolProperty( | ||||
|             name="Flip U", | ||||
|             default=False, | ||||
|             description="Flip grid's U") | ||||
|  | ||||
|     flipV : BoolProperty( | ||||
|             name="Flip V", | ||||
|             default=False, | ||||
|             description="Flip grid's V" | ||||
|             ) | ||||
|     flipW : BoolProperty( | ||||
|             name="Flip W", | ||||
|             default=False, | ||||
|             description="Flip grid's W" | ||||
|             ) | ||||
|     use_groups : BoolProperty( | ||||
|             name="Vertex Group", | ||||
|             default=False, | ||||
|             description="Use active Vertex Group for lattice's thickness" | ||||
|             ) | ||||
|     high_quality_lattice : BoolProperty( | ||||
|             name="High quality", | ||||
|             default=True, | ||||
|             description="Increase the the subdivisions in normal direction for a " | ||||
|                         "more correct result" | ||||
|             ) | ||||
|     hide_lattice : BoolProperty( | ||||
|             name="Hide Lattice", | ||||
|             default=True, | ||||
|             description="Automatically hide the Lattice object" | ||||
|             ) | ||||
|     scale_x : FloatProperty( | ||||
|             name="Scale X", | ||||
|             default=1, | ||||
|             min=0.001, | ||||
|             max=1, | ||||
|             description="Object scale" | ||||
|             ) | ||||
|     scale_y : FloatProperty( | ||||
|             name="Scale Y", default=1, | ||||
|             min=0.001, | ||||
|             max=1, | ||||
|             description="Object scale" | ||||
|             ) | ||||
|     scale_z : FloatProperty( | ||||
|             name="Scale Z", | ||||
|             default=1, | ||||
|             min=0.001, | ||||
|             max=1, | ||||
|             description="Object scale" | ||||
|             ) | ||||
|     thickness : FloatProperty( | ||||
|             name="Thickness", | ||||
|             default=1, | ||||
|             soft_min=0, | ||||
|             soft_max=5, | ||||
|             description="Lattice thickness" | ||||
|             ) | ||||
|     displace : FloatProperty( | ||||
|             name="Displace", | ||||
|             default=0, | ||||
|             soft_min=-1, | ||||
|             soft_max=1, | ||||
|             description="Lattice displace" | ||||
|             ) | ||||
|     grid_object = "" | ||||
|     source_object = "" | ||||
|  | ||||
|     @classmethod | ||||
|     def poll(cls, context): | ||||
|         try: return bpy.context.object.mode == 'OBJECT' | ||||
|         except: return False | ||||
|  | ||||
|     def draw(self, context): | ||||
|         layout = self.layout | ||||
|         col = layout.column(align=True) | ||||
|         col.label(text="Thickness:") | ||||
|         col.prop( | ||||
|             self, "thickness", text="Thickness", icon='NONE', expand=False, | ||||
|             slider=True, toggle=False, icon_only=False, event=False, | ||||
|             full_event=False, emboss=True, index=-1 | ||||
|             ) | ||||
|         col.prop( | ||||
|             self, "displace", text="Offset", icon='NONE', expand=False, | ||||
|             slider=True, toggle=False, icon_only=False, event=False, | ||||
|             full_event=False, emboss=True, index=-1 | ||||
|             ) | ||||
|         row = col.row() | ||||
|         row.prop(self, "use_groups") | ||||
|         col.separator() | ||||
|         col.label(text="Scale:") | ||||
|         col.prop( | ||||
|             self, "scale_x", text="U", icon='NONE', expand=False, | ||||
|             slider=True, toggle=False, icon_only=False, event=False, | ||||
|             full_event=False, emboss=True, index=-1 | ||||
|             ) | ||||
|         col.prop( | ||||
|             self, "scale_y", text="V", icon='NONE', expand=False, | ||||
|             slider=True, toggle=False, icon_only=False, event=False, | ||||
|             full_event=False, emboss=True, index=-1 | ||||
|             ) | ||||
|         col.separator() | ||||
|         col.label(text="Flip:") | ||||
|         row = col.row() | ||||
|         row.prop(self, "flipU", text="U") | ||||
|         row.prop(self, "flipV", text="V") | ||||
|         row.prop(self, "flipW", text="W") | ||||
|         col.prop(self, "swapUV") | ||||
|         col.prop(self, "flipNormals") | ||||
|         col.separator() | ||||
|         col.label(text="Lattice Options:") | ||||
|         col.prop(self, "high_quality_lattice") | ||||
|         col.prop(self, "hide_lattice") | ||||
|         col.prop(self, "set_parent") | ||||
|  | ||||
|     def execute(self, context): | ||||
|         if self.source_object == self.grid_object == "" or True: | ||||
|             if len(bpy.context.selected_objects) != 2: | ||||
|                 self.report({'ERROR'}, "Please, select two objects") | ||||
|                 return {'CANCELLED'} | ||||
|             grid_obj = bpy.context.object | ||||
|             if grid_obj.type not in ('MESH', 'CURVE', 'SURFACE'): | ||||
|                 self.report({'ERROR'}, "The surface object is not valid. Only Mesh," | ||||
|                             "Curve and Surface objects are allowed.") | ||||
|                 return {'CANCELLED'} | ||||
|             obj = None | ||||
|             for o in bpy.context.selected_objects: | ||||
|                 if o.name != grid_obj.name and o.type in \ | ||||
|                         ('MESH', 'CURVE', 'SURFACE', 'FONT'): | ||||
|                     obj = o | ||||
|                     o.select_set(False) | ||||
|                     break | ||||
|             try: | ||||
|                 obj_dim = obj.dimensions | ||||
|                 obj_me = simple_to_mesh(obj)#obj.to_mesh(bpy.context.depsgraph, apply_modifiers=True) | ||||
|             except: | ||||
|                 self.report({'ERROR'}, "The object to deform is not valid. Only " | ||||
|                             "Mesh, Curve, Surface and Font objects are allowed.") | ||||
|                 return {'CANCELLED'} | ||||
|             self.grid_object = grid_obj.name | ||||
|             self.source_object = obj.name | ||||
|         else: | ||||
|             grid_obj = bpy.data.objects[self.grid_object] | ||||
|             obj = bpy.data.objects[self.source_object] | ||||
|             obj_me = simple_to_mesh(obj)# obj.to_mesh(bpy.context.depsgraph, apply_modifiers=True) | ||||
|             for o in bpy.context.selected_objects: o.select_set(False) | ||||
|             grid_obj.select_set(True) | ||||
|             bpy.context.view_layer.objects.active = grid_obj | ||||
|  | ||||
|         temp_grid_obj = grid_obj.copy() | ||||
|         temp_grid_obj.data = simple_to_mesh(grid_obj) | ||||
|         grid_mesh = temp_grid_obj.data | ||||
|         for v in grid_mesh.vertices: | ||||
|             v.co = grid_obj.matrix_world @ v.co | ||||
|         grid_mesh.calc_normals() | ||||
|  | ||||
|         if len(grid_mesh.polygons) > 64 * 64: | ||||
|             bpy.data.objects.remove(temp_grid_obj) | ||||
|             bpy.context.view_layer.objects.active = obj | ||||
|             obj.select_set(True) | ||||
|             self.report({'ERROR'}, "Maximum resolution allowed for Lattice is 64") | ||||
|             return {'CANCELLED'} | ||||
|  | ||||
|         # CREATING LATTICE | ||||
|         min = Vector((0, 0, 0)) | ||||
|         max = Vector((0, 0, 0)) | ||||
|         first = True | ||||
|         for v in obj_me.vertices: | ||||
|             v0 = v.co.copy() | ||||
|             vert = obj.matrix_world @ v0 | ||||
|             if vert[0] < min[0] or first: | ||||
|                 min[0] = vert[0] | ||||
|             if vert[1] < min[1] or first: | ||||
|                 min[1] = vert[1] | ||||
|             if vert[2] < min[2] or first: | ||||
|                 min[2] = vert[2] | ||||
|             if vert[0] > max[0] or first: | ||||
|                 max[0] = vert[0] | ||||
|             if vert[1] > max[1] or first: | ||||
|                 max[1] = vert[1] | ||||
|             if vert[2] > max[2] or first: | ||||
|                 max[2] = vert[2] | ||||
|             first = False | ||||
|  | ||||
|         bb = max - min | ||||
|         lattice_loc = (max + min) / 2 | ||||
|         bpy.ops.object.add(type='LATTICE') | ||||
|         lattice = bpy.context.active_object | ||||
|         lattice.location = lattice_loc | ||||
|         lattice.scale = Vector((bb.x / self.scale_x, bb.y / self.scale_y, | ||||
|                                 bb.z / self.scale_z)) | ||||
|  | ||||
|         if bb.x == 0: | ||||
|             lattice.scale.x = 1 | ||||
|         if bb.y == 0: | ||||
|             lattice.scale.y = 1 | ||||
|         if bb.z == 0: | ||||
|             lattice.scale.z = 1 | ||||
|  | ||||
|         bpy.context.view_layer.objects.active = obj | ||||
|         bpy.ops.object.modifier_add(type='LATTICE') | ||||
|         obj.modifiers[-1].object = lattice | ||||
|  | ||||
|         # set as parent | ||||
|         if self.set_parent: | ||||
|             obj.select_set(True) | ||||
|             lattice.select_set(True) | ||||
|             bpy.context.view_layer.objects.active = lattice | ||||
|             bpy.ops.object.parent_set(type='LATTICE') | ||||
|  | ||||
|         # reading grid structure | ||||
|         verts_grid, edges_grid, faces_grid = grid_from_mesh( | ||||
|                                                 grid_mesh, | ||||
|                                                 swap_uv=self.swapUV | ||||
|                                                 ) | ||||
|         nu = len(verts_grid) | ||||
|         nv = len(verts_grid[0]) | ||||
|         nw = 2 | ||||
|         scale_normal = self.thickness | ||||
|  | ||||
|         try: | ||||
|             lattice.data.points_u = nu | ||||
|             lattice.data.points_v = nv | ||||
|             lattice.data.points_w = nw | ||||
|             for i in range(nu): | ||||
|                 for j in range(nv): | ||||
|                     for w in range(nw): | ||||
|                         if self.use_groups: | ||||
|                             try: | ||||
|                                 displace = temp_grid_obj.vertex_groups.active.weight( | ||||
|                                                     verts_grid[i][j]) * scale_normal * bb.z | ||||
|                             except: | ||||
|                                 displace = 0#scale_normal * bb.z | ||||
|                         else: | ||||
|                             displace = scale_normal * bb.z | ||||
|                         target_point = (grid_mesh.vertices[verts_grid[i][j]].co + | ||||
|                                         grid_mesh.vertices[verts_grid[i][j]].normal * | ||||
|                                         (w + self.displace / 2 - 0.5) * displace) - lattice.location | ||||
|                         if self.flipW: | ||||
|                             w = 1 - w | ||||
|                         if self.flipU: | ||||
|                             i = nu - i - 1 | ||||
|                         if self.flipV: | ||||
|                             j = nv - j - 1 | ||||
|  | ||||
|                         lattice.data.points[i + j * nu + w * nu * nv].co_deform.x = \ | ||||
|                                 target_point.x / bpy.data.objects[lattice.name].scale.x | ||||
|                         lattice.data.points[i + j * nu + w * nu * nv].co_deform.y = \ | ||||
|                                 target_point.y / bpy.data.objects[lattice.name].scale.y | ||||
|                         lattice.data.points[i + j * nu + w * nu * nv].co_deform.z = \ | ||||
|                                 target_point.z / bpy.data.objects[lattice.name].scale.z | ||||
|  | ||||
|         except: | ||||
|             bpy.ops.object.mode_set(mode='OBJECT') | ||||
|             temp_grid_obj.select_set(True) | ||||
|             lattice.select_set(True) | ||||
|             obj.select_set(False) | ||||
|             bpy.ops.object.delete(use_global=False) | ||||
|             bpy.context.view_layer.objects.active = obj | ||||
|             obj.select_set(True) | ||||
|             bpy.ops.object.modifier_remove(modifier=obj.modifiers[-1].name) | ||||
|             if nu > 64 or nv > 64: | ||||
|                 self.report({'ERROR'}, "Maximum resolution allowed for Lattice is 64") | ||||
|                 return {'CANCELLED'} | ||||
|             else: | ||||
|                 self.report({'ERROR'}, "The grid mesh is not correct") | ||||
|                 return {'CANCELLED'} | ||||
|  | ||||
|         bpy.ops.object.mode_set(mode='OBJECT') | ||||
|         #grid_obj.select_set(True) | ||||
|         #lattice.select_set(False) | ||||
|         obj.select_set(False) | ||||
|         #bpy.ops.object.delete(use_global=False) | ||||
|         bpy.context.view_layer.objects.active = lattice | ||||
|         lattice.select_set(True) | ||||
|  | ||||
|         if self.high_quality_lattice: | ||||
|             bpy.context.object.data.points_w = 8 | ||||
|         else: | ||||
|             bpy.context.object.data.use_outside = True | ||||
|  | ||||
|         if self.hide_lattice: | ||||
|             bpy.ops.object.hide_view_set(unselected=False) | ||||
|  | ||||
|         bpy.context.view_layer.objects.active = obj | ||||
|         obj.select_set(True) | ||||
|         lattice.select_set(False) | ||||
|  | ||||
|         if self.flipNormals: | ||||
|             try: | ||||
|                 bpy.ops.object.mode_set(mode='EDIT') | ||||
|                 bpy.ops.mesh.select_all(action='SELECT') | ||||
|                 bpy.ops.mesh.flip_normals() | ||||
|                 bpy.ops.object.mode_set(mode='OBJECT') | ||||
|             except: | ||||
|                 pass | ||||
|         bpy.data.meshes.remove(grid_mesh) | ||||
|         bpy.data.meshes.remove(obj_me) | ||||
|  | ||||
|         return {'FINISHED'} | ||||
		Reference in New Issue
	
	Block a user