forked from njanakiev/blender-scripting
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path__init__.py
193 lines (151 loc) · 5.46 KB
/
__init__.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
import bpy
import bmesh
from math import sin, cos, pi
TAU = 2*pi
import colorsys
import os
def remove_object(obj):
if obj.type == 'MESH':
if obj.data.name in bpy.data.meshes:
bpy.data.meshes.remove(obj.data)
if obj.name in bpy.context.scene.objects:
bpy.context.scene.objects.unlink(obj)
bpy.data.objects.remove(obj)
else:
raise NotImplementedError('Other types not implemented yet besides \'MESH\'')
def track_to_constraint(obj, target):
constraint = obj.constraints.new('TRACK_TO')
constraint.target = target
constraint.track_axis = 'TRACK_NEGATIVE_Z'
#constraint.track_axis = 'TRACK_Z'
constraint.up_axis = 'UP_Y'
#constraint.owner_space = 'LOCAL'
#constraint.target_space = 'LOCAL'
return constraint
def create_target(origin=(0,0,0)):
target = bpy.data.objects.new('Target', None)
bpy.context.collection.objects.link(target)
target.location = origin
return target
def create_camera(origin, target=None, lens=35, clip_start=0.1, clip_end=200, type='PERSP', ortho_scale=6):
# Create object and camera
camera = bpy.data.cameras.new("Camera")
camera.lens = lens
camera.clip_start = clip_start
camera.clip_end = clip_end
camera.type = type # 'PERSP', 'ORTHO', 'PANO'
if type == 'ORTHO':
camera.ortho_scale = ortho_scale
# Link object to scene
obj = bpy.data.objects.new("CameraObj", camera)
obj.location = origin
bpy.context.collection.objects.link(obj)
bpy.context.scene.camera = obj # Make this the current camera
if target:
track_to_constraint(obj, target)
return obj
def create_light(origin, type='POINT', energy=1, color=(1,1,1), target=None):
# Light types: 'POINT', 'SUN', 'SPOT', 'HEMI', 'AREA'
bpy.ops.object.add(type='LIGHT', location=origin)
obj = bpy.context.object
obj.data.type = type
obj.data.energy = energy
obj.data.color = color
if target:
track_to_constraint(obj, target)
return obj
def simple_scene(target_coords, camera_coords, sun_coords, lens=35):
target = create_target(target_coords)
camera = create_camera(camera_coords, target, lens)
sun = create_light(sun_coords, 'SUN', target=target)
return target, camera, sun
def set_smooth(obj, level=None, smooth=True):
if level:
# Add subsurf modifier
modifier = obj.modifiers.new('Subsurf', 'SUBSURF')
modifier.levels = level
modifier.render_levels = level
# Smooth surface
mesh = obj.data
for p in mesh.polygons:
p.use_smooth = smooth
def rainbow_lights(r=5, n=100, freq=2, energy=0.1):
for i in range(n):
t = float(i)/float(n)
pos = (r*sin(TAU*t), r*cos(TAU*t), r*sin(freq*TAU*t))
# Create lamp
bpy.ops.object.add(type='LIGHT', location=pos)
obj = bpy.context.object
obj.data.type = 'POINT'
# Apply gamma correction for Blender
color = tuple(pow(c, 2.2) for c in colorsys.hsv_to_rgb(t, 0.6, 1))
# Set HSV color and lamp energy
obj.data.color = color
obj.data.energy = energy
def remove_all(type=None):
# Possible type:
# "MESH", "CURVE", "SURFACE", "META", "FONT", "ARMATURE",
# "LATTICE", "EMPTY", "CAMERA", "LIGHT"
if type:
bpy.ops.object.select_all(action='DESELECT')
bpy.ops.object.select_by_type(type=type)
bpy.ops.object.delete()
else:
# Remove all elements in scene
bpy.ops.object.select_all(action="SELECT")
bpy.ops.object.delete(use_global=False)
def create_material(base_color=(1, 1, 1, 1), metalic=0.0, roughness=0.5):
mat = bpy.data.materials.new('Material')
if len(base_color) == 3:
base_color = list(base_color)
base_color.append(1)
mat.use_nodes = True
node = mat.node_tree.nodes[0]
node.inputs[0].default_value = base_color
node.inputs[4].default_value = metalic
node.inputs[7].default_value = roughness
return mat
def colorRGB_256(color):
return tuple(pow(float(c)/255.0, 2.2) for c in color)
def render(
render_folder='rendering',
render_name='render',
resolution_x=800,
resolution_y=800,
resolution_percentage=100,
animation=False,
frame_end=None,
render_engine='CYCLES'
):
scene = bpy.context.scene
scene.render.resolution_x = resolution_x
scene.render.resolution_y = resolution_y
scene.render.resolution_percentage = resolution_percentage
scene.render.engine = render_engine
if frame_end:
scene.frame_end = frame_end
# Check if script is executed inside Blender
if bpy.context.space_data is None:
# Specify folder to save rendering and check if it exists
render_folder = os.path.join(os.getcwd(), render_folder)
if(not os.path.exists(render_folder)):
os.mkdir(render_folder)
if animation:
# Render animation
scene.render.filepath = os.path.join(
render_folder,
render_name)
bpy.ops.render.render(animation=True)
else:
# Render still frame
scene.render.filepath = os.path.join(
render_folder,
render_name + '.png')
bpy.ops.render.render(write_still=True)
def bmesh_to_object(bm, name='Object'):
mesh = bpy.data.meshes.new(name + 'Mesh')
bm.to_mesh(mesh)
bm.free()
obj = bpy.data.objects.new(name, mesh)
bpy.context.scene.collection.objects.link(obj)
return obj