mirror of https://github.com/H-uru/korman.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
116 lines
3.6 KiB
116 lines
3.6 KiB
# This file is part of Korman. |
|
# |
|
# Korman 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 3 of the License, or |
|
# (at your option) any later version. |
|
# |
|
# Korman 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 Korman. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
import bmesh |
|
import bpy |
|
from contextlib import contextmanager |
|
import math |
|
|
|
@contextmanager |
|
def bmesh_from_object(bl): |
|
"""Converts a Blender Object to a BMesh with modifiers applied.""" |
|
mesh = bmesh.new() |
|
try: |
|
# Empirical evidence indicates that this applies Blender Modifiers |
|
mesh.from_object(bl, bpy.context.scene) |
|
yield mesh |
|
finally: |
|
mesh.free() |
|
|
|
class GoodNeighbor: |
|
"""Leave Things the Way You Found Them! (TM)""" |
|
|
|
def __enter__(self): |
|
self._tracking = {} |
|
return self |
|
|
|
def track(self, cls, attr, value): |
|
if (cls, attr) not in self._tracking: |
|
self._tracking[(cls, attr)] = getattr(cls, attr) |
|
setattr(cls, attr, value) |
|
|
|
def __exit__(self, type, value, traceback): |
|
for (cls, attr), value in self._tracking.items(): |
|
setattr(cls, attr, value) |
|
|
|
|
|
class TemporaryObject: |
|
def __init__(self, obj, remove_func): |
|
self._obj = obj |
|
self._remove_func = remove_func |
|
|
|
def __enter__(self): |
|
return self._obj |
|
|
|
def __exit__(self, type, value, traceback): |
|
self._remove_func(self._obj) |
|
|
|
def __getattr__(self, attr): |
|
return getattr(self._obj, attr) |
|
|
|
|
|
class UiHelper: |
|
"""This fun little helper makes sure that we don't wreck the UI""" |
|
def __init__(self, context): |
|
self.active_object = context.active_object |
|
self.selected_objects = context.selected_objects |
|
|
|
def __enter__(self): |
|
scene = bpy.context.scene |
|
self.layers = tuple(scene.layers) |
|
self.frame_num = scene.frame_current |
|
scene.frame_set(scene.frame_start) |
|
scene.update() |
|
|
|
# Some operators require there be an active_object even though they |
|
# don't actually use it... |
|
if scene.objects.active is None: |
|
scene.objects.active = scene.objects[0] |
|
return self |
|
|
|
def __exit__(self, type, value, traceback): |
|
for i in bpy.data.objects: |
|
i.select = (i in self.selected_objects) |
|
|
|
scene = bpy.context.scene |
|
scene.objects.active = self.active_object |
|
scene.layers = self.layers |
|
scene.frame_set(self.frame_num) |
|
scene.update() |
|
|
|
|
|
def ensure_power_of_two(value): |
|
return pow(2, math.floor(math.log(value, 2))) |
|
|
|
def fetch_fcurves(id_data, data_fcurves=True): |
|
"""Given a Blender ID, yields its FCurves""" |
|
def _fetch(source): |
|
if source is not None and source.action is not None: |
|
for i in source.action.fcurves: |
|
yield i |
|
|
|
# This seems rather unpythonic IMO |
|
for i in _fetch(id_data.animation_data): |
|
yield i |
|
if data_fcurves: |
|
for i in _fetch(id_data.data.animation_data): |
|
yield i |
|
|
|
def find_modifier(bo, modid): |
|
"""Given a Blender Object, finds a given modifier and returns it or None""" |
|
if bo is not None: |
|
# if they give us the wrong modid, it is a bug and an AttributeError |
|
return getattr(bo.plasma_modifiers, modid) |
|
return None
|
|
|