123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199 |
- # ##### 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,
- # Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- #
- # ##### END GPL LICENSE BLOCK #####
- # <pep8-80 compliant>
- # Originally written by Matt Ebb
- import bpy
- from bpy.types import Operator
- import os
- from bpy.app.translations import pgettext_tip as tip_
- def guess_player_path(preset):
- import sys
- if preset == 'INTERNAL':
- return bpy.app.binary_path
- elif preset == 'BLENDER24':
- player_path = "blender"
- if sys.platform == "darwin":
- test_path = "/Applications/blender 2.49.app/Contents/MacOS/blender"
- elif sys.platform[:3] == "win":
- test_path = "/Program Files/Blender Foundation/Blender/blender.exe"
- if os.path.exists(test_path):
- player_path = test_path
- elif preset == 'DJV':
- player_path = "djv_view"
- if sys.platform == "darwin":
- # TODO, crummy supporting only 1 version,
- # could find the newest installed version
- test_path = ("/Applications/djv-0.8.2.app"
- "/Contents/Resources/bin/djv_view")
- if os.path.exists(test_path):
- player_path = test_path
- elif preset == 'FRAMECYCLER':
- player_path = "framecycler"
- elif preset == 'RV':
- player_path = "rv"
- elif preset == 'MPLAYER':
- player_path = "mplayer"
- else:
- player_path = ""
- return player_path
- class PlayRenderedAnim(Operator):
- """Play back rendered frames/movies using an external player"""
- bl_idname = "render.play_rendered_anim"
- bl_label = "Play Rendered Animation"
- bl_options = {'REGISTER'}
- def execute(self, context):
- import subprocess
- from shlex import quote
- scene = context.scene
- rd = scene.render
- prefs = context.preferences
- fps_final = rd.fps / rd.fps_base
- preset = prefs.filepaths.animation_player_preset
- player_path = prefs.filepaths.animation_player
- # file_path = bpy.path.abspath(rd.filepath) # UNUSED
- is_movie = rd.is_movie_format
- # try and guess a command line if it doesn't exist
- if player_path == "":
- player_path = guess_player_path(preset)
- if is_movie is False and preset in {'FRAMECYCLER', 'RV', 'MPLAYER'}:
- # replace the number with '#'
- file_a = rd.frame_path(frame=0)
- # TODO, make an api call for this
- frame_tmp = 9
- file_b = rd.frame_path(frame=frame_tmp)
- while len(file_a) == len(file_b):
- frame_tmp = (frame_tmp * 10) + 9
- file_b = rd.frame_path(frame=frame_tmp)
- file_b = rd.frame_path(frame=int(frame_tmp / 10))
- file = ("".join((c if file_b[i] == c else "#")
- for i, c in enumerate(file_a)))
- del file_a, file_b, frame_tmp
- file = bpy.path.abspath(file) # expand '//'
- else:
- path_valid = True
- # works for movies and images
- file = rd.frame_path(frame=scene.frame_start, preview=scene.use_preview_range)
- file = bpy.path.abspath(file) # expand '//'
- if not os.path.exists(file):
- err_msg = tip_("File %r not found") % file
- self.report({'WARNING'}, err_msg)
- path_valid = False
- # one last try for full range if we used preview range
- if scene.use_preview_range and not path_valid:
- file = rd.frame_path(frame=scene.frame_start, preview=False)
- file = bpy.path.abspath(file) # expand '//'
- err_msg = tip_("File %r not found") % file
- if not os.path.exists(file):
- self.report({'WARNING'}, err_msg)
- cmd = [player_path]
- # extra options, fps controls etc.
- if scene.use_preview_range:
- frame_start = scene.frame_preview_start
- frame_end = scene.frame_preview_end
- else:
- frame_start = scene.frame_start
- frame_end = scene.frame_end
- if preset == 'INTERNAL':
- opts = [
- "-a",
- "-f", str(rd.fps), str(rd.fps_base),
- "-s", str(frame_start),
- "-e", str(frame_end),
- "-j", str(scene.frame_step),
- file,
- ]
- cmd.extend(opts)
- elif preset == 'DJV':
- opts = [file, "-playback_speed", str(int(fps_final))]
- cmd.extend(opts)
- elif preset == 'FRAMECYCLER':
- opts = [file, f"{scene.frame_start:d}-{scene.frame_end:d}"]
- cmd.extend(opts)
- elif preset == 'RV':
- opts = ["-fps", str(rd.fps), "-play", f"[ {file:s} ]"]
- cmd.extend(opts)
- elif preset == 'MPLAYER':
- opts = []
- if is_movie:
- opts.append(file)
- else:
- opts += [
- ("mf://" + file.replace("#", "?")),
- "-mf",
- f"fps={fps_final:4f}"
- ]
- opts += ["-loop", "0", "-really-quiet", "-fs"]
- cmd.extend(opts)
- else: # 'CUSTOM'
- cmd.append(file)
- # launch it
- print("Executing command:\n ", " ".join(quote(c) for c in cmd))
- # workaround for boost 1.46, can be eventually removed. bug: [#32350]
- env_copy = os.environ.copy()
- if preset == 'INTERNAL':
- env_copy["LC_ALL"] = "C"
- # end workaround
- try:
- subprocess.Popen(cmd, env=env_copy)
- except Exception as e:
- err_msg = tip_("Couldn't run external animation player with command %r\n%s") % (cmd, e)
- self.report(
- {'ERROR'},
- err_msg,
- )
- return {'CANCELLED'}
- return {'FINISHED'}
- classes = (
- PlayRenderedAnim,
- )
|