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-12 06:57:00 +00:00
|
|
|
from bpy.types import Operator
|
2015-01-27 17:46:07 +11:00
|
|
|
from bpy.props import (
|
2017-03-25 11:07:05 +11:00
|
|
|
BoolProperty,
|
|
|
|
EnumProperty,
|
|
|
|
FloatProperty,
|
|
|
|
IntProperty,
|
|
|
|
StringProperty,
|
|
|
|
)
|
2009-11-28 23:37:56 +00:00
|
|
|
|
|
|
|
|
2011-08-12 06:57:00 +00:00
|
|
|
class SelectPattern(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Select objects 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
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
pattern: StringProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
name="Pattern",
|
|
|
|
description="Name filter using '*', '?' and "
|
|
|
|
"'[abc]' unix style wildcards",
|
|
|
|
maxlen=64,
|
|
|
|
default="*",
|
|
|
|
)
|
2018-07-11 22:18:09 +02:00
|
|
|
case_sensitive: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
name="Case Sensitive",
|
|
|
|
description="Do a case sensitive compare",
|
|
|
|
default=False,
|
|
|
|
)
|
2018-07-11 22:18:09 +02:00
|
|
|
extend: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
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:
|
2012-02-02 05:11:42 +00:00
|
|
|
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
|
2018-04-05 18:20:27 +02:00
|
|
|
if obj and obj.mode == 'POSE':
|
2009-12-01 14:48:36 +00:00
|
|
|
items = obj.data.bones
|
2011-07-28 07:55:09 +00:00
|
|
|
if not self.extend:
|
|
|
|
bpy.ops.pose.select_all(action='DESELECT')
|
2018-04-05 18:20:27 +02:00
|
|
|
elif obj and obj.type == 'ARMATURE' and obj.mode == 'EDIT':
|
2009-12-01 14:48:36 +00:00
|
|
|
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):
|
2011-07-28 07:55:09 +00:00
|
|
|
|
|
|
|
# hrmf, perhaps there should be a utility function for this.
|
|
|
|
if is_ebone:
|
Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
Option to add objects based on their names
* Collection Manager operators
The existing buttons are placeholders
* Collection Manager drawing
The editor main region is empty
* Collection Override
* Per-Collection engine settings
This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
We are replacing Base by a new extended Base while keeping it backward
compatible with some legacy settings (i.e., lay, flag_legacy).
Renamed all Base to BaseLegacy to make it clear the areas of code that
still need to be converted
Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
- read, write, add/copy/remove objects, copy scene, collection
link/unlinking, context)
* New Editor: Collection Manager
Based on patch by Julian Eisel
This is extracted from the layer-manager branch. With the following changes:
- Renamed references of layer manager to collections manager
- I doesn't include the editors/space_collections/ draw and util files
- The drawing code itself will be implemented separately by Julian
* Base / Object:
A little note about them. Original Blender code would try to keep them
in sync through the code, juggling flags back and forth. This will now
be handled by Depsgraph, keeping Object and Bases more separated
throughout the non-rendering code.
Scene.base is being cleared in doversion, and the old viewport drawing
code was poorly converted to use the new bases while the new viewport
code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
2017-02-07 10:18:38 +01:00
|
|
|
item.select = True
|
2011-07-28 07:55:09 +00:00
|
|
|
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
|
Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
Option to add objects based on their names
* Collection Manager operators
The existing buttons are placeholders
* Collection Manager drawing
The editor main region is empty
* Collection Override
* Per-Collection engine settings
This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
We are replacing Base by a new extended Base while keeping it backward
compatible with some legacy settings (i.e., lay, flag_legacy).
Renamed all Base to BaseLegacy to make it clear the areas of code that
still need to be converted
Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
- read, write, add/copy/remove objects, copy scene, collection
link/unlinking, context)
* New Editor: Collection Manager
Based on patch by Julian Eisel
This is extracted from the layer-manager branch. With the following changes:
- Renamed references of layer manager to collections manager
- I doesn't include the editors/space_collections/ draw and util files
- The drawing code itself will be implemented separately by Julian
* Base / Object:
A little note about them. Original Blender code would try to keep them
in sync through the code, juggling flags back and forth. This will now
be handled by Depsgraph, keeping Object and Bases more separated
throughout the non-rendering code.
Scene.base is being cleared in doversion, and the old viewport drawing
code was poorly converted to use the new bases while the new viewport
code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
2017-02-07 10:18:38 +01:00
|
|
|
else:
|
2018-11-11 11:22:38 +01:00
|
|
|
item.select_set(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-12 06:57:00 +00:00
|
|
|
class SelectCamera(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Select the active camera"""
|
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
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
extend: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
name="Extend",
|
|
|
|
description="Extend the selection",
|
|
|
|
default=False
|
|
|
|
)
|
2013-01-12 10:48:10 +00:00
|
|
|
|
2010-02-11 21:17:05 +00:00
|
|
|
def execute(self, context):
|
|
|
|
scene = context.scene
|
2018-04-23 14:26:07 +02:00
|
|
|
view_layer = context.view_layer
|
2012-05-14 14:39:21 +00:00
|
|
|
view = context.space_data
|
|
|
|
if view.type == 'VIEW_3D' and not view.lock_camera_and_layers:
|
|
|
|
camera = view.camera
|
|
|
|
else:
|
|
|
|
camera = scene.camera
|
|
|
|
|
|
|
|
if camera is None:
|
|
|
|
self.report({'WARNING'}, "No camera found")
|
|
|
|
elif camera.name not in scene.objects:
|
2010-02-11 21:17:05 +00:00
|
|
|
self.report({'WARNING'}, "Active camera is not in this scene")
|
2012-05-14 14:39:21 +00:00
|
|
|
else:
|
2013-01-12 10:48:10 +00:00
|
|
|
if not self.extend:
|
|
|
|
bpy.ops.object.select_all(action='DESELECT')
|
2018-04-23 14:26:07 +02:00
|
|
|
view_layer.objects.active = camera
|
|
|
|
# camera.hide = False # XXX TODO where is this now?
|
2018-11-11 11:22:38 +01:00
|
|
|
camera.select_set(True)
|
2012-05-14 14:39:21 +00:00
|
|
|
return {'FINISHED'}
|
2010-02-11 21:17:05 +00:00
|
|
|
|
2012-05-14 14:39:21 +00:00
|
|
|
return {'CANCELLED'}
|
2010-02-11 21:17:05 +00:00
|
|
|
|
|
|
|
|
2011-08-12 06:57:00 +00:00
|
|
|
class SelectHierarchy(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Select object relative to the active object's position """ \
|
2018-06-26 19:41:37 +02:00
|
|
|
"""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
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
direction: EnumProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
items=(('PARENT', "Parent", ""),
|
|
|
|
('CHILD', "Child", ""),
|
|
|
|
),
|
|
|
|
name="Direction",
|
|
|
|
description="Direction to select in the hierarchy",
|
2018-07-11 22:18:09 +02:00
|
|
|
default='PARENT',
|
|
|
|
)
|
|
|
|
extend: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
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):
|
2018-04-23 14:26:07 +02:00
|
|
|
view_layer = context.view_layer
|
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
|
|
|
|
2011-10-17 06:58:07 +00:00
|
|
|
# don't edit any object settings above this
|
2010-07-23 05:49:12 +00:00
|
|
|
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:
|
2018-11-11 11:22:38 +01:00
|
|
|
obj.select_set(True)
|
2010-07-23 05:49:12 +00:00
|
|
|
|
2018-04-23 14:26:07 +02:00
|
|
|
view_layer.objects.active = act_new
|
2010-07-23 05:49:12 +00:00
|
|
|
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-12 06:57:00 +00:00
|
|
|
class SubdivisionSet(Operator):
|
2012-07-03 09:02:41 +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
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
level: IntProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
name="Level",
|
|
|
|
min=-100, max=100,
|
|
|
|
soft_min=-6, soft_max=6,
|
|
|
|
default=1,
|
|
|
|
)
|
2018-07-11 22:18:09 +02:00
|
|
|
relative: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
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:
|
2014-05-06 16:54:22 +03:00
|
|
|
if level > mod.total_levels:
|
2014-07-22 12:03:15 +10:00
|
|
|
sub = level - mod.total_levels
|
2018-10-25 12:03:34 +11:00
|
|
|
for _ in range(sub):
|
2014-07-22 12:03:15 +10:00
|
|
|
bpy.ops.object.multires_subdivide(modifier="Multires")
|
2014-05-06 16:54:22 +03:00
|
|
|
|
2018-04-05 18:20:27 +02:00
|
|
|
if obj.mode == 'SCULPT':
|
2014-05-06 16:54:22 +03:00
|
|
|
if mod.sculpt_levels != level:
|
|
|
|
mod.sculpt_levels = level
|
2018-04-05 18:20:27 +02:00
|
|
|
elif obj.mode == 'OBJECT':
|
2014-05-06 16:54:22 +03:00
|
|
|
if mod.levels != level:
|
|
|
|
mod.levels = level
|
2010-01-15 05:00:02 +00:00
|
|
|
return
|
|
|
|
else:
|
2018-04-05 18:20:27 +02:00
|
|
|
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
|
2018-04-05 18:20:27 +02: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:
|
2018-04-05 18:20:27 +02:00
|
|
|
if obj.mode == 'SCULPT':
|
2014-05-06 16:54:22 +03:00
|
|
|
mod = obj.modifiers.new("Multires", 'MULTIRES')
|
2014-05-06 17:14:28 +03:00
|
|
|
if level > 0:
|
2018-10-25 12:03:34 +11:00
|
|
|
for _ in range(level):
|
2014-05-06 17:14:28 +03:00
|
|
|
bpy.ops.object.multires_subdivide(modifier="Multires")
|
2014-05-06 16:54:22 +03:00
|
|
|
else:
|
|
|
|
mod = obj.modifiers.new("Subsurf", 'SUBSURF')
|
|
|
|
mod.levels = level
|
2010-12-19 01:14:57 +00:00
|
|
|
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-12 06:57:00 +00:00
|
|
|
class ShapeTransfer(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Copy another selected objects active shape to this one by """ \
|
2018-06-26 19:41:37 +02:00
|
|
|
"""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
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
mode: EnumProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
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',
|
|
|
|
)
|
2018-07-11 22:18:09 +02:00
|
|
|
use_clamp: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
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)
|
2012-02-08 04:37:37 +00:00
|
|
|
# actual mesh vertex location isn't as reliable as the base shape :S
|
2018-06-26 19:41:37 +02:00
|
|
|
# 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:
|
2014-05-23 11:39:22 +10:00
|
|
|
if ob_other.type != 'MESH':
|
|
|
|
self.report({'WARNING'},
|
|
|
|
("Skipping '%s', "
|
|
|
|
"not a mesh") % ob_other.name)
|
|
|
|
continue
|
2009-12-27 11:14:06 +00:00
|
|
|
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':
|
2012-03-27 13:52:30 +00:00
|
|
|
for poly in me.polygons:
|
2013-02-25 03:54:16 +00:00
|
|
|
idxs = poly.vertices[:]
|
2012-03-27 13:52:30 +00:00
|
|
|
v_before = idxs[-2]
|
|
|
|
v = idxs[-1]
|
|
|
|
for v_after in idxs:
|
|
|
|
pt = barycentric_transform(orig_shape_coords[v],
|
|
|
|
orig_coords[v_before],
|
|
|
|
orig_coords[v],
|
|
|
|
orig_coords[v_after],
|
|
|
|
target_coords[v_before],
|
|
|
|
target_coords[v],
|
|
|
|
target_coords[v_after],
|
2011-07-28 07:55:09 +00:00
|
|
|
)
|
2012-03-27 13:52:30 +00:00
|
|
|
median_coords[v].append(pt)
|
|
|
|
v_before = v
|
|
|
|
v = v_after
|
2009-12-27 11:14:06 +00:00
|
|
|
|
|
|
|
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],
|
2012-10-08 08:28:05 +00:00
|
|
|
v2, v1, n1loc,
|
|
|
|
v2_to, v1_to, n1loc_to)
|
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],
|
2012-10-08 08:28:05 +00:00
|
|
|
v1, v2, n2loc,
|
|
|
|
v1_to, v2_to, n2loc_to)
|
2009-12-27 11:14:06 +00:00
|
|
|
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
|
2018-04-05 18:20:27 +02:00
|
|
|
return (obj and obj.mode != 'EDIT')
|
2009-12-27 23:29:34 +00:00
|
|
|
|
2009-12-27 11:14:06 +00:00
|
|
|
def execute(self, context):
|
2012-01-29 21:28:04 +00:00
|
|
|
ob_act = context.active_object
|
2012-02-04 11:10:41 +00:00
|
|
|
objects = [ob for ob in context.selected_editable_objects
|
|
|
|
if ob != ob_act]
|
2009-12-27 11:14:06 +00:00
|
|
|
|
2018-09-03 16:49:08 +02:00
|
|
|
if 1: # swap from/to, means we can't 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':
|
2011-09-19 14:00:42 +00:00
|
|
|
self.report({'ERROR'}, "Other object is not a mesh")
|
2010-02-05 14:29:05 +00:00
|
|
|
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-12 06:57:00 +00:00
|
|
|
class JoinUVs(Operator):
|
2013-12-13 15:29:50 +01:00
|
|
|
"""Transfer UV Maps from active to selected objects """ \
|
2018-06-26 19:41:37 +02:00
|
|
|
"""(needs matching geometry)"""
|
2010-01-14 14:06:27 +00:00
|
|
|
bl_idname = "object.join_uvs"
|
2013-12-13 15:29:50 +01:00
|
|
|
bl_label = "Transfer UV Maps"
|
2014-01-07 18:09:02 +11:00
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2010-01-14 14:06:27 +00:00
|
|
|
|
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
|
|
|
|
|
2018-04-05 18:20:27 +02:00
|
|
|
is_editmode = (obj.mode == 'EDIT')
|
2010-01-14 14:06:27 +00:00
|
|
|
if is_editmode:
|
|
|
|
bpy.ops.object.mode_set(mode='OBJECT', toggle=False)
|
|
|
|
|
2017-05-25 15:11:00 +10:00
|
|
|
if not mesh.uv_layers:
|
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:
|
2012-03-27 13:52:30 +00:00
|
|
|
nbr_loops = len(mesh.loops)
|
2010-02-17 16:31:37 +00:00
|
|
|
|
2011-07-25 06:40:16 +00:00
|
|
|
# seems to be the fastest way to create an array
|
2012-03-27 13:52:30 +00:00
|
|
|
uv_array = array.array('f', [0.0] * 2) * nbr_loops
|
2012-04-22 23:51:50 +00:00
|
|
|
mesh.uv_layers.active.data.foreach_get("uv", 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:
|
2012-10-08 08:28:05 +00:00
|
|
|
if mesh_other.tag is False:
|
2010-02-17 16:31:37 +00:00
|
|
|
mesh_other.tag = True
|
|
|
|
|
2012-03-27 13:52:30 +00:00
|
|
|
if len(mesh_other.loops) != nbr_loops:
|
2011-07-28 07:55:09 +00:00
|
|
|
self.report({'WARNING'}, "Object: %s, Mesh: "
|
2012-03-27 13:52:30 +00:00
|
|
|
"'%s' has %d loops (for %d faces),"
|
|
|
|
" expected %d\n"
|
2011-07-28 07:55:09 +00:00
|
|
|
% (obj_other.name,
|
|
|
|
mesh_other.name,
|
2012-03-27 13:52:30 +00:00
|
|
|
len(mesh_other.loops),
|
|
|
|
len(mesh_other.polygons),
|
|
|
|
nbr_loops,
|
|
|
|
),
|
2012-10-08 08:28:05 +00:00
|
|
|
)
|
2010-02-17 16:31:37 +00:00
|
|
|
else:
|
2012-04-22 23:51:50 +00:00
|
|
|
uv_other = mesh_other.uv_layers.active
|
2010-02-17 16:31:37 +00:00
|
|
|
if not uv_other:
|
2017-05-25 15:11:00 +10:00
|
|
|
mesh_other.uv_layers.new()
|
2012-04-22 23:51:50 +00:00
|
|
|
uv_other = mesh_other.uv_layers.active
|
2012-03-27 13:52:30 +00:00
|
|
|
if not uv_other:
|
|
|
|
self.report({'ERROR'}, "Could not add "
|
|
|
|
"a new UV map tp object "
|
|
|
|
"'%s' (Mesh '%s')\n"
|
|
|
|
% (obj_other.name,
|
|
|
|
mesh_other.name,
|
|
|
|
),
|
|
|
|
)
|
2010-01-31 14:46:28 +00:00
|
|
|
|
2010-02-17 16:31:37 +00:00
|
|
|
# finally do the copy
|
2012-03-27 13:52:30 +00:00
|
|
|
uv_other.data.foreach_set("uv", 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-12 06:57:00 +00:00
|
|
|
class MakeDupliFace(Operator):
|
2014-01-15 10:40:28 +01:00
|
|
|
"""Convert objects into dupli-face instanced"""
|
2010-02-10 16:10:47 +00:00
|
|
|
bl_idname = "object.make_dupli_face"
|
2010-12-19 07:14:42 +00:00
|
|
|
bl_label = "Make Dupli-Face"
|
2014-01-07 18:09:02 +11:00
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2010-02-10 16:10:47 +00:00
|
|
|
|
2015-06-07 17:40:39 +10:00
|
|
|
@staticmethod
|
|
|
|
def _main(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
|
|
|
|
2012-01-29 21:28:04 +00:00
|
|
|
def matrix_to_quad(matrix):
|
2010-02-10 16:10:47 +00:00
|
|
|
# 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
|
|
|
|
2018-08-12 15:01:26 +10:00
|
|
|
return [(rot @ b) + trans for b in base_tri]
|
2012-01-29 21:28:04 +00:00
|
|
|
scene = context.scene
|
2010-02-10 16:10:47 +00:00
|
|
|
linked = {}
|
2012-01-29 21:28:04 +00:00
|
|
|
for obj in context.selected_objects:
|
2010-02-10 16:10:47 +00:00
|
|
|
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
|
2012-01-29 21:28:04 +00:00
|
|
|
for v in matrix_to_quad(obj.matrix_world)
|
2011-07-25 06:40:16 +00:00
|
|
|
for axis in v]
|
2012-03-27 13:52:30 +00:00
|
|
|
nbr_verts = len(face_verts) // 3
|
|
|
|
nbr_faces = nbr_verts // 4
|
2011-07-25 06:40:16 +00:00
|
|
|
|
2012-03-27 13:52:30 +00:00
|
|
|
faces = list(range(nbr_verts))
|
2010-02-10 16:10:47 +00:00
|
|
|
|
|
|
|
mesh = bpy.data.meshes.new(data.name + "_dupli")
|
|
|
|
|
2012-03-27 13:52:30 +00:00
|
|
|
mesh.vertices.add(nbr_verts)
|
|
|
|
mesh.loops.add(nbr_faces * 4) # Safer than nbr_verts.
|
|
|
|
mesh.polygons.add(nbr_faces)
|
2010-08-26 22:44:05 +00:00
|
|
|
|
2010-08-18 03:42:26 +00:00
|
|
|
mesh.vertices.foreach_set("co", face_verts)
|
2012-03-27 13:52:30 +00:00
|
|
|
mesh.loops.foreach_set("vertex_index", faces)
|
|
|
|
mesh.polygons.foreach_set("loop_start", range(0, nbr_faces * 4, 4))
|
|
|
|
mesh.polygons.foreach_set("loop_total", (4,) * nbr_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
|
|
|
|
|
2018-11-11 11:22:38 +01:00
|
|
|
ob_inst.select_set(True)
|
|
|
|
ob_new.select_set(True)
|
2015-06-07 17:44:25 +10:00
|
|
|
|
2010-02-10 16:10:47 +00:00
|
|
|
def execute(self, context):
|
|
|
|
self._main(context)
|
|
|
|
return {'FINISHED'}
|
|
|
|
|
2010-01-14 14:06:27 +00:00
|
|
|
|
2011-08-12 06:57:00 +00:00
|
|
|
class IsolateTypeRender(Operator):
|
2012-02-08 04:37:37 +00:00
|
|
|
"""Hide unselected render objects of same type as active """ \
|
2018-06-26 19:41:37 +02:00
|
|
|
"""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
|
|
|
|
Render Layers and Collections (merge from render-layers)
Design Documents
----------------
* https://wiki.blender.org/index.php/Dev:2.8/Source/Layers
* https://wiki.blender.org/index.php/Dev:2.8/Source/DataDesignRevised
User Commit Log
---------------
* New Layer and Collection system to replace render layers and viewport layers.
* A layer is a set of collections of objects (and their drawing options) required for specific tasks.
* A collection is a set of objects, equivalent of the old layers in Blender. A collection can be shared across multiple layers.
* All Scenes have a master collection that all other collections are children of.
* New collection "context" tab (in Properties Editor)
* New temporary viewport "collections" panel to control per-collection
visibility
Missing User Features
---------------------
* Collection "Filter"
Option to add objects based on their names
* Collection Manager operators
The existing buttons are placeholders
* Collection Manager drawing
The editor main region is empty
* Collection Override
* Per-Collection engine settings
This will come as a separate commit, as part of the clay-engine branch
Dev Commit Log
--------------
* New DNA file (DNA_layer_types.h) with the new structs
We are replacing Base by a new extended Base while keeping it backward
compatible with some legacy settings (i.e., lay, flag_legacy).
Renamed all Base to BaseLegacy to make it clear the areas of code that
still need to be converted
Note: manual changes were required on - deg_builder_nodes.h, rna_object.c, KX_Light.cpp
* Unittesting for main syncronization requirements
- read, write, add/copy/remove objects, copy scene, collection
link/unlinking, context)
* New Editor: Collection Manager
Based on patch by Julian Eisel
This is extracted from the layer-manager branch. With the following changes:
- Renamed references of layer manager to collections manager
- I doesn't include the editors/space_collections/ draw and util files
- The drawing code itself will be implemented separately by Julian
* Base / Object:
A little note about them. Original Blender code would try to keep them
in sync through the code, juggling flags back and forth. This will now
be handled by Depsgraph, keeping Object and Bases more separated
throughout the non-rendering code.
Scene.base is being cleared in doversion, and the old viewport drawing
code was poorly converted to use the new bases while the new viewport
code doesn't get merged and replace the old one.
Python API Changes
------------------
```
- scene.layers
+ # no longer exists
- scene.objects
+ scene.scene_layers.active.objects
- scene.objects.active
+ scene.render_layers.active.objects.active
- bpy.context.scene.objects.link()
+ bpy.context.scene_collection.objects.link()
- bpy_extras.object_utils.object_data_add(context, obdata, operator=None, use_active_layer=True, name=None)
+ bpy_extras.object_utils.object_data_add(context, obdata, operator=None, name=None)
- bpy.context.object.select
+ bpy.context.object.select = True
+ bpy.context.object.select = False
+ bpy.context.object.select_get()
+ bpy.context.object.select_set(action='SELECT')
+ bpy.context.object.select_set(action='DESELECT')
-AddObjectHelper.layers
+ # no longer exists
```
2017-02-07 10:18:38 +01:00
|
|
|
if obj.select_get():
|
2010-07-15 16:56:04 +00:00
|
|
|
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-12 06:57:00 +00:00
|
|
|
class ClearAllRestrictRender(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Reveal all render objects by setting the hide render flag"""
|
2010-07-30 23:25:26 +00:00
|
|
|
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'}
|
2011-05-24 11:15:21 +00:00
|
|
|
|
2011-08-30 10:49:58 +00:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
class TransformsToDeltas(Operator):
|
|
|
|
"""Convert normal object transforms to delta transforms, """ \
|
2018-06-26 19:41:37 +02:00
|
|
|
"""any existing delta transforms will be included as well"""
|
2016-07-04 12:52:51 +12:00
|
|
|
bl_idname = "object.transforms_to_deltas"
|
|
|
|
bl_label = "Transforms to Deltas"
|
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2018-07-11 22:18:09 +02:00
|
|
|
mode: EnumProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
items=(
|
|
|
|
('ALL', "All Transforms", "Transfer location, rotation, and scale transforms"),
|
|
|
|
('LOC', "Location", "Transfer location transforms only"),
|
|
|
|
('ROT', "Rotation", "Transfer rotation transforms only"),
|
|
|
|
('SCALE', "Scale", "Transfer scale transforms only"),
|
|
|
|
),
|
|
|
|
name="Mode",
|
|
|
|
description="Which transforms to transfer",
|
|
|
|
default='ALL',
|
|
|
|
)
|
2018-07-11 22:18:09 +02:00
|
|
|
reset_values: BoolProperty(
|
2018-06-26 19:41:37 +02:00
|
|
|
name="Reset Values",
|
|
|
|
description=("Clear transform values after transferring to deltas"),
|
|
|
|
default=True,
|
|
|
|
)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
obs = context.selected_editable_objects
|
|
|
|
return (obs is not None)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
def execute(self, context):
|
|
|
|
for obj in context.selected_editable_objects:
|
2016-07-08 00:28:41 +10:00
|
|
|
if self.mode in {'ALL', 'LOC'}:
|
2016-07-04 12:52:51 +12:00
|
|
|
self.transfer_location(obj)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
|
|
|
if self.mode in {'ALL', 'ROT'}:
|
2016-07-04 12:52:51 +12:00
|
|
|
self.transfer_rotation(obj)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
|
|
|
if self.mode in {'ALL', 'SCALE'}:
|
2016-07-04 12:52:51 +12:00
|
|
|
self.transfer_scale(obj)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
return {'FINISHED'}
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
def transfer_location(self, obj):
|
|
|
|
obj.delta_location += obj.location
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
if self.reset_values:
|
|
|
|
obj.location.zero()
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
def transfer_rotation(self, obj):
|
|
|
|
# TODO: add transforms together...
|
|
|
|
if obj.rotation_mode == 'QUATERNION':
|
|
|
|
obj.delta_rotation_quaternion += obj.rotation_quaternion
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
if self.reset_values:
|
2016-07-08 00:28:41 +10:00
|
|
|
obj.rotation_quaternion.identity()
|
2016-07-04 12:52:51 +12:00
|
|
|
elif obj.rotation_mode == 'AXIS_ANGLE':
|
|
|
|
pass # Unsupported
|
|
|
|
else:
|
|
|
|
delta = obj.delta_rotation_euler.copy()
|
2016-07-08 00:28:41 +10:00
|
|
|
obj.delta_rotation_euler = obj.rotation_euler
|
2016-07-04 12:52:51 +12:00
|
|
|
obj.delta_rotation_euler.rotate(delta)
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
if self.reset_values:
|
|
|
|
obj.rotation_euler.zero()
|
2016-07-08 00:28:41 +10:00
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
def transfer_scale(self, obj):
|
2016-07-09 11:52:41 +12:00
|
|
|
obj.delta_scale[0] *= obj.scale[0]
|
|
|
|
obj.delta_scale[1] *= obj.scale[1]
|
|
|
|
obj.delta_scale[2] *= obj.scale[2]
|
|
|
|
|
2016-07-04 12:52:51 +12:00
|
|
|
if self.reset_values:
|
|
|
|
obj.scale[:] = (1, 1, 1)
|
|
|
|
|
|
|
|
|
2011-08-30 10:49:58 +00:00
|
|
|
class TransformsToDeltasAnim(Operator):
|
2012-07-03 09:02:41 +00:00
|
|
|
"""Convert object animation for normal transforms to delta transforms"""
|
2011-05-24 11:15:21 +00:00
|
|
|
bl_idname = "object.anim_transforms_to_deltas"
|
|
|
|
bl_label = "Animated Transforms to Deltas"
|
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2011-08-30 10:49:58 +00:00
|
|
|
|
2011-05-24 11:15:21 +00:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
obs = context.selected_editable_objects
|
|
|
|
return (obs is not None)
|
2011-08-30 10:49:58 +00:00
|
|
|
|
2011-05-24 11:15:21 +00:00
|
|
|
def execute(self, context):
|
2012-11-20 01:26:42 +00:00
|
|
|
# map from standard transform paths to "new" transform paths
|
|
|
|
STANDARD_TO_DELTA_PATHS = {
|
2018-06-26 19:41:37 +02:00
|
|
|
"location": "delta_location",
|
|
|
|
"rotation_euler": "delta_rotation_euler",
|
|
|
|
"rotation_quaternion": "delta_rotation_quaternion",
|
|
|
|
# "rotation_axis_angle" : "delta_rotation_axis_angle",
|
|
|
|
"scale": "delta_scale"
|
2012-11-20 01:26:42 +00:00
|
|
|
}
|
|
|
|
DELTA_PATHS = STANDARD_TO_DELTA_PATHS.values()
|
2013-01-15 23:15:32 +00:00
|
|
|
|
2012-11-20 01:26:42 +00:00
|
|
|
# try to apply on each selected object
|
2011-05-24 11:15:21 +00:00
|
|
|
for obj in context.selected_editable_objects:
|
|
|
|
adt = obj.animation_data
|
|
|
|
if (adt is None) or (adt.action is None):
|
2011-08-30 10:49:58 +00:00
|
|
|
self.report({'WARNING'},
|
|
|
|
"No animation data to convert on object: %r" %
|
|
|
|
obj.name)
|
2011-05-24 11:15:21 +00:00
|
|
|
continue
|
2013-01-15 23:15:32 +00:00
|
|
|
|
2012-11-20 01:26:42 +00:00
|
|
|
# first pass over F-Curves: ensure that we don't have conflicting
|
|
|
|
# transforms already (e.g. if this was applied already) [#29110]
|
|
|
|
existingFCurves = {}
|
|
|
|
for fcu in adt.action.fcurves:
|
|
|
|
# get "delta" path - i.e. the final paths which may clash
|
|
|
|
path = fcu.data_path
|
|
|
|
if path in STANDARD_TO_DELTA_PATHS:
|
|
|
|
# to be converted - conflicts may exist...
|
|
|
|
dpath = STANDARD_TO_DELTA_PATHS[path]
|
|
|
|
elif path in DELTA_PATHS:
|
|
|
|
# already delta - check for conflicts...
|
|
|
|
dpath = path
|
|
|
|
else:
|
|
|
|
# non-transform - ignore
|
|
|
|
continue
|
2013-01-15 23:15:32 +00:00
|
|
|
|
2012-11-20 01:26:42 +00:00
|
|
|
# a delta path like this for the same index shouldn't
|
|
|
|
# exist already, otherwise we've got a conflict
|
|
|
|
if dpath in existingFCurves:
|
|
|
|
# ensure that this index hasn't occurred before
|
|
|
|
if fcu.array_index in existingFCurves[dpath]:
|
|
|
|
# conflict
|
|
|
|
self.report({'ERROR'},
|
2013-01-15 23:15:32 +00:00
|
|
|
"Object '%r' already has '%r' F-Curve(s). "
|
|
|
|
"Remove these before trying again" %
|
|
|
|
(obj.name, dpath))
|
2012-11-20 01:26:42 +00:00
|
|
|
return {'CANCELLED'}
|
|
|
|
else:
|
|
|
|
# no conflict here
|
|
|
|
existingFCurves[dpath] += [fcu.array_index]
|
|
|
|
else:
|
|
|
|
# no conflict yet
|
|
|
|
existingFCurves[dpath] = [fcu.array_index]
|
2013-01-15 23:15:32 +00:00
|
|
|
|
2011-08-30 10:49:58 +00:00
|
|
|
# if F-Curve uses standard transform path
|
|
|
|
# just append "delta_" to this path
|
2011-05-24 11:15:21 +00:00
|
|
|
for fcu in adt.action.fcurves:
|
|
|
|
if fcu.data_path == "location":
|
|
|
|
fcu.data_path = "delta_location"
|
|
|
|
obj.location.zero()
|
|
|
|
elif fcu.data_path == "rotation_euler":
|
|
|
|
fcu.data_path = "delta_rotation_euler"
|
|
|
|
obj.rotation_euler.zero()
|
|
|
|
elif fcu.data_path == "rotation_quaternion":
|
|
|
|
fcu.data_path = "delta_rotation_quaternion"
|
|
|
|
obj.rotation_quaternion.identity()
|
2011-08-30 10:49:58 +00:00
|
|
|
# XXX: currently not implemented
|
2018-06-26 19:41:37 +02:00
|
|
|
# ~ elif fcu.data_path == "rotation_axis_angle":
|
|
|
|
# ~ fcu.data_path = "delta_rotation_axis_angle"
|
2011-05-24 11:15:21 +00:00
|
|
|
elif fcu.data_path == "scale":
|
|
|
|
fcu.data_path = "delta_scale"
|
2011-08-30 10:49:58 +00:00
|
|
|
obj.scale = 1.0, 1.0, 1.0
|
|
|
|
|
2011-05-24 11:15:21 +00:00
|
|
|
# hack: force animsys flush by changing frame, so that deltas get run
|
|
|
|
context.scene.frame_set(context.scene.frame_current)
|
2011-08-30 10:49:58 +00:00
|
|
|
|
2011-05-24 11:15:21 +00:00
|
|
|
return {'FINISHED'}
|
2012-04-13 12:14:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
class DupliOffsetFromCursor(Operator):
|
2018-06-01 16:10:48 +02:00
|
|
|
"""Set offset used for collection instances based on cursor position"""
|
2012-04-13 12:14:20 +00:00
|
|
|
bl_idname = "object.dupli_offset_from_cursor"
|
|
|
|
bl_label = "Set Offset From Cursor"
|
2017-05-28 17:35:24 +02:00
|
|
|
bl_options = {'INTERNAL', 'UNDO'}
|
2012-04-13 12:14:20 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2013-01-15 23:15:32 +00:00
|
|
|
return (context.active_object is not None)
|
2012-04-13 12:14:20 +00:00
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
scene = context.scene
|
2018-06-01 16:10:48 +02:00
|
|
|
collection = context.collection
|
2012-04-13 12:14:20 +00:00
|
|
|
|
2018-06-01 16:10:48 +02:00
|
|
|
collection.dupli_offset = scene.cursor_location
|
2012-04-13 12:14:20 +00:00
|
|
|
|
|
|
|
return {'FINISHED'}
|
2013-12-17 14:42:47 -08:00
|
|
|
|
2018-09-30 17:55:39 +10:00
|
|
|
|
2018-09-27 16:21:51 +02:00
|
|
|
class LoadImageAsEmpty(Operator):
|
|
|
|
"""Select an image file and create a new image empty with it"""
|
|
|
|
bl_idname = "object.load_image_as_empty"
|
|
|
|
bl_label = "Load Image as Empty"
|
2018-10-09 14:36:15 +02:00
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
2018-09-27 16:21:51 +02:00
|
|
|
|
|
|
|
filepath: StringProperty(
|
|
|
|
subtype='FILE_PATH'
|
|
|
|
)
|
|
|
|
|
|
|
|
filter_image: BoolProperty(default=True, options={'HIDDEN', 'SKIP_SAVE'})
|
|
|
|
filter_folder: BoolProperty(default=True, options={'HIDDEN', 'SKIP_SAVE'})
|
|
|
|
|
2018-10-10 08:55:23 +11:00
|
|
|
view_align: BoolProperty(
|
2018-10-09 14:36:15 +02:00
|
|
|
name="Align to view",
|
|
|
|
default=True
|
|
|
|
)
|
|
|
|
|
2018-09-27 16:21:51 +02:00
|
|
|
def invoke(self, context, event):
|
|
|
|
context.window_manager.fileselect_add(self)
|
|
|
|
return {'RUNNING_MODAL'}
|
|
|
|
|
|
|
|
def execute(self, context):
|
2018-09-30 17:55:39 +10:00
|
|
|
scene = context.scene
|
|
|
|
space = context.space_data
|
|
|
|
cursor = (space if space and space.type == 'VIEW_3D' else scene).cursor_location
|
2018-09-27 16:21:51 +02:00
|
|
|
try:
|
|
|
|
image = bpy.data.images.load(self.filepath, check_existing=True)
|
2018-09-30 17:55:39 +10:00
|
|
|
except RuntimeError as ex:
|
|
|
|
self.report({"ERROR"}, str(ex))
|
2018-09-27 16:21:51 +02:00
|
|
|
return {"CANCELLED"}
|
|
|
|
|
2018-10-10 08:55:23 +11:00
|
|
|
bpy.ops.object.empty_add(
|
|
|
|
'INVOKE_REGION_WIN',
|
|
|
|
type='IMAGE',
|
|
|
|
location=cursor,
|
|
|
|
view_align=self.view_align,
|
|
|
|
)
|
|
|
|
obj = context.active_object
|
|
|
|
obj.data = image
|
|
|
|
obj.empty_display_size = 5.0
|
2018-09-27 16:21:51 +02:00
|
|
|
return {'FINISHED'}
|
|
|
|
|
2018-10-10 08:55:23 +11:00
|
|
|
|
2017-03-18 20:03:24 +11:00
|
|
|
classes = (
|
|
|
|
ClearAllRestrictRender,
|
|
|
|
DupliOffsetFromCursor,
|
|
|
|
IsolateTypeRender,
|
|
|
|
JoinUVs,
|
2018-09-27 16:21:51 +02:00
|
|
|
LoadImageAsEmpty,
|
2017-03-18 20:03:24 +11:00
|
|
|
MakeDupliFace,
|
|
|
|
SelectCamera,
|
|
|
|
SelectHierarchy,
|
|
|
|
SelectPattern,
|
|
|
|
ShapeTransfer,
|
|
|
|
SubdivisionSet,
|
|
|
|
TransformsToDeltas,
|
|
|
|
TransformsToDeltasAnim,
|
2017-03-25 11:07:05 +11:00
|
|
|
)
|