| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | # ##### 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. | 
					
						
							| 
									
										
										
										
											2009-11-21 23:55:14 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | #  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. | 
					
						
							| 
									
										
										
										
											2009-11-21 23:55:14 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | #  You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | #  along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  | #  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | # | 
					
						
							|  |  |  | # ##### END GPL LICENSE BLOCK ##### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  | # <pep8-80 compliant> | 
					
						
							| 
									
										
										
										
											2009-12-13 14:38:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | import bpy | 
					
						
							| 
									
										
										
										
											2011-08-12 06:57:00 +00:00
										 |  |  | from bpy.types import Operator | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-25 11:07:05 +11:00
										 |  |  | from bpy.props import ( | 
					
						
							|  |  |  |     EnumProperty, | 
					
						
							|  |  |  |     IntProperty, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2011-04-29 05:32:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-12 06:57:00 +00:00
										 |  |  | class MeshMirrorUV(Operator): | 
					
						
							| 
									
										
										
										
											2012-07-03 09:02:41 +00:00
										 |  |  |     """Copy mirror UV coordinates on the X axis based on a mirrored mesh""" | 
					
						
							| 
									
										
										
										
											2011-04-29 05:32:27 +00:00
										 |  |  |     bl_idname = "mesh.faces_mirror_uv" | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |     bl_label = "Copy Mirrored UV Coords" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 22:18:09 +02:00
										 |  |  |     direction: EnumProperty( | 
					
						
							| 
									
										
										
										
											2018-06-26 19:41:37 +02:00
										 |  |  |         name="Axis Direction", | 
					
						
							|  |  |  |         items=( | 
					
						
							|  |  |  |             ('POSITIVE', "Positive", ""), | 
					
						
							|  |  |  |             ('NEGATIVE', "Negative", ""), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2011-04-29 05:32:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 22:18:09 +02:00
										 |  |  |     precision: IntProperty( | 
					
						
							| 
									
										
										
										
											2018-06-26 19:41:37 +02:00
										 |  |  |         name="Precision", | 
					
						
							|  |  |  |         description=("Tolerance for finding vertex duplicates"), | 
					
						
							|  |  |  |         min=1, max=16, | 
					
						
							|  |  |  |         soft_min=1, soft_max=16, | 
					
						
							|  |  |  |         default=3, | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |     # Returns has_active_UV_layer, double_warn. | 
					
						
							|  |  |  |     def do_mesh_mirror_UV(self, mesh, DIR): | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  |         precision = self.precision | 
					
						
							|  |  |  |         double_warn = 0 | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |         if not mesh.uv_layers.active: | 
					
						
							|  |  |  |             # has_active_UV_layer, double_warn | 
					
						
							|  |  |  |             return False, 0 | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |         # mirror lookups | 
					
						
							|  |  |  |         mirror_gt = {} | 
					
						
							|  |  |  |         mirror_lt = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  |         vcos = (v.co.to_tuple(precision) for v in mesh.vertices) | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for i, co in enumerate(vcos): | 
					
						
							| 
									
										
										
										
											2012-06-29 12:41:39 +00:00
										 |  |  |             if co[0] >= 0.0: | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  |                 double_warn += co in mirror_gt | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 mirror_gt[co] = i | 
					
						
							| 
									
										
										
										
											2012-06-29 12:41:39 +00:00
										 |  |  |             if co[0] <= 0.0: | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  |                 double_warn += co in mirror_lt | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 mirror_lt[co] = i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vmap = {} | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |         for mirror_a, mirror_b in ((mirror_gt, mirror_lt), | 
					
						
							|  |  |  |                                    (mirror_lt, mirror_gt)): | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |             for co, i in mirror_a.items(): | 
					
						
							|  |  |  |                 nco = (-co[0], co[1], co[2]) | 
					
						
							|  |  |  |                 j = mirror_b.get(nco) | 
					
						
							|  |  |  |                 if j is not None: | 
					
						
							|  |  |  |                     vmap[i] = j | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |         polys = mesh.polygons | 
					
						
							|  |  |  |         loops = mesh.loops | 
					
						
							| 
									
										
										
										
											2012-04-22 23:51:50 +00:00
										 |  |  |         uv_loops = mesh.uv_layers.active.data | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |         nbr_polys = len(polys) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         mirror_pm = {} | 
					
						
							|  |  |  |         pmap = {} | 
					
						
							|  |  |  |         puvs = [None] * nbr_polys | 
					
						
							|  |  |  |         puvs_cpy = [None] * nbr_polys | 
					
						
							|  |  |  |         puvsel = [None] * nbr_polys | 
					
						
							|  |  |  |         pcents = [None] * nbr_polys | 
					
						
							|  |  |  |         vidxs = [None] * nbr_polys | 
					
						
							|  |  |  |         for i, p in enumerate(polys): | 
					
						
							|  |  |  |             lstart = lend = p.loop_start | 
					
						
							|  |  |  |             lend += p.loop_total | 
					
						
							|  |  |  |             puvs[i] = tuple(uv.uv for uv in uv_loops[lstart:lend]) | 
					
						
							|  |  |  |             puvs_cpy[i] = tuple(uv.copy() for uv in puvs[i]) | 
					
						
							|  |  |  |             puvsel[i] = (False not in | 
					
						
							| 
									
										
										
										
											2012-10-08 08:28:05 +00:00
										 |  |  |                          (uv.select for uv in uv_loops[lstart:lend])) | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             # Vert idx of the poly. | 
					
						
							| 
									
										
										
										
											2012-06-29 12:41:39 +00:00
										 |  |  |             vidxs[i] = tuple(l.vertex_index for l in loops[lstart:lend]) | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  |             pcents[i] = p.center | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             # Preparing next step finding matching polys. | 
					
						
							| 
									
										
										
										
											2012-06-29 12:41:39 +00:00
										 |  |  |             mirror_pm[tuple(sorted(vidxs[i]))] = i | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for i in range(nbr_polys): | 
					
						
							|  |  |  |             # Find matching mirror poly. | 
					
						
							|  |  |  |             tvidxs = [vmap.get(j) for j in vidxs[i]] | 
					
						
							|  |  |  |             if None not in tvidxs: | 
					
						
							|  |  |  |                 tvidxs.sort() | 
					
						
							|  |  |  |                 j = mirror_pm.get(tuple(tvidxs)) | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 if j is not None: | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |                     pmap[i] = j | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |         for i, j in pmap.items(): | 
					
						
							|  |  |  |             if not puvsel[i] or not puvsel[j]: | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             elif DIR == 0 and pcents[i][0] < 0.0: | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             elif DIR == 1 and pcents[i][0] > 0.0: | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # copy UVs | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             uv1 = puvs[i] | 
					
						
							|  |  |  |             uv2 = puvs_cpy[j] | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |             # get the correct rotation | 
					
						
							| 
									
										
										
										
											2012-04-03 17:19:58 +00:00
										 |  |  |             v1 = vidxs[j] | 
					
						
							|  |  |  |             v2 = tuple(vmap[k] for k in vidxs[i]) | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-21 02:16:22 +00:00
										 |  |  |             if len(v1) == len(v2): | 
					
						
							|  |  |  |                 for k in range(len(v1)): | 
					
						
							|  |  |  |                     k_map = v1.index(v2[k]) | 
					
						
							|  |  |  |                     uv1[k].xy = - (uv2[k_map].x - 0.5) + 0.5, uv2[k_map].y | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |         # has_active_UV_layer, double_warn | 
					
						
							|  |  |  |         return True, double_warn | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							|  |  |  |         obj = context.view_layer.objects.active | 
					
						
							|  |  |  |         return (obj and obj.type == 'MESH') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         DIR = (self.direction == 'NEGATIVE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         total_no_active_UV = 0 | 
					
						
							|  |  |  |         total_duplicates = 0 | 
					
						
							|  |  |  |         meshes_with_duplicates = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ob = context.view_layer.objects.active | 
					
						
							|  |  |  |         is_editmode = (ob.mode == 'EDIT') | 
					
						
							|  |  |  |         if is_editmode: | 
					
						
							|  |  |  |             bpy.ops.object.mode_set(mode='OBJECT', toggle=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         meshes = [ob.data for ob in context.view_layer.objects.selected | 
					
						
							|  |  |  |                   if ob.type == 'MESH' and ob.data.library is None] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for mesh in meshes: | 
					
						
							|  |  |  |             mesh.tag = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for mesh in meshes: | 
					
						
							|  |  |  |             if mesh.tag: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             mesh.tag = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             has_active_UV_layer, double_warn = self.do_mesh_mirror_UV(mesh, DIR) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if not has_active_UV_layer: | 
					
						
							|  |  |  |                 total_no_active_UV = total_no_active_UV + 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             elif double_warn: | 
					
						
							|  |  |  |                 total_duplicates += double_warn | 
					
						
							|  |  |  |                 meshes_with_duplicates = meshes_with_duplicates + 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |         if is_editmode: | 
					
						
							|  |  |  |             bpy.ops.object.mode_set(mode='EDIT', toggle=False) | 
					
						
							| 
									
										
										
										
											2009-11-05 12:37:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |         if total_duplicates and total_no_active_UV: | 
					
						
							|  |  |  |             self.report({'WARNING'}, "%d %s with no active UV layer. " | 
					
						
							| 
									
										
										
										
											2018-09-05 17:31:04 -03:00
										 |  |  |                         "%d duplicates found in %d %s, mirror may be incomplete." | 
					
						
							| 
									
										
										
										
											2018-09-04 16:35:03 -03:00
										 |  |  |                         % (total_no_active_UV, | 
					
						
							|  |  |  |                            "mesh" if total_no_active_UV == 1 else "meshes", | 
					
						
							|  |  |  |                            total_duplicates, | 
					
						
							|  |  |  |                            meshes_with_duplicates, | 
					
						
							|  |  |  |                            "mesh" if meshes_with_duplicates == 1 else "meshes")) | 
					
						
							|  |  |  |         elif total_no_active_UV: | 
					
						
							|  |  |  |             self.report({'WARNING'}, "%d %s with no active UV layer." | 
					
						
							|  |  |  |                         % (total_no_active_UV, | 
					
						
							|  |  |  |                            "mesh" if total_no_active_UV == 1 else "meshes")) | 
					
						
							|  |  |  |         elif total_duplicates: | 
					
						
							|  |  |  |             self.report({'WARNING'}, "%d duplicates found in %d %s," | 
					
						
							|  |  |  |                         " mirror may be incomplete." | 
					
						
							|  |  |  |                         % (total_duplicates, | 
					
						
							|  |  |  |                            meshes_with_duplicates, | 
					
						
							|  |  |  |                            "mesh" if meshes_with_duplicates == 1 else "meshes")) | 
					
						
							| 
									
										
										
										
											2013-06-05 03:10:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-13 17:58:26 +00:00
										 |  |  |         return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2016-01-08 02:54:15 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MeshSelectNext(Operator): | 
					
						
							|  |  |  |     """Select the next element (using selection order)""" | 
					
						
							|  |  |  |     bl_idname = "mesh.select_next_item" | 
					
						
							|  |  |  |     bl_label = "Select Next Element" | 
					
						
							|  |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							|  |  |  |         return (context.mode == 'EDIT_MESH') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         import bmesh | 
					
						
							|  |  |  |         from .bmesh import find_adjacent | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         me = obj.data | 
					
						
							|  |  |  |         bm = bmesh.from_edit_mesh(me) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if find_adjacent.select_next(bm, self.report): | 
					
						
							|  |  |  |             bm.select_flush_mode() | 
					
						
							|  |  |  |             bmesh.update_edit_mesh(me, False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MeshSelectPrev(Operator): | 
					
						
							| 
									
										
										
										
											2017-10-19 11:09:27 +11:00
										 |  |  |     """Select the previous element (using selection order)""" | 
					
						
							| 
									
										
										
										
											2016-01-08 02:54:15 +11:00
										 |  |  |     bl_idname = "mesh.select_prev_item" | 
					
						
							|  |  |  |     bl_label = "Select Previous Element" | 
					
						
							|  |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							|  |  |  |         return (context.mode == 'EDIT_MESH') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         import bmesh | 
					
						
							|  |  |  |         from .bmesh import find_adjacent | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         me = obj.data | 
					
						
							|  |  |  |         bm = bmesh.from_edit_mesh(me) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if find_adjacent.select_prev(bm, self.report): | 
					
						
							|  |  |  |             bm.select_flush_mode() | 
					
						
							|  |  |  |             bmesh.update_edit_mesh(me, False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2016-11-02 15:11:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 20:03:24 +11:00
										 |  |  | classes = ( | 
					
						
							|  |  |  |     MeshMirrorUV, | 
					
						
							|  |  |  |     MeshSelectNext, | 
					
						
							|  |  |  |     MeshSelectPrev, | 
					
						
							| 
									
										
										
										
											2017-03-25 11:07:05 +11:00
										 |  |  | ) |