| 
									
										
										
										
											2009-11-26 15:36:23 +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. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #  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, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  | #  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | # | 
					
						
							|  |  |  | # ##### END GPL LICENSE BLOCK ##### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  | # <pep8-80 compliant> | 
					
						
							| 
									
										
										
										
											2009-12-05 22:03:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | import bpy | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | from bpy.types import Operator | 
					
						
							| 
									
										
										
										
											2011-02-27 15:25:24 +00:00
										 |  |  | from bpy.props import StringProperty, BoolProperty, EnumProperty, IntProperty | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class SelectPattern(Operator): | 
					
						
							| 
									
										
										
										
											2010-02-11 23:13:47 +00:00
										 |  |  |     '''Select object matching a naming pattern''' | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  |     bl_idname = "object.select_pattern" | 
					
						
							|  |  |  |     bl_label = "Select Pattern" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     pattern = StringProperty( | 
					
						
							|  |  |  |             name="Pattern", | 
					
						
							|  |  |  |             description="Name filter using '*' and '?' wildcard chars", | 
					
						
							|  |  |  |             maxlen=32, | 
					
						
							|  |  |  |             default="*", | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |     case_sensitive = BoolProperty( | 
					
						
							|  |  |  |             name="Case Sensitive", | 
					
						
							|  |  |  |             description="Do a case sensitive compare", | 
					
						
							|  |  |  |             default=False, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |     extend = BoolProperty( | 
					
						
							|  |  |  |             name="Extend", | 
					
						
							|  |  |  |             description="Extend the existing selection", | 
					
						
							|  |  |  |             default=True, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         import fnmatch | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |         if self.case_sensitive: | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  |             pattern_match = fnmatch.fnmatchcase | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |             pattern_match = (lambda a, b: | 
					
						
							|  |  |  |                                  fnmatch.fnmatchcase(a.upper(), b.upper())) | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |         is_ebone = False | 
					
						
							| 
									
										
										
										
											2009-12-01 14:48:36 +00:00
										 |  |  |         obj = context.object | 
					
						
							|  |  |  |         if obj and obj.mode == 'POSE': | 
					
						
							|  |  |  |             items = obj.data.bones | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |             if not self.extend: | 
					
						
							|  |  |  |                 bpy.ops.pose.select_all(action='DESELECT') | 
					
						
							| 
									
										
										
										
											2009-12-01 14:48:36 +00:00
										 |  |  |         elif obj and obj.type == 'ARMATURE' and obj.mode == 'EDIT': | 
					
						
							|  |  |  |             items = obj.data.edit_bones | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |             if not self.extend: | 
					
						
							|  |  |  |                 bpy.ops.armature.select_all(action='DESELECT') | 
					
						
							|  |  |  |             is_ebone = True | 
					
						
							| 
									
										
										
										
											2009-12-01 14:48:36 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             items = context.visible_objects | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |             if not self.extend: | 
					
						
							|  |  |  |                 bpy.ops.object.select_all(action='DESELECT') | 
					
						
							| 
									
										
										
										
											2009-12-01 14:48:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Can be pose bones or objects | 
					
						
							|  |  |  |         for item in items: | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |             if pattern_match(item.name, self.pattern): | 
					
						
							| 
									
										
										
										
											2010-07-15 16:56:04 +00:00
										 |  |  |                 item.select = True | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 # hrmf, perhaps there should be a utility function for this. | 
					
						
							|  |  |  |                 if is_ebone: | 
					
						
							|  |  |  |                     item.select_head = True | 
					
						
							|  |  |  |                     item.select_tail = True | 
					
						
							|  |  |  |                     if item.use_connect: | 
					
						
							|  |  |  |                         item_parent = item.parent | 
					
						
							|  |  |  |                         if item_parent is not None: | 
					
						
							|  |  |  |                             item_parent.select_tail = True | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-24 19:50:43 +00:00
										 |  |  |         return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def invoke(self, context, event): | 
					
						
							| 
									
										
										
										
											2010-09-02 04:53:05 +00:00
										 |  |  |         wm = context.window_manager | 
					
						
							| 
									
										
										
										
											2010-10-31 14:43:30 +00:00
										 |  |  |         return wm.invoke_props_popup(self, event) | 
					
						
							| 
									
										
										
										
											2009-12-05 22:03:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-29 01:49:22 +00:00
										 |  |  |     def draw(self, context): | 
					
						
							|  |  |  |         layout = self.layout | 
					
						
							| 
									
										
										
										
											2009-12-05 22:03:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-24 03:48:26 +00:00
										 |  |  |         layout.prop(self, "pattern") | 
					
						
							| 
									
										
										
										
											2009-11-29 01:49:22 +00:00
										 |  |  |         row = layout.row() | 
					
						
							| 
									
										
										
										
											2010-09-24 03:48:26 +00:00
										 |  |  |         row.prop(self, "case_sensitive") | 
					
						
							|  |  |  |         row.prop(self, "extend") | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class SelectCamera(Operator): | 
					
						
							| 
									
										
										
										
											2010-02-11 23:13:47 +00:00
										 |  |  |     '''Select object matching a naming pattern''' | 
					
						
							| 
									
										
										
										
											2010-02-11 21:17:05 +00:00
										 |  |  |     bl_idname = "object.select_camera" | 
					
						
							|  |  |  |     bl_label = "Select Camera" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2010-02-11 21:17:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2010-02-11 21:17:05 +00:00
										 |  |  |         return context.scene.camera is not None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         scene = context.scene | 
					
						
							|  |  |  |         camera = scene.camera | 
					
						
							|  |  |  |         if camera.name not in scene.objects: | 
					
						
							|  |  |  |             self.report({'WARNING'}, "Active camera is not in this scene") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         context.scene.objects.active = camera | 
					
						
							| 
									
										
										
										
											2010-07-15 16:56:04 +00:00
										 |  |  |         camera.select = True | 
					
						
							| 
									
										
										
										
											2010-02-11 21:17:05 +00:00
										 |  |  |         return {'FINISHED'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class SelectHierarchy(Operator): | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |     '''Select object relative to the active objects position''' \ | 
					
						
							|  |  |  |     '''in the hierarchy''' | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  |     bl_idname = "object.select_hierarchy" | 
					
						
							|  |  |  |     bl_label = "Select Hierarchy" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     direction = EnumProperty( | 
					
						
							|  |  |  |             items=(('PARENT', "Parent", ""), | 
					
						
							|  |  |  |                    ('CHILD', "Child", ""), | 
					
						
							|  |  |  |                    ), | 
					
						
							|  |  |  |             name="Direction", | 
					
						
							|  |  |  |             description="Direction to select in the hierarchy", | 
					
						
							|  |  |  |             default='PARENT') | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     extend = BoolProperty( | 
					
						
							|  |  |  |             name="Extend", | 
					
						
							|  |  |  |             description="Extend the existing selection", | 
					
						
							|  |  |  |             default=False, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  |         return context.object | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |         select_new = [] | 
					
						
							|  |  |  |         act_new = None | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |         selected_objects = context.selected_objects | 
					
						
							| 
									
										
										
										
											2010-07-14 09:40:26 +00:00
										 |  |  |         obj_act = context.object | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |         if context.object not in selected_objects: | 
					
						
							|  |  |  |             selected_objects.append(context.object) | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |         if self.direction == 'PARENT': | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |             for obj in selected_objects: | 
					
						
							| 
									
										
										
										
											2010-07-14 09:40:26 +00:00
										 |  |  |                 parent = obj.parent | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if parent: | 
					
						
							| 
									
										
										
										
											2010-07-14 11:58:19 +00:00
										 |  |  |                     if obj_act == obj: | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |                         act_new = parent | 
					
						
							| 
									
										
										
										
											2010-07-14 11:58:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |                     select_new.append(parent) | 
					
						
							| 
									
										
										
										
											2010-07-14 09:40:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |             for obj in selected_objects: | 
					
						
							|  |  |  |                 select_new.extend(obj.children) | 
					
						
							| 
									
										
										
										
											2010-07-14 09:40:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |             if select_new: | 
					
						
							|  |  |  |                 select_new.sort(key=lambda obj_iter: obj_iter.name) | 
					
						
							|  |  |  |                 act_new = select_new[0] | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |         # dont edit any object settings above this | 
					
						
							|  |  |  |         if select_new: | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |             if not self.extend: | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |                 bpy.ops.object.select_all(action='DESELECT') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for obj in select_new: | 
					
						
							|  |  |  |                 obj.select = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             context.scene.objects.active = act_new | 
					
						
							|  |  |  |             return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 05:49:12 +00:00
										 |  |  |         return {'CANCELLED'} | 
					
						
							| 
									
										
										
										
											2010-02-25 15:41:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class SubdivisionSet(Operator): | 
					
						
							| 
									
										
										
										
											2009-11-27 18:55:59 +00:00
										 |  |  |     '''Sets a Subdivision Surface Level (1-5)''' | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 19:56:59 +00:00
										 |  |  |     bl_idname = "object.subdivision_set" | 
					
						
							|  |  |  |     bl_label = "Subdivision Set" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2009-11-28 23:37:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     level = IntProperty(name="Level", | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  |             default=1, min=-100, max=100, soft_min=-6, soft_max=6) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     relative = BoolProperty( | 
					
						
							|  |  |  |             name="Relative", | 
					
						
							|  |  |  |             description=("Apply the subsurf level as an offset " | 
					
						
							|  |  |  |                          "relative to the current level"), | 
					
						
							|  |  |  |             default=False, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2009-12-21 23:14:16 +00:00
										 |  |  |         obs = context.selected_editable_objects | 
					
						
							|  |  |  |         return (obs is not None) | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |         level = self.level | 
					
						
							|  |  |  |         relative = self.relative | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if relative and level == 0: | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  |             return {'CANCELLED'}  # nothing to do | 
					
						
							| 
									
										
										
										
											2009-12-18 23:17:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-25 06:44:43 +00:00
										 |  |  |         if not relative and level < 0: | 
					
						
							|  |  |  |             self.level = level = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-18 23:17:23 +00:00
										 |  |  |         def set_object_subd(obj): | 
					
						
							|  |  |  |             for mod in obj.modifiers: | 
					
						
							|  |  |  |                 if mod.type == 'MULTIRES': | 
					
						
							| 
									
										
										
										
											2010-01-15 05:00:02 +00:00
										 |  |  |                     if not relative: | 
					
						
							|  |  |  |                         if level <= mod.total_levels: | 
					
						
							|  |  |  |                             if obj.mode == 'SCULPT': | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  |                                 if mod.sculpt_levels != level: | 
					
						
							|  |  |  |                                     mod.sculpt_levels = level | 
					
						
							| 
									
										
										
										
											2010-01-15 05:00:02 +00:00
										 |  |  |                             elif obj.mode == 'OBJECT': | 
					
						
							|  |  |  |                                 if mod.levels != level: | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  |                                     mod.levels = level | 
					
						
							| 
									
										
										
										
											2010-01-15 05:00:02 +00:00
										 |  |  |                         return | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         if obj.mode == 'SCULPT': | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  |                             if mod.sculpt_levels + level <= mod.total_levels: | 
					
						
							| 
									
										
										
										
											2010-01-15 05:00:02 +00:00
										 |  |  |                                 mod.sculpt_levels += level | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  |                         elif obj.mode == 'OBJECT': | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  |                             if mod.levels + level <= mod.total_levels: | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  |                                 mod.levels += level | 
					
						
							| 
									
										
										
										
											2010-01-15 05:00:02 +00:00
										 |  |  |                         return | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-18 23:17:23 +00:00
										 |  |  |                 elif mod.type == 'SUBSURF': | 
					
						
							| 
									
										
										
										
											2009-12-23 12:13:48 +00:00
										 |  |  |                     if relative: | 
					
						
							|  |  |  |                         mod.levels += level | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         if mod.levels != level: | 
					
						
							|  |  |  |                             mod.levels = level | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-18 23:17:23 +00:00
										 |  |  |                     return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-19 01:14:57 +00:00
										 |  |  |             # add a new modifier | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 mod = obj.modifiers.new("Subsurf", 'SUBSURF') | 
					
						
							|  |  |  |                 mod.levels = level | 
					
						
							|  |  |  |             except: | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |                 self.report({'WARNING'}, | 
					
						
							|  |  |  |                             "Modifiers cannot be added to object: " + obj.name) | 
					
						
							| 
									
										
										
										
											2009-12-18 23:17:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for obj in context.selected_editable_objects: | 
					
						
							|  |  |  |             set_object_subd(obj) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-24 19:50:43 +00:00
										 |  |  |         return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2009-11-26 15:36:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class ShapeTransfer(Operator): | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     '''Copy another selected objects active shape to this one by ''' \ | 
					
						
							|  |  |  |     '''applying the relative offsets''' | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bl_idname = "object.shape_key_transfer" | 
					
						
							|  |  |  |     bl_label = "Transfer Shape Key" | 
					
						
							| 
									
										
										
										
											2010-03-01 00:03:51 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     mode = EnumProperty( | 
					
						
							|  |  |  |             items=(('OFFSET', | 
					
						
							|  |  |  |                     "Offset", | 
					
						
							|  |  |  |                     "Apply the relative positional offset", | 
					
						
							|  |  |  |                     ), | 
					
						
							|  |  |  |                    ('RELATIVE_FACE', | 
					
						
							|  |  |  |                     "Relative Face", | 
					
						
							|  |  |  |                     "Calculate relative position (using faces).", | 
					
						
							|  |  |  |                     ), | 
					
						
							|  |  |  |                    ('RELATIVE_EDGE', | 
					
						
							|  |  |  |                    "Relative Edge", | 
					
						
							|  |  |  |                    "Calculate relative position (using edges).", | 
					
						
							|  |  |  |                    ), | 
					
						
							|  |  |  |                    ), | 
					
						
							|  |  |  |             name="Transformation Mode", | 
					
						
							|  |  |  |             description="Relative shape positions to the new shape method", | 
					
						
							|  |  |  |             default='OFFSET', | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |     use_clamp = BoolProperty( | 
					
						
							|  |  |  |             name="Clamp Offset", | 
					
						
							|  |  |  |             description=("Clamp the transformation to the distance each " | 
					
						
							|  |  |  |                          "vertex moves in the original shape."), | 
					
						
							|  |  |  |             default=False, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _main(self, ob_act, objects, mode='OFFSET', use_clamp=False): | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |         def me_nos(verts): | 
					
						
							|  |  |  |             return [v.normal.copy() for v in verts] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def me_cos(verts): | 
					
						
							|  |  |  |             return [v.co.copy() for v in verts] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 12:19:46 +00:00
										 |  |  |         def ob_add_shape(ob, name): | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |             me = ob.data | 
					
						
							| 
									
										
										
										
											2010-12-08 11:42:11 +00:00
										 |  |  |             key = ob.shape_key_add(from_mix=False) | 
					
						
							| 
									
										
										
										
											2011-03-25 02:12:44 +00:00
										 |  |  |             if len(me.shape_keys.key_blocks) == 1: | 
					
						
							| 
									
										
										
										
											2010-02-01 15:09:35 +00:00
										 |  |  |                 key.name = "Basis" | 
					
						
							| 
									
										
										
										
											2010-12-08 11:42:11 +00:00
										 |  |  |                 key = ob.shape_key_add(from_mix=False)  # we need a rest | 
					
						
							| 
									
										
										
										
											2010-01-06 12:19:46 +00:00
										 |  |  |             key.name = name | 
					
						
							| 
									
										
										
										
											2011-03-25 02:12:44 +00:00
										 |  |  |             ob.active_shape_key_index = len(me.shape_keys.key_blocks) - 1 | 
					
						
							| 
									
										
										
										
											2010-10-27 22:15:55 +00:00
										 |  |  |             ob.show_only_shape_key = True | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |         from mathutils.geometry import barycentric_transform | 
					
						
							| 
									
										
										
										
											2010-04-11 14:22:27 +00:00
										 |  |  |         from mathutils import Vector | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if use_clamp and mode == 'OFFSET': | 
					
						
							|  |  |  |             use_clamp = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         me = ob_act.data | 
					
						
							| 
									
										
										
										
											2010-01-06 12:19:46 +00:00
										 |  |  |         orig_key_name = ob_act.active_shape_key.name | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         orig_shape_coords = me_cos(ob_act.active_shape_key.data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |         orig_normals = me_nos(me.vertices) | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |         # the actual mverts location isnt as relyable as the base shape :S | 
					
						
							|  |  |  |         # orig_coords = me_cos(me.vertices) | 
					
						
							| 
									
										
										
										
											2011-03-25 02:12:44 +00:00
										 |  |  |         orig_coords = me_cos(me.shape_keys.key_blocks[0].data) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for ob_other in objects: | 
					
						
							|  |  |  |             me_other = ob_other.data | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |             if len(me_other.vertices) != len(me.vertices): | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |                 self.report({'WARNING'}, | 
					
						
							|  |  |  |                             ("Skipping '%s', " | 
					
						
							|  |  |  |                              "vertex count differs") % ob_other.name) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |             target_normals = me_nos(me_other.vertices) | 
					
						
							| 
									
										
										
										
											2010-02-05 14:29:05 +00:00
										 |  |  |             if me_other.shape_keys: | 
					
						
							| 
									
										
										
										
											2011-03-25 02:12:44 +00:00
										 |  |  |                 target_coords = me_cos(me_other.shape_keys.key_blocks[0].data) | 
					
						
							| 
									
										
										
										
											2010-02-05 14:29:05 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |                 target_coords = me_cos(me_other.vertices) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 12:19:46 +00:00
										 |  |  |             ob_add_shape(ob_other, orig_key_name) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # editing the final coords, only list that stores wrapped coords | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |             target_shape_coords = [v.co for v in | 
					
						
							|  |  |  |                                    ob_other.active_shape_key.data] | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |             median_coords = [[] for i in range(len(me.vertices))] | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Method 1, edge | 
					
						
							|  |  |  |             if mode == 'OFFSET': | 
					
						
							|  |  |  |                 for i, vert_cos in enumerate(median_coords): | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                     vert_cos.append(target_coords[i] + | 
					
						
							|  |  |  |                                     (orig_shape_coords[i] - orig_coords[i])) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             elif mode == 'RELATIVE_FACE': | 
					
						
							|  |  |  |                 for face in me.faces: | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |                     i1, i2, i3, i4 = face.vertices_raw | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                     if i4 != 0: | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i1], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i4], | 
					
						
							|  |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i4], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i1].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i2], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i2].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i3], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    orig_coords[i4], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i4], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i3].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i4], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    orig_coords[i4], | 
					
						
							|  |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i4], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i4].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i1], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i1].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i2], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i2].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                         pt = barycentric_transform(orig_shape_coords[i3], | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                                    orig_coords[i2], | 
					
						
							|  |  |  |                                                    orig_coords[i3], | 
					
						
							|  |  |  |                                                    orig_coords[i1], | 
					
						
							|  |  |  |                                                    target_coords[i2], | 
					
						
							|  |  |  |                                                    target_coords[i3], | 
					
						
							|  |  |  |                                                    target_coords[i1], | 
					
						
							|  |  |  |                                                    ) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         median_coords[i3].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             elif mode == 'RELATIVE_EDGE': | 
					
						
							|  |  |  |                 for ed in me.edges: | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |                     i1, i2 = ed.vertices | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                     v1, v2 = orig_coords[i1], orig_coords[i2] | 
					
						
							|  |  |  |                     edge_length = (v1 - v2).length | 
					
						
							|  |  |  |                     n1loc = v1 + orig_normals[i1] * edge_length | 
					
						
							|  |  |  |                     n2loc = v2 + orig_normals[i2] * edge_length | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     # now get the target nloc's | 
					
						
							|  |  |  |                     v1_to, v2_to = target_coords[i1], target_coords[i2] | 
					
						
							|  |  |  |                     edlen_to = (v1_to - v2_to).length | 
					
						
							|  |  |  |                     n1loc_to = v1_to + target_normals[i1] * edlen_to | 
					
						
							|  |  |  |                     n2loc_to = v2_to + target_normals[i2] * edlen_to | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                     pt = barycentric_transform(orig_shape_coords[i1], | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         v2, v1, n1loc, | 
					
						
							|  |  |  |                         v2_to, v1_to, n1loc_to) | 
					
						
							|  |  |  |                     median_coords[i1].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 02:11:38 +00:00
										 |  |  |                     pt = barycentric_transform(orig_shape_coords[i2], | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         v1, v2, n2loc, | 
					
						
							|  |  |  |                         v1_to, v2_to, n2loc_to) | 
					
						
							|  |  |  |                     median_coords[i2].append(pt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # apply the offsets to the new shape | 
					
						
							|  |  |  |             from functools import reduce | 
					
						
							|  |  |  |             VectorAdd = Vector.__add__ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for i, vert_cos in enumerate(median_coords): | 
					
						
							|  |  |  |                 if vert_cos: | 
					
						
							|  |  |  |                     co = reduce(VectorAdd, vert_cos) / len(vert_cos) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if use_clamp: | 
					
						
							|  |  |  |                         # clamp to the same movement as the original | 
					
						
							|  |  |  |                         # breaks copy between different scaled meshes. | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                         len_from = (orig_shape_coords[i] - | 
					
						
							|  |  |  |                                     orig_coords[i]).length | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |                         ofs = co - target_coords[i] | 
					
						
							|  |  |  |                         ofs.length = len_from | 
					
						
							|  |  |  |                         co = target_coords[i] + ofs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     target_shape_coords[i][:] = co | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2009-12-27 23:29:34 +00:00
										 |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         return (obj and obj.mode != 'EDIT') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         C = bpy.context | 
					
						
							|  |  |  |         ob_act = C.active_object | 
					
						
							|  |  |  |         objects = [ob for ob in C.selected_editable_objects if ob != ob_act] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  |         if 1:  # swap from/to, means we cant copy to many at once. | 
					
						
							| 
									
										
										
										
											2010-02-05 14:29:05 +00:00
										 |  |  |             if len(objects) != 1: | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |                 self.report({'ERROR'}, | 
					
						
							|  |  |  |                             ("Expected one other selected " | 
					
						
							|  |  |  |                              "mesh object to copy from")) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-05 14:29:05 +00:00
										 |  |  |                 return {'CANCELLED'} | 
					
						
							|  |  |  |             ob_act, objects = objects[0], [ob_act] | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-05 14:29:05 +00:00
										 |  |  |         if ob_act.type != 'MESH': | 
					
						
							|  |  |  |             self.report({'ERROR'}, "Other object is not a mesh.") | 
					
						
							|  |  |  |             return {'CANCELLED'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ob_act.active_shape_key is None: | 
					
						
							|  |  |  |             self.report({'ERROR'}, "Other object has no shape key") | 
					
						
							|  |  |  |             return {'CANCELLED'} | 
					
						
							| 
									
										
										
										
											2010-09-09 18:03:57 +00:00
										 |  |  |         return self._main(ob_act, objects, self.mode, self.use_clamp) | 
					
						
							| 
									
										
										
										
											2009-12-27 11:14:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-06 01:40:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class JoinUVs(Operator): | 
					
						
							| 
									
										
										
										
											2010-01-14 14:06:27 +00:00
										 |  |  |     '''Copy UV Layout to objects with matching geometry''' | 
					
						
							|  |  |  |     bl_idname = "object.join_uvs" | 
					
						
							|  |  |  |     bl_label = "Join as UVs" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2010-01-14 14:06:27 +00:00
										 |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         return (obj and obj.type == 'MESH') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _main(self, context): | 
					
						
							|  |  |  |         import array | 
					
						
							|  |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         mesh = obj.data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         is_editmode = (obj.mode == 'EDIT') | 
					
						
							|  |  |  |         if is_editmode: | 
					
						
							|  |  |  |             bpy.ops.object.mode_set(mode='OBJECT', toggle=False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-23 22:16:45 +00:00
										 |  |  |         if not mesh.uv_textures: | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |             self.report({'WARNING'}, | 
					
						
							|  |  |  |                         "Object: %s, Mesh: '%s' has no UVs" | 
					
						
							|  |  |  |                         % (obj.name, mesh.name)) | 
					
						
							| 
									
										
										
										
											2010-02-17 16:31:37 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             len_faces = len(mesh.faces) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |             # seems to be the fastest way to create an array | 
					
						
							|  |  |  |             uv_array = array.array('f', [0.0] * 8) * len_faces | 
					
						
							| 
									
										
										
										
											2010-08-23 22:16:45 +00:00
										 |  |  |             mesh.uv_textures.active.data.foreach_get("uv_raw", uv_array) | 
					
						
							| 
									
										
										
										
											2010-02-17 16:31:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             objects = context.selected_editable_objects[:] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for obj_other in objects: | 
					
						
							|  |  |  |                 if obj_other.type == 'MESH': | 
					
						
							|  |  |  |                     obj_other.data.tag = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for obj_other in objects: | 
					
						
							|  |  |  |                 if obj_other != obj and obj_other.type == 'MESH': | 
					
						
							|  |  |  |                     mesh_other = obj_other.data | 
					
						
							|  |  |  |                     if mesh_other != mesh: | 
					
						
							|  |  |  |                         if mesh_other.tag == False: | 
					
						
							|  |  |  |                             mesh_other.tag = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             if len(mesh_other.faces) != len_faces: | 
					
						
							| 
									
										
										
										
											2011-07-28 07:55:09 +00:00
										 |  |  |                                 self.report({'WARNING'}, "Object: %s, Mesh: " | 
					
						
							|  |  |  |                                             "'%s' has %d faces, expected %d\n" | 
					
						
							|  |  |  |                                             % (obj_other.name, | 
					
						
							|  |  |  |                                                mesh_other.name, | 
					
						
							|  |  |  |                                                len(mesh_other.faces), | 
					
						
							|  |  |  |                                                len_faces), | 
					
						
							|  |  |  |                                                ) | 
					
						
							| 
									
										
										
										
											2010-02-17 16:31:37 +00:00
										 |  |  |                             else: | 
					
						
							| 
									
										
										
										
											2010-08-23 22:16:45 +00:00
										 |  |  |                                 uv_other = mesh_other.uv_textures.active | 
					
						
							| 
									
										
										
										
											2010-02-17 16:31:37 +00:00
										 |  |  |                                 if not uv_other: | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |                                     # should return the texture it adds | 
					
						
							|  |  |  |                                     uv_other = mesh_other.uv_textures.new() | 
					
						
							| 
									
										
										
										
											2010-01-31 14:46:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 16:31:37 +00:00
										 |  |  |                                 # finally do the copy | 
					
						
							|  |  |  |                                 uv_other.data.foreach_set("uv_raw", uv_array) | 
					
						
							| 
									
										
										
										
											2010-01-14 14:06:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if is_editmode: | 
					
						
							|  |  |  |             bpy.ops.object.mode_set(mode='EDIT', toggle=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         self._main(context) | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-06 01:40:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class MakeDupliFace(Operator): | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |     '''Make linked objects into dupli-faces''' | 
					
						
							|  |  |  |     bl_idname = "object.make_dupli_face" | 
					
						
							| 
									
										
										
										
											2010-12-19 07:14:42 +00:00
										 |  |  |     bl_label = "Make Dupli-Face" | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 01:37:09 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def poll(cls, context): | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |         obj = context.active_object | 
					
						
							|  |  |  |         return (obj and obj.type == 'MESH') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _main(self, context): | 
					
						
							| 
									
										
										
										
											2010-04-11 14:22:27 +00:00
										 |  |  |         from mathutils import Vector | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         SCALE_FAC = 0.01 | 
					
						
							|  |  |  |         offset = 0.5 * SCALE_FAC | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |         base_tri = (Vector((-offset, -offset, 0.0)), | 
					
						
							|  |  |  |                     Vector((+offset, -offset, 0.0)), | 
					
						
							|  |  |  |                     Vector((+offset, +offset, 0.0)), | 
					
						
							|  |  |  |                     Vector((-offset, +offset, 0.0)), | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def matrix_to_quat(matrix): | 
					
						
							|  |  |  |             # scale = matrix.median_scale | 
					
						
							| 
									
										
										
										
											2011-02-05 07:04:23 +00:00
										 |  |  |             trans = matrix.to_translation() | 
					
						
							|  |  |  |             rot = matrix.to_3x3()  # also contains scale | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 09:31:39 +00:00
										 |  |  |             return [(rot * b) + trans for b in base_tri] | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |         scene = bpy.context.scene | 
					
						
							|  |  |  |         linked = {} | 
					
						
							|  |  |  |         for obj in bpy.context.selected_objects: | 
					
						
							|  |  |  |             data = obj.data | 
					
						
							|  |  |  |             if data: | 
					
						
							|  |  |  |                 linked.setdefault(data, []).append(obj) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for data, objects in linked.items(): | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |             face_verts = [axis for obj in objects | 
					
						
							|  |  |  |                           for v in matrix_to_quat(obj.matrix_world) | 
					
						
							|  |  |  |                           for axis in v] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-26 22:44:05 +00:00
										 |  |  |             faces = list(range(len(face_verts) // 3)) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             mesh = bpy.data.meshes.new(data.name + "_dupli") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-26 22:44:05 +00:00
										 |  |  |             mesh.vertices.add(len(face_verts) // 3) | 
					
						
							|  |  |  |             mesh.faces.add(len(face_verts) // 12) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-18 03:42:26 +00:00
										 |  |  |             mesh.vertices.foreach_set("co", face_verts) | 
					
						
							|  |  |  |             mesh.faces.foreach_set("vertices_raw", faces) | 
					
						
							| 
									
										
										
										
											2011-02-08 21:32:26 +00:00
										 |  |  |             mesh.update()  # generates edge data | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  |             # pick an object to use | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |             obj = objects[0] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-22 00:07:46 +00:00
										 |  |  |             ob_new = bpy.data.objects.new(mesh.name, mesh) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |             base = scene.objects.link(ob_new) | 
					
						
							|  |  |  |             base.layers[:] = obj.layers | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-22 00:07:46 +00:00
										 |  |  |             ob_inst = bpy.data.objects.new(data.name, data) | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |             base = scene.objects.link(ob_inst) | 
					
						
							|  |  |  |             base.layers[:] = obj.layers | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |             for obj in objects: | 
					
						
							|  |  |  |                 scene.objects.unlink(obj) | 
					
						
							| 
									
										
										
										
											2010-02-22 23:32:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 16:10:47 +00:00
										 |  |  |             ob_new.dupli_type = 'FACES' | 
					
						
							|  |  |  |             ob_inst.parent = ob_new | 
					
						
							|  |  |  |             ob_new.use_dupli_faces_scale = True | 
					
						
							|  |  |  |             ob_new.dupli_faces_scale = 1.0 / SCALE_FAC | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         self._main(context) | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-14 14:06:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class IsolateTypeRender(Operator): | 
					
						
							| 
									
										
										
										
											2011-07-25 06:40:16 +00:00
										 |  |  |     '''Hide unselected render objects of same type as active ''' \ | 
					
						
							|  |  |  |     '''by setting the hide render flag''' | 
					
						
							| 
									
										
										
										
											2010-05-20 15:30:28 +00:00
										 |  |  |     bl_idname = "object.isolate_type_render" | 
					
						
							| 
									
										
										
										
											2010-07-30 23:25:26 +00:00
										 |  |  |     bl_label = "Restrict Render Unselected" | 
					
						
							| 
									
										
										
										
											2010-05-20 15:30:28 +00:00
										 |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         act_type = context.object.type | 
					
						
							| 
									
										
										
										
											2010-06-09 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-20 15:30:28 +00:00
										 |  |  |         for obj in context.visible_objects: | 
					
						
							| 
									
										
										
										
											2010-06-09 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:56:04 +00:00
										 |  |  |             if obj.select: | 
					
						
							|  |  |  |                 obj.hide_render = False | 
					
						
							| 
									
										
										
										
											2010-05-20 15:30:28 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 if obj.type == act_type: | 
					
						
							| 
									
										
										
										
											2010-07-15 16:56:04 +00:00
										 |  |  |                     obj.hide_render = True | 
					
						
							| 
									
										
										
										
											2010-05-20 15:30:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return {'FINISHED'} | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 18:59:45 +00:00
										 |  |  | class ClearAllRestrictRender(Operator): | 
					
						
							| 
									
										
										
										
											2010-07-30 23:25:26 +00:00
										 |  |  |     '''Reveal all render objects by setting the hide render flag''' | 
					
						
							|  |  |  |     bl_idname = "object.hide_render_clear_all" | 
					
						
							|  |  |  |     bl_label = "Clear All Restrict Render" | 
					
						
							|  |  |  |     bl_options = {'REGISTER', 'UNDO'} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def execute(self, context): | 
					
						
							|  |  |  |         for obj in context.scene.objects: | 
					
						
							| 
									
										
										
										
											2010-09-07 15:17:42 +00:00
										 |  |  |             obj.hide_render = False | 
					
						
							| 
									
										
										
										
											2010-07-30 23:25:26 +00:00
										 |  |  |         return {'FINISHED'} |