1 # ##### BEGIN GPL LICENSE BLOCK #####
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.
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.
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.
17 # ##### END GPL LICENSE BLOCK #####
23 class BrushButtonsPanel():
24 bl_space_type = 'IMAGE_EDITOR'
28 def poll(cls, context):
29 sima = context.space_data
30 toolsettings = context.tool_settings.image_paint
31 return sima.show_paint and toolsettings.brush
34 class IMAGE_MT_view(bpy.types.Menu):
37 def draw(self, context):
40 sima = context.space_data
42 toolsettings = context.tool_settings
44 show_uvedit = sima.show_uvedit
46 layout.operator("image.properties", icon='MENU_PANEL')
47 layout.operator("image.scopes", icon='MENU_PANEL')
51 layout.prop(sima, "use_realtime_update")
53 layout.prop(toolsettings, "show_uv_local_view")
54 layout.prop(uv, "show_other_objects")
58 layout.operator("image.view_zoom_in")
59 layout.operator("image.view_zoom_out")
63 ratios = [[1, 8], [1, 4], [1, 2], [1, 1], [2, 1], [4, 1], [8, 1]]
66 text = "Zoom %d:%d" % (a, b)
67 layout.operator("image.view_zoom_ratio", text=text).ratio = a / b
72 layout.operator("image.view_selected")
74 layout.operator("image.view_all")
78 layout.operator("screen.area_dupli")
79 layout.operator("screen.screen_full_area")
82 class IMAGE_MT_select(bpy.types.Menu):
85 def draw(self, context):
88 layout.operator("uv.select_border")
89 layout.operator("uv.select_border").pinned = True
93 layout.operator("uv.select_all")
94 layout.operator("uv.select_inverse")
95 layout.operator("uv.unlink_selection")
99 layout.operator("uv.select_pinned")
100 layout.operator("uv.select_linked")
103 class IMAGE_MT_image(bpy.types.Menu):
106 def draw(self, context):
109 sima = context.space_data
112 layout.operator("image.new")
113 layout.operator("image.open")
115 show_render = sima.show_render
119 layout.operator("image.replace")
120 layout.operator("image.reload")
122 layout.operator("image.save")
123 layout.operator("image.save_as")
124 layout.operator("image.save_as", text="Save a Copy").copy = True
126 if ima.source == 'SEQUENCE':
127 layout.operator("image.save_sequence")
129 layout.operator("image.external_edit", "Edit Externally")
133 layout.menu("IMAGE_MT_image_invert")
139 layout.operator("image.unpack")
141 layout.operator("image.pack")
143 # only for dirty && specific image types, perhaps
144 # this could be done in operator poll too
146 if ima.source in ('FILE', 'GENERATED') and ima.type != 'MULTILAYER':
147 layout.operator("image.pack", text="Pack As PNG").as_png = True
151 layout.prop(sima, "use_image_paint")
154 class IMAGE_MT_image_invert(bpy.types.Menu):
157 def draw(self, context):
160 op = layout.operator("image.invert", text="Invert Image Colors")
167 op = layout.operator("image.invert", text="Invert Red Channel")
170 op = layout.operator("image.invert", text="Invert Green Channel")
173 op = layout.operator("image.invert", text="Invert Blue Channel")
176 op = layout.operator("image.invert", text="Invert Alpha Channel")
180 class IMAGE_MT_uvs_showhide(bpy.types.Menu):
181 bl_label = "Show/Hide Faces"
183 def draw(self, context):
186 layout.operator("uv.reveal")
187 layout.operator("uv.hide", text="Hide Selected")
188 layout.operator("uv.hide", text="Hide Unselected").unselected = True
191 class IMAGE_MT_uvs_transform(bpy.types.Menu):
192 bl_label = "Transform"
194 def draw(self, context):
197 layout.operator("transform.translate")
198 layout.operator("transform.rotate")
199 layout.operator("transform.resize")
202 class IMAGE_MT_uvs_snap(bpy.types.Menu):
205 def draw(self, context):
207 layout.operator_context = 'EXEC_REGION_WIN'
209 layout.operator("uv.snap_selection", text="Selected to Pixels").target = 'PIXELS'
210 layout.operator("uv.snap_selection", text="Selected to Cursor").target = 'CURSOR'
211 layout.operator("uv.snap_selection", text="Selected to Adjacent Unselected").target = 'ADJACENT_UNSELECTED'
215 layout.operator("uv.snap_cursor", text="Cursor to Pixels").target = 'PIXELS'
216 layout.operator("uv.snap_cursor", text="Cursor to Selection").target = 'SELECTION'
219 class IMAGE_MT_uvs_mirror(bpy.types.Menu):
222 def draw(self, context):
224 layout.operator_context = 'EXEC_REGION_WIN'
226 layout.operator("transform.mirror", text="X Axis").constraint_axis[0] = True
227 layout.operator("transform.mirror", text="Y Axis").constraint_axis[1] = True
230 class IMAGE_MT_uvs_weldalign(bpy.types.Menu):
231 bl_label = "Weld/Align"
233 def draw(self, context):
236 layout.operator("uv.weld") # W, 1
237 layout.operator_enum("uv.align", "axis") # W, 2/3/4
240 class IMAGE_MT_uvs(bpy.types.Menu):
243 def draw(self, context):
246 sima = context.space_data
248 toolsettings = context.tool_settings
250 layout.prop(uv, "use_snap_to_pixels")
251 layout.prop(uv, "lock_bounds")
255 layout.prop(uv, "use_live_unwrap")
256 layout.operator("uv.unwrap")
257 layout.operator("uv.pin", text="Unpin").clear = True
258 layout.operator("uv.pin")
262 layout.operator("uv.pack_islands")
263 layout.operator("uv.average_islands_scale")
264 layout.operator("uv.minimize_stretch")
265 layout.operator("uv.stitch")
266 layout.operator("mesh.faces_miror_uv")
270 layout.menu("IMAGE_MT_uvs_transform")
271 layout.menu("IMAGE_MT_uvs_mirror")
272 layout.menu("IMAGE_MT_uvs_snap")
273 layout.menu("IMAGE_MT_uvs_weldalign")
277 layout.prop_menu_enum(toolsettings, "proportional_edit")
278 layout.prop_menu_enum(toolsettings, "proportional_edit_falloff")
282 layout.menu("IMAGE_MT_uvs_showhide")
285 class IMAGE_MT_uvs_select_mode(bpy.types.Menu):
286 bl_label = "UV Select Mode"
288 def draw(self, context):
291 layout.operator_context = 'INVOKE_REGION_WIN'
292 toolsettings = context.tool_settings
294 # do smart things depending on whether uv_select_sync is on
296 if toolsettings.use_uv_select_sync:
297 prop = layout.operator("wm.context_set_value", text="Vertex", icon='VERTEXSEL')
298 prop.value = "(True, False, False)"
299 prop.data_path = "tool_settings.mesh_select_mode"
301 prop = layout.operator("wm.context_set_value", text="Edge", icon='EDGESEL')
302 prop.value = "(False, True, False)"
303 prop.data_path = "tool_settings.mesh_select_mode"
305 prop = layout.operator("wm.context_set_value", text="Face", icon='FACESEL')
306 prop.value = "(False, False, True)"
307 prop.data_path = "tool_settings.mesh_select_mode"
310 prop = layout.operator("wm.context_set_string", text="Vertex", icon='UV_VERTEXSEL')
311 prop.value = "VERTEX"
312 prop.data_path = "tool_settings.uv_select_mode"
314 prop = layout.operator("wm.context_set_string", text="Edge", icon='UV_EDGESEL')
316 prop.data_path = "tool_settings.uv_select_mode"
318 prop = layout.operator("wm.context_set_string", text="Face", icon='UV_FACESEL')
320 prop.data_path = "tool_settings.uv_select_mode"
322 prop = layout.operator("wm.context_set_string", text="Island", icon='UV_ISLANDSEL')
323 prop.value = "ISLAND"
324 prop.data_path = "tool_settings.uv_select_mode"
327 class IMAGE_HT_header(bpy.types.Header):
328 bl_space_type = 'IMAGE_EDITOR'
330 def draw(self, context):
333 sima = context.space_data
335 iuser = sima.image_user
336 toolsettings = context.tool_settings
338 show_render = sima.show_render
339 # show_paint = sima.show_paint
340 show_uvedit = sima.show_uvedit
342 row = layout.row(align=True)
343 row.template_header()
346 if context.area.show_menus:
347 sub = row.row(align=True)
348 sub.menu("IMAGE_MT_view")
351 sub.menu("IMAGE_MT_select")
353 if ima and ima.is_dirty:
354 sub.menu("IMAGE_MT_image", text="Image*")
356 sub.menu("IMAGE_MT_image", text="Image")
359 sub.menu("IMAGE_MT_uvs")
361 layout.template_ID(sima, "image", new="image.new")
363 layout.prop(sima, "use_image_pin", text="")
367 uvedit = sima.uv_editor
369 layout.prop(uvedit, "pivot_point", text="", icon_only=True)
370 layout.prop(toolsettings, "use_uv_select_sync", text="")
372 if toolsettings.use_uv_select_sync:
373 row = layout.row(align=True)
374 row.prop(toolsettings, "mesh_select_mode", text="", index=0, icon='VERTEXSEL')
375 row.prop(toolsettings, "mesh_select_mode", text="", index=1, icon='EDGESEL')
376 row.prop(toolsettings, "mesh_select_mode", text="", index=2, icon='FACESEL')
378 layout.prop(toolsettings, "uv_select_mode", text="", expand=True)
379 layout.prop(uvedit, "sticky_select_mode", text="", icon_only=True)
381 row = layout.row(align=True)
382 row.prop(toolsettings, "proportional_edit", text="", icon_only=True)
383 if toolsettings.proportional_edit != 'DISABLED':
384 row.prop(toolsettings, "proportional_edit_falloff", text="", icon_only=True)
386 row = layout.row(align=True)
387 row.prop(toolsettings, "use_snap", text="")
388 row.prop(toolsettings, "snap_element", text="", icon_only=True)
390 # mesh = context.edit_object.data
391 # row.prop_search(mesh.uv_textures, "active", mesh, "uv_textures")
395 layout.template_image_layers(ima, iuser)
398 layout.prop(sima, "use_image_paint", text="")
401 row = layout.row(align=True)
402 row.prop(sima, "draw_channels", text="", expand=True)
404 row = layout.row(align=True)
405 if ima.type == 'COMPOSITE':
406 row.operator("image.record_composite", icon='REC')
407 if ima.type == 'COMPOSITE' and ima.source in ('MOVIE', 'SEQUENCE'):
408 row.operator("image.play_composite", icon='PLAY')
410 if show_uvedit or sima.use_image_paint:
411 layout.prop(sima, "use_realtime_update", text="", icon_only=True, icon='LOCKED')
414 class IMAGE_PT_image_properties(bpy.types.Panel):
415 bl_space_type = 'IMAGE_EDITOR'
416 bl_region_type = 'UI'
420 def poll(cls, context):
421 sima = context.space_data
424 def draw(self, context):
427 sima = context.space_data
429 iuser = sima.image_user
431 layout.template_image(sima, "image", iuser)
434 class IMAGE_PT_game_properties(bpy.types.Panel):
435 bl_space_type = 'IMAGE_EDITOR'
436 bl_region_type = 'UI'
437 bl_label = "Game Properties"
440 def poll(cls, context):
441 rd = context.scene.render
442 sima = context.space_data
443 return (sima and sima.image) and (rd.engine == 'BLENDER_GAME')
445 def draw(self, context):
448 sima = context.space_data
451 split = layout.split()
455 sub = col.column(align=True)
456 sub.prop(ima, "use_animation")
458 subsub = sub.column()
459 subsub.active = ima.use_animation
460 subsub.prop(ima, "frame_start", text="Start")
461 subsub.prop(ima, "frame_end", text="End")
462 subsub.prop(ima, "fps", text="Speed")
464 col.prop(ima, "use_tiles")
465 sub = col.column(align=True)
466 sub.active = ima.use_tiles or ima.use_animation
467 sub.prop(ima, "tiles_x", text="X")
468 sub.prop(ima, "tiles_y", text="Y")
471 col.label(text="Clamp:")
472 col.prop(ima, "use_clamp_x", text="X")
473 col.prop(ima, "use_clamp_y", text="Y")
475 col.prop(ima, "mapping", expand=True)
478 class IMAGE_PT_view_histogram(bpy.types.Panel):
479 bl_space_type = 'IMAGE_EDITOR'
480 bl_region_type = 'PREVIEW'
481 bl_label = "Histogram"
484 def poll(cls, context):
485 sima = context.space_data
486 return (sima and sima.image)
488 def draw(self, context):
491 sima = context.space_data
493 layout.template_histogram(sima.scopes, "histogram")
494 layout.prop(sima.scopes.histogram, "mode", icon_only=True)
497 class IMAGE_PT_view_waveform(bpy.types.Panel):
498 bl_space_type = 'IMAGE_EDITOR'
499 bl_region_type = 'PREVIEW'
500 bl_label = "Waveform"
503 def poll(cls, context):
504 sima = context.space_data
505 return (sima and sima.image)
507 def draw(self, context):
510 sima = context.space_data
511 layout.template_waveform(sima, "scopes")
512 sub = layout.row().split(percentage=0.75)
513 sub.prop(sima.scopes, "waveform_alpha")
514 sub.prop(sima.scopes, "waveform_mode", text="", icon_only=True)
517 class IMAGE_PT_view_vectorscope(bpy.types.Panel):
518 bl_space_type = 'IMAGE_EDITOR'
519 bl_region_type = 'PREVIEW'
520 bl_label = "Vectorscope"
523 def poll(cls, context):
524 sima = context.space_data
525 return (sima and sima.image)
527 def draw(self, context):
530 sima = context.space_data
531 layout.template_vectorscope(sima, "scopes")
532 layout.prop(sima.scopes, "vectorscope_alpha")
535 class IMAGE_PT_sample_line(bpy.types.Panel):
536 bl_space_type = 'IMAGE_EDITOR'
537 bl_region_type = 'PREVIEW'
538 bl_label = "Sample Line"
541 def poll(cls, context):
542 sima = context.space_data
543 return (sima and sima.image)
545 def draw(self, context):
547 layout.operator("image.sample_line")
548 sima = context.space_data
549 layout.template_histogram(sima, "sample_histogram")
550 layout.prop(sima.sample_histogram, "mode")
553 class IMAGE_PT_scope_sample(bpy.types.Panel):
554 bl_space_type = 'IMAGE_EDITOR'
555 bl_region_type = 'PREVIEW'
556 bl_label = "Scope Samples"
559 def poll(cls, context):
560 sima = context.space_data
563 def draw(self, context):
565 sima = context.space_data
566 split = layout.split()
568 row.prop(sima.scopes, "use_full_resolution")
570 row.active = not sima.scopes.use_full_resolution
571 row.prop(sima.scopes, "accuracy")
574 class IMAGE_PT_view_properties(bpy.types.Panel):
575 bl_space_type = 'IMAGE_EDITOR'
576 bl_region_type = 'UI'
580 def poll(cls, context):
581 sima = context.space_data
582 return (sima and (sima.image or sima.show_uvedit))
584 def draw(self, context):
587 sima = context.space_data
589 show_uvedit = sima.show_uvedit
590 uvedit = sima.uv_editor
592 split = layout.split()
596 col.prop(ima, "display_aspect", text="Aspect Ratio")
599 col.label(text="Coordinates:")
600 col.prop(sima, "show_repeat", text="Repeat")
602 col.prop(uvedit, "show_normalized_coords", text="Normalized")
605 col.label(text="Coordinates:")
606 col.prop(uvedit, "show_normalized_coords", text="Normalized")
610 col = layout.column()
611 col.prop(uvedit, "cursor_location")
613 col = layout.column()
614 col.label(text="UVs:")
616 row.prop(uvedit, "edge_draw_type", expand=True)
618 split = layout.split()
620 col.prop(uvedit, "show_smooth_edges", text="Smooth")
621 col.prop(uvedit, "show_modified_edges", text="Modified")
622 #col.prop(uvedit, "show_edges")
623 #col.prop(uvedit, "show_faces")
626 col.prop(uvedit, "show_stretch", text="Stretch")
628 sub.active = uvedit.show_stretch
629 sub.row().prop(uvedit, "draw_stretch_type", expand=True)
632 class IMAGE_PT_paint(bpy.types.Panel):
633 bl_space_type = 'IMAGE_EDITOR'
634 bl_region_type = 'UI'
638 def poll(cls, context):
639 sima = context.space_data
640 return sima.show_paint
642 def draw(self, context):
645 toolsettings = context.tool_settings.image_paint
646 brush = toolsettings.brush
648 col = layout.split().column()
650 col.template_ID_preview(toolsettings, "brush", new="brush.add", rows=3, cols=8)
653 col = layout.column()
654 col.template_color_wheel(brush, "color", value_slider=True)
655 col.prop(brush, "color", text="")
657 row = col.row(align=True)
658 row.prop(brush, "size", slider=True)
659 row.prop(brush, "use_pressure_size", toggle=True, text="")
661 row = col.row(align=True)
662 row.prop(brush, "strength", slider=True)
663 row.prop(brush, "use_pressure_strength", toggle=True, text="")
665 row = col.row(align=True)
666 row.prop(brush, "jitter", slider=True)
667 row.prop(brush, "use_pressure_jitter", toggle=True, text="")
669 col.prop(brush, "blend", text="Blend")
671 if brush.imagepaint_tool == 'CLONE':
673 col.prop(brush, "clone_image", text="Image")
674 col.prop(brush, "clone_alpha", text="Alpha")
677 class IMAGE_PT_tools_brush_texture(BrushButtonsPanel, bpy.types.Panel):
679 bl_options = {'DEFAULT_CLOSED'}
681 def draw(self, context):
684 toolsettings = context.tool_settings.image_paint
685 brush = toolsettings.brush
687 # tex_slot = brush.texture_slot
689 col = layout.column()
691 col.template_ID_preview(brush, "texture", new="texture.new", rows=3, cols=8)
694 class IMAGE_PT_tools_brush_tool(BrushButtonsPanel, bpy.types.Panel):
696 bl_options = {'DEFAULT_CLOSED'}
698 def draw(self, context):
700 settings = context.tool_settings.image_paint
701 brush = settings.brush
703 col = layout.column(align=True)
705 col.prop(brush, "imagepaint_tool", expand=False, text="")
707 row = layout.row(align=True)
708 row.prop(brush, "use_paint_sculpt", text="", icon='SCULPTMODE_HLT')
709 row.prop(brush, "use_paint_vertex", text="", icon='VPAINT_HLT')
710 row.prop(brush, "use_paint_weight", text="", icon='WPAINT_HLT')
711 row.prop(brush, "use_paint_texture", text="", icon='TPAINT_HLT')
714 class IMAGE_PT_paint_stroke(BrushButtonsPanel, bpy.types.Panel):
715 bl_label = "Paint Stroke"
716 bl_options = {'DEFAULT_CLOSED'}
718 def draw(self, context):
721 toolsettings = context.tool_settings.image_paint
722 brush = toolsettings.brush
724 layout.prop(brush, "use_airbrush")
725 col = layout.column()
726 col.active = brush.use_airbrush
727 col.prop(brush, "rate", slider=True)
729 layout.prop(brush, "use_space")
730 row = layout.row(align=True)
731 row.active = brush.use_space
732 row.prop(brush, "spacing", text="Distance", slider=True)
733 row.prop(brush, "use_pressure_spacing", toggle=True, text="")
735 layout.prop(brush, "use_wrap")
738 class IMAGE_PT_paint_curve(BrushButtonsPanel, bpy.types.Panel):
739 bl_label = "Paint Curve"
740 bl_options = {'DEFAULT_CLOSED'}
742 def draw(self, context):
745 toolsettings = context.tool_settings.image_paint
746 brush = toolsettings.brush
748 layout.template_curve_mapping(brush, "curve")
750 row = layout.row(align=True)
751 row.operator("brush.curve_preset", icon="SMOOTHCURVE", text="").shape = 'SMOOTH'
752 row.operator("brush.curve_preset", icon="SPHERECURVE", text="").shape = 'ROUND'
753 row.operator("brush.curve_preset", icon="ROOTCURVE", text="").shape = 'ROOT'
754 row.operator("brush.curve_preset", icon="SHARPCURVE", text="").shape = 'SHARP'
755 row.operator("brush.curve_preset", icon="LINCURVE", text="").shape = 'LINE'
756 row.operator("brush.curve_preset", icon="NOCURVE", text="").shape = 'MAX'
760 bpy.utils.register_module(__name__)
764 bpy.utils.unregister_module(__name__)
766 if __name__ == "__main__":