Fixing a bug found while checking "[#31937] UV/Image Editor: Copy Mirrored UV Coords...
[blender.git] / release / scripts / startup / bl_operators / mesh.py
1 # ##### BEGIN GPL LICENSE BLOCK #####
2 #
3 #  This program is free software; you can redistribute it and/or
4 #  modify it under the terms of the GNU General Public License
5 #  as published by the Free Software Foundation; either version 2
6 #  of the License, or (at your option) any later version.
7 #
8 #  This program is distributed in the hope that it will be useful,
9 #  but WITHOUT ANY WARRANTY; without even the implied warranty of
10 #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11 #  GNU General Public License for more details.
12 #
13 #  You should have received a copy of the GNU General Public License
14 #  along with this program; if not, write to the Free Software Foundation,
15 #  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 #
17 # ##### END GPL LICENSE BLOCK #####
18
19 # <pep8-80 compliant>
20
21 import bpy
22 from bpy.types import Operator
23
24 from bpy.props import EnumProperty
25
26
27 class MeshMirrorUV(Operator):
28     '''Copy mirror UV coordinates on the X axis based on a mirrored mesh'''
29     bl_idname = "mesh.faces_mirror_uv"
30     bl_label = "Copy Mirrored UV coords"
31     bl_options = {'REGISTER', 'UNDO'}
32
33     direction = EnumProperty(
34             name="Axis Direction",
35             items=(('POSITIVE', "Positive", ""),
36                    ('NEGATIVE', "Negative", "")),
37             )
38
39     @classmethod
40     def poll(cls, context):
41         obj = context.active_object
42         return (obj and obj.type == 'MESH' and obj.data.uv_textures.active)
43
44     def execute(self, context):
45         DIR = (self.direction == 'NEGATIVE')
46
47         ob = context.active_object
48         is_editmode = (ob.mode == 'EDIT')
49         if is_editmode:
50             bpy.ops.object.mode_set(mode='OBJECT', toggle=False)
51
52         mesh = ob.data
53
54         # mirror lookups
55         mirror_gt = {}
56         mirror_lt = {}
57
58         vcos = (v.co.to_tuple(5) for v in mesh.vertices)
59
60         for i, co in enumerate(vcos):
61             if co[0] >= 0.0:
62                 mirror_gt[co] = i
63             if co[0] <= 0.0:
64                 mirror_lt[co] = i
65
66         #for i, v in enumerate(mesh.vertices):
67         vmap = {}
68         for mirror_a, mirror_b in ((mirror_gt, mirror_lt),
69                                    (mirror_lt, mirror_gt)):
70             for co, i in mirror_a.items():
71                 nco = (-co[0], co[1], co[2])
72                 j = mirror_b.get(nco)
73                 if j is not None:
74                     vmap[i] = j
75
76         polys = mesh.polygons
77         loops = mesh.loops
78         verts = mesh.vertices
79         uv_loops = mesh.uv_layers.active.data
80         nbr_polys = len(polys)
81
82         mirror_pm = {}
83         pmap = {}
84         puvs = [None] * nbr_polys
85         puvs_cpy = [None] * nbr_polys
86         puvsel = [None] * nbr_polys
87         pcents = [None] * nbr_polys
88         vidxs = [None] * nbr_polys
89         for i, p in enumerate(polys):
90             lstart = lend = p.loop_start
91             lend += p.loop_total
92             puvs[i] = tuple(uv.uv for uv in uv_loops[lstart:lend])
93             puvs_cpy[i] = tuple(uv.copy() for uv in puvs[i])
94             puvsel[i] = (False not in
95                                (uv.select for uv in uv_loops[lstart:lend]))
96             # Vert idx of the poly.
97             vidxs[i] = tuple(l.vertex_index for l in loops[lstart:lend])
98             # As we have no poly.center yet...
99             pcents[i] = tuple(map(lambda x: x / p.loop_total,
100                                   map(sum, zip(*(verts[idx].co
101                                                  for idx in vidxs[i])))))
102             # Preparing next step finding matching polys.
103             mirror_pm[tuple(sorted(vidxs[i]))] = i
104
105         for i in range(nbr_polys):
106             # Find matching mirror poly.
107             tvidxs = [vmap.get(j) for j in vidxs[i]]
108             if None not in tvidxs:
109                 tvidxs.sort()
110                 j = mirror_pm.get(tuple(tvidxs))
111                 if j is not None:
112                     pmap[i] = j
113
114         for i, j in pmap.items():
115             if not puvsel[i] or not puvsel[j]:
116                 continue
117             elif DIR == 0 and pcents[i][0] < 0.0:
118                 continue
119             elif DIR == 1 and pcents[i][0] > 0.0:
120                 continue
121
122             # copy UVs
123             uv1 = puvs[i]
124             uv2 = puvs_cpy[j]
125
126             # get the correct rotation
127             v1 = vidxs[j]
128             v2 = tuple(vmap[k] for k in vidxs[i])
129
130             if len(v1) == len(v2):
131                 for k in range(len(v1)):
132                     k_map = v1.index(v2[k])
133                     uv1[k].xy = - (uv2[k_map].x - 0.5) + 0.5, uv2[k_map].y
134
135         if is_editmode:
136             bpy.ops.object.mode_set(mode='EDIT', toggle=False)
137
138         return {'FINISHED'}