Newer
Older
Constantin Rahn
committed
tabWrite('}\n')
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('pigment {uv_mapping image_map {%s \"%s\" %s}%s}\n' % (imageFormat(texturesDif),texturesDif,(imgGamma + imgMap(t_dif)),mappingDif))
Constantin Rahn
committed
tabWrite('finish {%s}\n' % (safety(material_finish, Level=3)))# Level 3 is full specular
Constantin Rahn
committed
tabWrite('finish {%s}\n' % (safety(material_finish, Level=2)))# Level 2 is translated specular
#imageMap = ('{image_map {%s \"%s\" %s }' % (imageFormat(textures),textures,imgMap(t_dif)))
#file.write('\n\t\t\tuv_mapping pigment %s} %s finish {%s}' % (imageMap,mapping,safety(material_finish)))
#file.write('\n\t\t\tpigment {uv_mapping image_map {%s \"%s\" %s}%s} finish {%s}' % (imageFormat(texturesDif),texturesDif,imgMap(t_dif),mappingDif,safety(material_finish)))
mappingNor = (' translate <%.4g-0.75,%.4g-0.75,%.4g-0.75> scale <%.4g,%.4g,%.4g>\n' % (t_nor.offset.x / 10 ,t_nor.offset.y / 10 ,t_nor.offset.z / 10, t_nor.scale.x / 2.25, t_nor.scale.y / 2.25, t_nor.scale.z / 2.25))
#imageMapNor = ('{bump_map {%s \"%s\" %s mapping}' % (imageFormat(texturesNorm),texturesNorm,imgMap(t_nor)))
Constantin Rahn
committed
tabWrite('normal {uv_mapping bump_map {%s \"%s\" %s bump_size %.4g }%s}\n' % (imageFormat(texturesNorm),texturesNorm,imgMap(t_nor),(t_nor.normal_factor * 10),mappingNor))
Constantin Rahn
committed
tabWrite(']\n')
Constantin Rahn
committed
tabWrite('}\n')
#End of slope/ior texture_map
if material.diffuse_shader == 'MINNAERT' or material.diffuse_shader == 'FRESNEL':
Constantin Rahn
committed
tabWrite(']\n')
tabWrite('}\n')
tabWrite('}\n') #THEN IT CAN CLOSE IT --MR
############################################################################################################
Constantin Rahn
committed
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('face_indices {\n')
tabWrite('%d' % (len(me.faces) + quadCount)) # faces count
for fi, f in enumerate(me.faces):
fv = faces_verts[fi]
material_index = f.material_index
if len(fv) == 4:
indicies = (0, 1, 2), (0, 2, 3)
else:
indicies = ((0, 1, 2),)
if vcol_layer:
col = vcol_layer[fi]
if len(fv) == 4:
cols = col.color1, col.color2, col.color3, col.color4
else:
cols = col.color1, col.color2, col.color3
if not me_materials or me_materials[material_index] is None: # No materials
Constantin Rahn
committed
file.write(',\n')
tabWrite('<%d,%d,%d>' % (fv[i1], fv[i2], fv[i3])) # vert count
else:
material = me_materials[material_index]
for i1, i2, i3 in indicies:
if me.vertex_colors and material.use_vertex_color_paint:
# Colour per vertex - vertex colour
col1 = cols[i1]
col2 = cols[i2]
col3 = cols[i3]
ci1 = vertCols[col1[0], col1[1], col1[2], material_index][0]
ci2 = vertCols[col2[0], col2[1], col2[2], material_index][0]
ci3 = vertCols[col3[0], col3[1], col3[2], material_index][0]
else:
# Colour per material - flat material colour
diffuse_color = material.diffuse_color
ci1 = ci2 = ci3 = vertCols[diffuse_color[0], diffuse_color[1], diffuse_color[2], f.material_index][0]
Constantin Rahn
committed
file.write(',\n')
tabWrite('<%d,%d,%d>, %d,%d,%d' % (fv[i1], fv[i2], fv[i3], ci1, ci2, ci3)) # vert count
Constantin Rahn
committed
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('normal_indices {\n')
tabWrite('%d' % (len(me.faces) + quadCount)) # faces count
for fi, fv in enumerate(faces_verts):
if len(fv) == 4:
indicies = (0, 1, 2), (0, 2, 3)
else:
indicies = ((0, 1, 2),)
for i1, i2, i3 in indicies:
Constantin Rahn
committed
file.write(',\n')
tabWrite('<%d,%d,%d>' %\
(uniqueNormals[verts_normals[fv[i1]]][0],\
uniqueNormals[verts_normals[fv[i2]]][0],\
uniqueNormals[verts_normals[fv[i3]]][0])) # vert count
else:
idx = uniqueNormals[faces_normals[fi]][0]
Constantin Rahn
committed
file.write(',\n')
tabWrite('<%d,%d,%d>' % (idx, idx, idx)) # vert count
Constantin Rahn
committed
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('uv_indices {\n')
tabWrite('%d' % (len(me.faces) + quadCount)) # faces count
for fi, fv in enumerate(faces_verts):
if len(fv) == 4:
indicies = (0, 1, 2), (0, 2, 3)
else:
indicies = ((0, 1, 2),)
uv = uv_layer[fi]
if len(faces_verts[fi]) == 4:
uvs = tuple(uv.uv1), tuple(uv.uv2), tuple(uv.uv3), tuple(uv.uv4)
else:
uvs = tuple(uv.uv1), tuple(uv.uv2), tuple(uv.uv3)
for i1, i2, i3 in indicies:
Constantin Rahn
committed
file.write(',\n')
tabWrite('<%d,%d,%d>' %\
Constantin Rahn
committed
file.write('}\n')
try:
material = me.materials[0] # dodgy
writeObjectMaterial(material)
except IndexError:
print(me)
Constantin Rahn
committed
tabWrite('radiosity { \n')
tabWrite('importance %3g \n' % importance)
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('}\n') # End of mesh block
tabWrite('%s\n' % name) # Use named declaration to allow reference e.g. for baking. MR
bpy.data.meshes.remove(me)
def exportWorld(world):
render = scene.render
matrix = global_matrix * camera.matrix_world
#############Maurice####################################
#These lines added to get sky gradient (visible with PNG output)
if world:
#For simple flat background:
if not world.use_sky_blend:
#Non fully transparent background could premultiply alpha and avoid anti-aliasing display issue:
if render.alpha_mode == 'PREMUL' or render.alpha_mode == 'PREMUL' :
Constantin Rahn
committed
tabWrite('background {rgbt<%.3g, %.3g, %.3g, 0.75>}\n' % (tuple(world.horizon_color)))
#Currently using no alpha with Sky option:
elif render.alpha_mode == 'SKY':
Constantin Rahn
committed
tabWrite('background {rgbt<%.3g, %.3g, %.3g, 0>}\n' % (tuple(world.horizon_color)))
Constantin Rahn
committed
tabWrite('background {rgbt<%.3g, %.3g, %.3g, 1>}\n' % (tuple(world.horizon_color)))
#For Background image textures
for t in world.texture_slots: #risk to write several sky_spheres but maybe ok.
if t and t.texture.type == 'IMAGE': #and t.use: #No enable checkbox for world textures yet (report it?)
image_filename = path_image(t.texture.image.filepath)
if t.texture.image.filepath != image_filename: t.texture.image.filepath = image_filename
if image_filename != '' and t.use_map_blend:
texturesBlend = image_filename
#colvalue = t.default_value
t_blend = t
#commented below was an idea to make the Background image oriented as camera taken here: http://news.povray.org/povray.newusers/thread/%3Cweb.4a5cddf4e9c9822ba2f93e20@news.povray.org%3E/
#mappingBlend = (' translate <%.4g,%.4g,%.4g> rotate z*degrees(atan((camLocation - camLookAt).x/(camLocation - camLookAt).y)) rotate x*degrees(atan((camLocation - camLookAt).y/(camLocation - camLookAt).z)) rotate y*degrees(atan((camLocation - camLookAt).z/(camLocation - camLookAt).x)) scale <%.4g,%.4g,%.4g>b' % (t_blend.offset.x / 10 ,t_blend.offset.y / 10 ,t_blend.offset.z / 10, t_blend.scale.x ,t_blend.scale.y ,t_blend.scale.z))#replace 4/3 by the ratio of each image found by some custom or existing function
#using camera rotation valuesdirectly from blender seems much easier
Maurice Raybaud
committed
if t_blend.texture_coords=='ANGMAP':
Maurice Raybaud
committed
else:
mappingBlend = (' translate <%.4g-0.5,%.4g-0.5,%.4g-0.5> rotate<0,0,0> scale <%.4g,%.4g,%.4g>' % (t_blend.offset.x / 10 ,t_blend.offset.y / 10 ,t_blend.offset.z / 10, t_blend.scale.x*0.85 , t_blend.scale.y*0.85 , t_blend.scale.z*0.85 ))
#The initial position and rotation of the pov camera is probably creating the rotation offset should look into it someday but at least background won't rotate with the camera now.
#Putting the map on a plane would not introduce the skysphere distortion and allow for better image scale matching but also some waay to chose depth and size of the plane relative to camera.
Constantin Rahn
committed
tabWrite('sky_sphere {\n')
tabWrite('pigment {\n')
tabWrite('image_map{%s \"%s\" %s}\n' % (imageFormat(texturesBlend),texturesBlend,imgMapBG(t_blend)))
tabWrite('}\n')
tabWrite('%s\n' % (mappingBlend))
tabWrite('}\n')
#tabWrite('scale 2\n')
#tabWrite('translate -1\n')
Constantin Rahn
committed
tabWrite('sky_sphere {\n')
tabWrite('pigment {\n')
tabWrite('gradient y\n')#maybe Should follow the advice of POV doc about replacing gradient for skysphere..5.5
tabWrite('color_map {\n')
if render.alpha_mode == 'STRAIGHT':
Constantin Rahn
committed
tabWrite('[0.0 rgbt<%.3g, %.3g, %.3g, 1>]\n' % (tuple(world.horizon_color)))
tabWrite('[1.0 rgbt<%.3g, %.3g, %.3g, 1>]\n' % (tuple(world.zenith_color)))
elif render.alpha_mode == 'PREMUL':
Constantin Rahn
committed
tabWrite('[0.0 rgbt<%.3g, %.3g, %.3g, 0.99>]\n' % (tuple(world.horizon_color)))
tabWrite('[1.0 rgbt<%.3g, %.3g, %.3g, 0.99>]\n' % (tuple(world.zenith_color))) #aa premult not solved with transmit 1
Constantin Rahn
committed
tabWrite('[0.0 rgbt<%.3g, %.3g, %.3g, 0>]\n' % (tuple(world.horizon_color)))
tabWrite('[1.0 rgbt<%.3g, %.3g, %.3g, 0>]\n' % (tuple(world.zenith_color)))
tabWrite('}\n')
tabWrite('}\n')
tabWrite('}\n')
#sky_sphere alpha (transmit) is not translating into image alpha the same way as 'background'
Maurice Raybaud
committed
#if world.light_settings.use_indirect_light:
# scene.pov_radio_enable=1
Maurice Raybaud
committed
#Maybe change the above to a funtion copyInternalRenderer settings when user pushes a button, then:
#scene.pov_radio_enable = world.light_settings.use_indirect_light
#and other such translations but maybe this would not be allowed either?
###############################################################
Constantin Rahn
committed
tabWrite('fog {\n')
tabWrite('distance %.6f\n' % mist.depth)
tabWrite('color rgbt<%.3g, %.3g, %.3g, %.3g>\n' % (tuple(world.horizon_color) + (1 - mist.intensity,)))
#tabWrite('fog_offset %.6f\n' % mist.start)
#tabWrite('fog_alt 5\n')
#tabWrite('turbulence 0.2\n')
#tabWrite('turb_depth 0.3\n')
tabWrite('fog_type 1\n')
tabWrite('}\n')
if scene.pov_media_enable:
Constantin Rahn
committed
tabWrite('media {\n')
tabWrite('scattering { 1, rgb %.3g}\n' % scene.pov_media_color)
tabWrite('samples %.d\n' % scene.pov_media_samples)
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('global_settings {\n')
tabWrite('assumed_gamma 1.0\n')
tabWrite('max_trace_level %d\n' % scene.pov_max_trace_level)
Constantin Rahn
committed
tabWrite('radiosity {\n')
tabWrite('adc_bailout %.4g\n' % scene.pov_radio_adc_bailout)
tabWrite('always_sample %d\n' % scene.pov_radio_always_sample)
tabWrite('brightness %.4g\n' % scene.pov_radio_brightness)
tabWrite('count %d\n' % scene.pov_radio_count)
tabWrite('error_bound %.4g\n' % scene.pov_radio_error_bound)
tabWrite('gray_threshold %.4g\n' % scene.pov_radio_gray_threshold)
tabWrite('low_error_factor %.4g\n' % scene.pov_radio_low_error_factor)
tabWrite('media %d\n' % scene.pov_radio_media)
tabWrite('minimum_reuse %.4g\n' % scene.pov_radio_minimum_reuse)
tabWrite('nearest_count %d\n' % scene.pov_radio_nearest_count)
tabWrite('normal %d\n' % scene.pov_radio_normal)
tabWrite('pretrace_start %.3g\n' % scene.pov_radio_pretrace_start)
tabWrite('pretrace_end %.3g\n' % scene.pov_radio_pretrace_end)
tabWrite('recursion_limit %d\n' % scene.pov_radio_recursion_limit)
tabWrite('}\n')
once=1
for material in bpy.data.materials:
if material.subsurface_scattering.use and once:
Constantin Rahn
committed
tabWrite('mm_per_unit %.6f\n' % (material.subsurface_scattering.scale * (-100) + 15))#In pov, the scale has reversed influence compared to blender. these number should correct that
once=0 #In povray, the scale factor for all subsurface shaders needs to be the same
Constantin Rahn
committed
tabWrite('ambient_light rgb<%.3g, %.3g, %.3g>\n' % tuple(world.ambient_color))
if material.pov_photons_refraction or material.pov_photons_reflection:
Constantin Rahn
committed
tabWrite('photons {\n')
tabWrite('spacing 0.003\n')
tabWrite('max_trace_level 5\n')
tabWrite('adc_bailout 0.1\n')
tabWrite('gather 30, 150\n')
tabWrite('}\n')
Constantin Rahn
committed
tabWrite('}\n')
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
sel = scene.objects
comments = scene.pov_comments_enable
if comments: file.write('//---------------------------------------------\n//--Exported with Povray exporter for Blender--\n//---------------------------------------------\n')
if comments: file.write('\n//--Global settings and background--\n\n')
exportGlobalSettings(scene)
if comments: file.write('\n')
exportWorld(scene.world)
if comments: file.write('\n//--Cameras--\n\n')
exportCamera()
if comments: file.write('\n//--Lamps--\n\n')
exportLamps([l for l in sel if l.type == 'LAMP'])
if comments: file.write('\n//--Material Definitions--\n\n')
# Convert all materials to strings we can access directly per vertex.
writeMaterial(None) # default material
for material in bpy.data.materials:
writeMaterial(material)
Constantin Rahn
committed
if comments: file.write('//--Meta objects--\n\n') # <- How can this be written only if the scene contains META?
if comments: file.write('\n') # <- How can this be written only if the scene contains META?
if comments: file.write('//--Mesh objecs--\n\n')
#What follow used to happen here:
#exportCamera()
#exportWorld(scene.world)
#exportGlobalSettings(scene)
Constantin Rahn
committed
# MR:..and the order was important for an attempt to implement pov 3.7 baking (mesh camera) comment for the record
# CR: Baking should be a special case than. If "baking", than we could change the order.
def write_pov_ini(filename_ini, filename_pov, filename_image):
scene = bpy.data.scenes[0]
render = scene.render
Constantin Rahn
committed
x = int(render.resolution_x * render.resolution_percentage * 0.01)
y = int(render.resolution_y * render.resolution_percentage * 0.01)
file = open(filename_ini.name, 'w')
file.write("Input_File_Name='%s'\n" % filename_pov.name)
file.write("Output_File_Name='%s'\n" % filename_image.name)
file.write('Width=%d\n' % x)
file.write('Height=%d\n' % y)
# Needed for border render.
'''
file.write('Start_Column=%d\n' % part.x)
file.write('End_Column=%d\n' % (part.x+part.w))
file.write('Start_Row=%d\n' % (part.y))
file.write('End_Row=%d\n' % (part.y+part.h))
'''
file.write('Bounding_Method=2\n')#The new automatic BSP is faster in most scenes
file.write('Display=1\n')#Activated (turn this back off when better live exchange is done between the two programs (see next comment)
file.write('Output_File_Type=N\n') # PNG, with POV 3.7, can show background color with alpha. In the long run using the Povray interactive preview like bishop 3D could solve the preview for all formats.
#file.write('Output_File_Type=T\n') # TGA, best progressive loading
aa_mapping = {'5': 2, '8': 3, '11': 4, '16': 5} # method 2 (recursive) with higher max subdiv forced because no mipmapping in povray needs higher sampling.
file.write('Sampling_Method=2\n')
file.write('Antialias_Depth=%d\n' % aa_mapping[render.antialiasing_samples])
file.write('Antialias_Threshold=0.1\n')#rather high settings but necessary.
file.write('Jitter=off\n')#prevent animation flicker
Maurice Raybaud
committed
file.write('Version=3.7')
Doug Hammond
committed
DELAY = 0.05
Constantin Rahn
committed
# mktemp is Deprecated since version 2.3, replaced with NamedTemporaryFile() #CR
self._temp_file_in = tempfile.NamedTemporaryFile(suffix='.pov', delete=False)
self._temp_file_out = tempfile.NamedTemporaryFile(suffix='.png', delete=False)#PNG with POV 3.7, can show the background color with alpha. In the long run using the Povray interactive preview like bishop 3D could solve the preview for all formats.
#self._temp_file_out = tempfile.NamedTemporaryFile(suffix='.tga', delete=False)
self._temp_file_ini = tempfile.NamedTemporaryFile(suffix='.ini', delete=False)
self._temp_file_out = '/test.png'#PNG with POV 3.7, can show the background color with alpha. In the long run using the Povray interactive preview like bishop 3D could solve the preview for all formats.
#self._temp_file_out = '/test.tga'
self._temp_file_ini = '/test.ini'
'''
def info_callback(txt):
write_pov(self._temp_file_in, scene, info_callback)
Constantin Rahn
committed
def _render(self, scene):
os.remove(self._temp_file_out.name) # so as not to load the old file
pass
write_pov_ini(self._temp_file_ini, self._temp_file_in, self._temp_file_out)
Doug Hammond
committed
extra_args = []
Constantin Rahn
committed
if scene.pov_command_line_switches != "":
for newArg in scene.pov_command_line_switches.split(' '):
extra_args.append(newArg)
regKey = winreg.OpenKey(winreg.HKEY_CURRENT_USER, 'Software\\POV-Ray\\v3.7\\Windows')
if bitness == 64:
pov_binary = winreg.QueryValueEx(regKey, 'Home')[0] + '\\bin\\pvengine64'
else:
pov_binary = winreg.QueryValueEx(regKey, 'Home')[0] + '\\bin\\pvengine'
Doug Hammond
committed
else:
# DH - added -d option to prevent render window popup which leads to segfault on linux
Constantin Rahn
committed
# print('Extra Args: ' + str(extra_args))
if 1:
# TODO, when povray isnt found this gives a cryptic error, would be nice to be able to detect if it exists
try:
self._process = subprocess.Popen([pov_binary, self._temp_file_ini.name] + extra_args) # stdout=subprocess.PIPE, stderr=subprocess.PIPE
print("POVRAY 3.7: could not execute '%s', possibly povray isn't installed" % pov_binary)
return False
else:
# This works too but means we have to wait until its done
os.system('%s %s' % (pov_binary, self._temp_file_ini.name))
return True
def _cleanup(self):
for f in (self._temp_file_in, self._temp_file_ini, self._temp_file_out):
#print('Name: %s' % f.name)
#print('File closed %s' % f.closed)
f.close() # Why do I have to close them again? Without closeing the pov and ini files are not deletable. PNG is not closable!
except OSError: #was that the proper error type?
#print('Couldn't remove/unlink TEMP file %s' % f.name)
self.update_stats('', 'POVRAY 3.7: Exporting data from Blender')
self.update_stats('', 'POVRAY 3.7: Parsing File')
Constantin Rahn
committed
if not self._render(scene):
##WIP output format
## if r.file_format == 'OPENEXR':
## fformat = 'EXR'
## render.color_mode = 'RGBA'
## else:
## fformat = 'TGA'
## r.file_format = 'TARGA'
## r.color_mode = 'RGBA'
# compute resolution
x = int(r.resolution_x * r.resolution_percentage * 0.01)
y = int(r.resolution_y * r.resolution_percentage * 0.01)
# Wait for the file to be created
while not os.path.exists(self._temp_file_out.name):
# print('***POV WAITING FOR FILE***')
Doug Hammond
committed
poll_result = self._process.poll()
print('***POV PROCESS FAILED : %s ***' % poll_result)
self.update_stats('', 'POVRAY 3.7: Failed')
if os.path.exists(self._temp_file_out.name):
# print('***POV FILE OK***')
self.update_stats('', 'POVRAY 3.7: Rendering')
result = self.begin_result(0, 0, x, y)
lay = result.layers[0]
# possible the image wont load early on.
try:
pass
self.end_result(result)
# Update while povray renders
while True:
update_image()
break
# user exit
if self.test_break():
try:
pass
break
# Would be nice to redirect the output
# stdout_value, stderr_value = self._process.communicate() # locks
# check if the file updated
new_size = os.path.getsize(self._temp_file_out.name)
if new_size != prev_size:
update_image()
prev_size = new_size
time.sleep(self.DELAY)
Doug Hammond
committed
else:
Doug Hammond
committed
print('***POV FINISHED***')
#time.sleep(self.DELAY)