import bpy import os from os import listdir, scandir from os.path import join, dirname, basename, exists, isfile, isdir, splitext import re, fnmatch, glob from pathlib import Path from time import strftime import subprocess, threading # viewport draw import gpu, blf from gpu_extras.batch import batch_for_shader from .utils import open_file, open_folder, get_addon_prefs, detect_OS ## based on playblaster exclude = ( ### add lines here to exclude specific attribute 'bl_rna', 'identifier','name_property','rna_type','properties', 'compare', 'to_string',#basic ) def delete_file(filepath) : try: if os.path.isfile(filepath) : print('removing', filepath) os.remove(filepath) return True except PermissionError: print(f'impossible to remove {filepath}') return False # render function def render_function(cmd, total_frame, scene) : debug = bpy.context.window_manager.pblast_debug # launch rendering if debug : print(cmd) process = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) frame_count = 0 while True : if not bpy.context.window_manager.pblast_is_rendering : # print('!!! Not rendering') if detect_OS() == 'Windows': subprocess.Popen(f"TASKKILL /F /PID {process.pid} /T") else: process.kill() break line = process.stdout.readline() if line != '' : #debug if debug : print(line) if b'Traceback' in line: print('/!\\ Traceback in line return') if b"Append frame " in line : frame_count += 1 try : # print('frame_count: ', frame_count, 'total_frame: ', total_frame) bpy.context.window_manager.pblast_completion = frame_count / total_frame * 100 except AttributeError : #debug if debug : print("AttributeError avoided") pass if b"Blender quit" in line : break else: break print('ENDED') # launch threading def threading_render(arguments) : render_thread = threading.Thread(target=render_function, args=arguments) render_thread.start() # callback for loading bar in 3D view def draw_callback_px(self, context): # get color and size of progress bar # prefs = get_addon_prefs() color_bar = [1, 1, 1] # prefs.progress_bar_color background = [0.2, 0.2, 0.2] # prefs.progress_bar_background_color bar_thickness = 10 # prefs.progress_bar_size # Progress Bar width = context.area.width # context.window_manager.pblast_completion complete = self.completion / 100# context.window_manager.pblast_completion / 100 size = int(width * complete) # rectangle background vertices_2 = ( (0, 0), (width, 0), (0, bar_thickness + 20), (width, bar_thickness + 20)) indices = ( (0, 1, 2), (2, 1, 3)) shader2 = gpu.shader.from_builtin('2D_UNIFORM_COLOR') batch2 = batch_for_shader(shader2, 'TRIS', {"pos": vertices_2}, indices=indices) shader2.bind() shader2.uniform_float("color", [*background, 1]) batch2.draw(shader2) # rectangle 1 vertices = ( (0, 0), (size, 0), (0, bar_thickness), (size, bar_thickness)) shader = gpu.shader.from_builtin('2D_UNIFORM_COLOR') batch = batch_for_shader(shader, 'TRIS', {"pos": vertices}, indices=indices) shader.bind() shader.uniform_float("color", [*color_bar, 1]) batch.draw(shader) # Text # text = f'Playblast in Progress ({complete}%) - Shift + Esc to Cancel' text = f'Playblast in Progress - Shift + Esc to Cancel' blf.color(0, *color_bar, 1) blf.size(0, 12, 72) blf.position(0, 10, bar_thickness + 5, 0) blf.draw(0, text) class BGBLAST_OT_playblast_modal_check(bpy.types.Operator): '''Modal and external render from Samy Tichadou (Tonton)''' bl_idname = "render.playblast_modal_check" bl_label = "Playblast Modal Check" bl_options = {'INTERNAL'} _timer = None blend_pb : bpy.props.StringProperty() video_pb : bpy.props.StringProperty() @classmethod def poll(cls, context): return context.window_manager.pblast_is_rendering def modal(self, context, event): self.completion = context.window_manager.pblast_completion # redraw area try: context.area.tag_redraw() except AttributeError: pass # handle cancelling if event.type in {'ESC'} and event.shift: self.cancel(context) return {'CANCELLED'} if event.type == 'TIMER' : if self.completion == 100 : self.finish(context) return {'FINISHED'} return {'PASS_THROUGH'} # def invoke(self, context, event): def execute(self, context): context.window_manager.pblast_completion = 0 wm = context.window_manager args = (self, context) wm = context.window_manager self.debug = wm.pblast_debug self.completion = wm.pblast_completion self._timer = wm.event_timer_add(0.1, window = context.window) self._handle = bpy.types.SpaceView3D.draw_handler_add(draw_callback_px, args, 'WINDOW', 'POST_PIXEL') wm.modal_handler_add(self) return {'RUNNING_MODAL'} def cancel(self, context): print('in CANCEL') bpy.types.SpaceView3D.draw_handler_remove(self._handle, 'WINDOW') wm = context.window_manager wm.event_timer_remove(self._timer) # turn off is_rendering wm.pblast_is_rendering = False wm.pblast_completion = 0 # delete temp file delete_file(self.blend_pb) self.blend_pb = "" # delete temp video delete_file(self.video_pb) self.video_pb = "" self.report({'WARNING'}, "Render Canceled") def finish(self, context): print('in FINISH') bpy.types.SpaceView3D.draw_handler_remove(self._handle, 'WINDOW') wm = context.window_manager wm.event_timer_remove(self._timer) # turn off is_rendering wm.pblast_is_rendering = False wm.pblast_completion = 0 # debug if self.debug : print("blend temp : " + self.blend_pb) print("video temp : " + self.video_pb) # delete temp file delete_file(self.blend_pb) # open video file and/or folder prefs = get_addon_prefs() if prefs.playblast_auto_play: # open_file(self.video_pb) bpy.ops.wm.path_open(filepath=self.video_pb) if prefs.playblast_auto_open_folder: # open_folder(dirname(self.video_pb)) bpy.ops.wm.path_open(filepath=dirname(self.video_pb)) wm.pblast_previous_render = self.video_pb self.report({'INFO'}, "Render Finished") ### classic sauce def render_with_restore(): class RenderFileRestorer: rd = bpy.context.scene.render im = rd.image_settings ff = rd.ffmpeg # ffmpeg (ff) need to be before image_settings(im) in list # otherwise __exit__ may try to restore settings of image mode in video mode ! # ex : "RGBA" not found in ('BW', 'RGB') (will still not stop thx to try block) zones = [rd, ff, im] val_dic = {} def __enter__(self): ## store attribute of data_path in self.zones list. for data_path in self.zones: self.val_dic[data_path] = {} for attr in dir(data_path):#iterate in attribute of given datapath if attr not in exclude and not attr.startswith('__') and not callable(getattr(data_path, attr)) and not data_path.is_property_readonly(attr): self.val_dic[data_path][attr] = getattr(data_path, attr) def __exit__(self, type, value, traceback): ## restore attribute from self.zones list for data_path, prop_dic in self.val_dic.items(): for attr, val in prop_dic.items(): try: setattr(data_path, attr, val) except Exception as e: print(f"/!\ Impossible to re-assign: {attr} = {val}") print(e) return RenderFileRestorer() def playblast(context, viewport = False, stamping = True): scn = bpy.context.scene res_factor = scn.gptoolprops.resolution_percentage rd = scn.render ff = rd.ffmpeg prefix = 'tempblast_' # delete old playblast and blend files folder = dirname(bpy.data.filepath) for f in os.listdir(folder): if f.startswith(prefix): delete_file(join(folder, f)) pblast_folder = join(folder, 'playblast') if exists(pblast_folder): for f in os.listdir(pblast_folder): if f.startswith(prefix): delete_file(join(pblast_folder, f)) tempblend = str( Path(folder) / f'{prefix}{basename(bpy.data.filepath)}' ) with render_with_restore(): ### can add propeties for personalisation as toolsetting props rd.resolution_percentage = res_factor while ( rd.resolution_x * res_factor / 100 ) % 2 != 0:# rd.resolution_percentage rd.resolution_x = rd.resolution_x + 1 while ( rd.resolution_y * res_factor / 100 ) % 2 != 0:# rd.resolution_percentage rd.resolution_y = rd.resolution_y + 1 backup_img_settings = rd.image_settings.file_format rd.image_settings.file_format = 'FFMPEG' ff.format = 'MPEG4' ff.codec = 'H264' ff.constant_rate_factor = 'HIGH'# MEDIUM ff.ffmpeg_preset = 'REALTIME' ff.gopsize = 10 ff.audio_codec = 'AAC' ff.audio_bitrate = 128 rd.use_sequencer = False # rd.use_compositing # rd.filepath = join(dirname(bpy.data.filepath), basename(bpy.data.filepath)) # rd.frame_path(frame=0, preview=0, view="_sauce")## give absolute render filepath with some suffix # rd.is_movie_format# check if its movie mode ## set filepath # mode incermental or just use fulldate (cannot create conflict and filter OK but long name) blend = Path(bpy.data.filepath) date_format = "%Y-%m-%d_%H-%M-%S" fp = join(blend.parent, "playblast", f'{prefix}{blend.stem}_{strftime(date_format)}.mp4') #may need a properties for choosing location : bpy.types.Scene.qrd_savepath = bpy.props.StringProperty(subtype='DIR_PATH', description="Export location, if not specify, create a 'quick_render' directory aside blend location")#(change defaut name in user_prefernece) rd.filepath = fp rd.use_stamp = stamping# toolsetting.use_stamp# True for playblast #stamp options rd.stamp_font_size = rd.stamp_font_size * res_factor / 100# rd.resolution_percentage # get total number of frames total_frame = context.scene.frame_end - context.scene.frame_start + 1 # bpy.ops.render.render_wrap(use_view=viewport) bpy.ops.wm.save_as_mainfile(filepath = tempblend, copy=True) cmd = f'"{bpy.app.binary_path}" -b "{tempblend}" -a' # cmd = '"' + bpy.app.binary_path + '"' + " -b " + '"' + new_blend_filepath + '"' + " -E " + render_engine + " -a" threading_render([cmd, total_frame, context.scene]) rd.image_settings.file_format = backup_img_settings#seems it tries to set some properties before the man imgs settings # lauch BG render modal (ovrerride ?) context.window_manager.pblast_completion = 0 context.window_manager.pblast_is_rendering = True bpy.ops.render.playblast_modal_check(blend_pb=tempblend,video_pb=fp) # print("Playblast Done :", fp)#Dbg return fp class BGBLAST_OT_playblast_anim(bpy.types.Operator): bl_idname = "render.thread_playblast" bl_label = "Playblast anim" bl_description = "Launch animation playblast in a thread (non-blocking)" bl_options = {"REGISTER"} def execute(self, context): if not bpy.data.is_saved: self.report({'ERROR'}, 'File is not saved, Playblast cancelled') return {"CANCELLED"} playblast(context, viewport = False, stamping = True) return {"FINISHED"} def register(): bpy.types.WindowManager.pblast_is_rendering = bpy.props.BoolProperty() bpy.types.WindowManager.pblast_completion = bpy.props.IntProperty(min = 0, max = 100) bpy.types.WindowManager.pblast_previous_render = bpy.props.StringProperty() bpy.types.WindowManager.pblast_debug = bpy.props.BoolProperty('INVOKE_DEFAULT', name="Debug", default=False) bpy.utils.register_class(BGBLAST_OT_playblast_anim) bpy.utils.register_class(BGBLAST_OT_playblast_modal_check) def unregister(): bpy.utils.unregister_class(BGBLAST_OT_playblast_modal_check) bpy.utils.unregister_class(BGBLAST_OT_playblast_anim) del bpy.types.WindowManager.pblast_is_rendering del bpy.types.WindowManager.pblast_completion del bpy.types.WindowManager.pblast_previous_render del bpy.types.WindowManager.pblast_debug