-
Notifications
You must be signed in to change notification settings - Fork 140
/
op_unwrap_edge_peel.py
168 lines (136 loc) · 5.44 KB
/
op_unwrap_edge_peel.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
import bpy
import bmesh
from . import utilities_ui
from . import utilities_uv
from . import op_rectify
from . import settings
class op(bpy.types.Operator):
bl_idname = "uv.textools_unwrap_edge_peel"
bl_label = "Edge Peel"
bl_description = "Unwrap as a pipe along the edges selected in 3D Space"
bl_options = {'REGISTER', 'UNDO'}
@classmethod
def poll(cls, context):
if not bpy.context.active_object:
return False
if bpy.context.active_object.type != 'MESH':
return False
if bpy.context.active_object.mode != 'EDIT':
return False
if tuple(bpy.context.scene.tool_settings.mesh_select_mode)[1] == False:
return False
return True
def execute(self, context):
selection_mode = bpy.context.scene.tool_settings.uv_select_mode
is_sync = bpy.context.scene.tool_settings.use_uv_select_sync
context_override = utilities_ui.GetContextViewUV()
if not context_override:
self.report({'ERROR_INVALID_INPUT'}, "This tool requires an available UV/Image view")
return {'CANCELLED'}
padding = utilities_ui.get_padding()
utilities_uv.multi_object_loop(unwrap_edges_pipe, self, context, padding)
bpy.ops.uv.average_islands_scale()
bpy.ops.uv.pack_islands(rotate=False, margin=padding)
# Move to active UDIM Tile TODO pack if not {'CANCELLED'} in the active UDIM Tile when implemented in Blender master (watch out for versioning)
udim_tile, column, row = utilities_uv.get_UDIM_tile_coords(bpy.context.active_object)
if udim_tile != 1001:
if settings.bversion >= 3.2:
with bpy.context.temp_override(**context_override):
bpy.ops.transform.translate(value=(column, row, 0), mirror=False, use_proportional_edit=False)
else:
bpy.ops.transform.translate(context_override, value=(column, row, 0), mirror=False, use_proportional_edit=False)
# Workaround for selection not flushing properly from loops to EDGE Selection Mode, apparently since UV edge selection support was added to the UV space
if settings.bversion >= 3.2:
with bpy.context.temp_override(**context_override):
bpy.ops.uv.select_mode(type='VERTEX')
bpy.context.scene.tool_settings.uv_select_mode = selection_mode
if is_sync:
bpy.context.scene.tool_settings.use_uv_select_sync = True
return {'FINISHED'}
def unwrap_edges_pipe(self, context, padding):
me = bpy.context.active_object.data
bm = bmesh.from_edit_mesh(me)
uv_layers = bm.loops.layers.uv.verify()
# Verify that no faces are selected
for face in bm.faces:
if face.select:
bpy.ops.mesh.select_all(action='DESELECT')
self.report({'INFO'}, "No faces should be selected, only edge rings")
return
# Extend loop selection
bpy.ops.mesh.loop_multi_select(ring=False)
selected_edges = {edge for edge in bm.edges if edge.select}
if len(selected_edges) == 0:
bpy.ops.mesh.select_all(action='DESELECT')
#self.report({'ERROR_INVALID_INPUT'}, "No edges selected in the view" )
return
bpy.ops.mesh.select_linked(delimit=set())
bpy.ops.mesh.mark_seam(clear=True)
selected_faces = {face for face in bm.faces if face.select}
if not selected_faces:
bpy.ops.mesh.select_all(action='DESELECT')
self.report({'INFO'}, "It's not possible to perform the unwrap on loose edges" )
return
for edge in selected_edges:
edge.seam = True
# "Follow active quad" kind of unwrap
bpy.context.scene.tool_settings.use_uv_select_sync = False
bpy.ops.mesh.select_all(action='DESELECT')
bpy.ops.uv.select_all(action='DESELECT')
for face in selected_faces:
face.select = True
for loop in face.loops:
loop[uv_layers].select = True
bpy.ops.uv.unwrap(method='ANGLE_BASED', margin=padding)
# Rectify the unwrapped islands
islands = utilities_uv.getSelectionIslands(bm, uv_layers, extend_selection_to_islands=True, selected_faces=selected_faces)
for island in islands:
unrectified_faces = set()
rectified_faces = set()
face_loops = {face : [loop for loop in face.loops] for face in island}
active = {bm.faces.active}
count = 3
# Repeat Rectify until the result is not self-overlapping; max 3 iterations
while count > 0:
if count == 3:
unrectified_faces = op_rectify.main(me, bm, uv_layers, island, face_loops, return_discarded_faces=True)
rectified_faces = island.difference(unrectified_faces)
for face in unrectified_faces:
face.select_set(False)
else:
for f in rectified_faces:
if f not in active:
bm.faces.active = f
active.add(f)
break
bpy.ops.mesh.select_all(action='DESELECT')
for face in rectified_faces:
face.select_set(True)
for loop in face.loops:
loop[uv_layers].select = True
bpy.ops.uv.unwrap(method='ANGLE_BASED', margin=padding)
op_rectify.main(me, bm, uv_layers, island, face_loops)
if not rectified_faces:
count = 0
continue
bpy.ops.uv.select_all(action='DESELECT')
bpy.ops.uv.select_overlap(extend=False)
for f in rectified_faces:
if f.loops[0][uv_layers].select:
count -= 1
else:
count = 0
break
# Reproject ngons and triangular faces
if unrectified_faces:
bpy.ops.mesh.select_all(action='DESELECT')
for face in unrectified_faces:
face.select_set(True)
for loop in face.loops:
loop[uv_layers].select = True
bpy.ops.uv.unwrap(method='ANGLE_BASED', margin=padding)
# Restore selection
for face in selected_faces:
face.select_set(True)
for loop in face.loops:
loop[uv_layers].select = True