vapoursynth: Allow scripts to control the progress dialog

This happens via VapourSynth's message logger.
Also add wrapper functions for this to aegisub_vs.py and add progress
updates everywhere - both to improve UX and to help with debugging for
when scripts get stuck somewhere.
This commit is contained in:
arch1t3cht 2023-10-18 14:20:37 +02:00
parent 79b3f4ccb0
commit a1b3e0d9f1
2 changed files with 72 additions and 6 deletions

View file

@ -22,6 +22,9 @@ Aegisub using the following variables:
- __aegi_hasaudio: int: If nonzero, Aegisub will try to load an audio track - __aegi_hasaudio: int: If nonzero, Aegisub will try to load an audio track
from the same file. from the same file.
The script can control the progress dialog shown by Aegisub with certain log
messages. Check the functions defined below for more information.
This module provides some utility functions to obtain timecodes, keyframes, and This module provides some utility functions to obtain timecodes, keyframes, and
other data. other data.
""" """
@ -40,6 +43,28 @@ aegi_vsplugins: str = ""
plugin_extension = ".dll" if os.name == "nt" else ".so" plugin_extension = ".dll" if os.name == "nt" else ".so"
def progress_set_message(message: str):
"""
Sets the message of Aegisub's progress dialog.
"""
vs.core.log_message(vs.MESSAGE_TYPE_DEBUG, f"__aegi_set_message,{message}")
def progress_set_progress(percent: float):
"""
Sets the progress shown in Aegisub's progress dialog to
the given percentage.
"""
vs.core.log_message(vs.MESSAGE_TYPE_DEBUG, f"__aegi_set_progress,{percent}")
def progress_set_indeterminate():
"""
Sets Aegisub's progress dialog to show indeterminate progress.
"""
vs.core.log_message(vs.MESSAGE_TYPE_DEBUG, f"__aegi_set_indeterminate,")
def set_paths(vars: dict): def set_paths(vars: dict):
""" """
Initialize the wrapper library with the given configuration directories. Initialize the wrapper library with the given configuration directories.
@ -157,6 +182,9 @@ def wrap_lwlibavsource(filename: str, cachedir: str | None = None, **kwargs: Any
pass pass
cachefile = os.path.join(cachedir, make_lwi_cache_filename(filename)) cachefile = os.path.join(cachedir, make_lwi_cache_filename(filename))
progress_set_message("Loading video file")
progress_set_indeterminate()
ensure_plugin("lsmas", "libvslsmashsource", "To use Aegisub's LWLibavSource wrapper, the `lsmas` plugin for VapourSynth must be installed") ensure_plugin("lsmas", "libvslsmashsource", "To use Aegisub's LWLibavSource wrapper, the `lsmas` plugin for VapourSynth must be installed")
if b"-Dcachedir" not in core.lsmas.Version()["config"]: # type: ignore if b"-Dcachedir" not in core.lsmas.Version()["config"]: # type: ignore
@ -164,6 +192,7 @@ def wrap_lwlibavsource(filename: str, cachedir: str | None = None, **kwargs: Any
clip = core.lsmas.LWLibavSource(source=filename, cachefile=cachefile, **kwargs) clip = core.lsmas.LWLibavSource(source=filename, cachefile=cachefile, **kwargs)
progress_set_message("Getting timecodes and keyframes from the index file")
return clip, info_from_lwindex(cachefile) return clip, info_from_lwindex(cachefile)
@ -181,6 +210,9 @@ def make_keyframes(clip: vs.VideoNode, use_scxvid: bool = False,
The remaining keyword arguments are passed on to the respective filter. The remaining keyword arguments are passed on to the respective filter.
""" """
progress_set_message("Generating keyframes")
progress_set_progress(1)
clip = core.resize.Bilinear(clip, width=resize_h * clip.width // clip.height, height=resize_h, format=resize_format) clip = core.resize.Bilinear(clip, width=resize_h * clip.width // clip.height, height=resize_h, format=resize_format)
if use_scxvid: if use_scxvid:
@ -196,12 +228,12 @@ def make_keyframes(clip: vs.VideoNode, use_scxvid: bool = False,
nonlocal done nonlocal done
keyframes[n] = f.props._SceneChangePrev if use_scxvid else f.props.Scenechange # type: ignore keyframes[n] = f.props._SceneChangePrev if use_scxvid else f.props.Scenechange # type: ignore
done += 1 done += 1
if done % (clip.num_frames // 25) == 0: if done % (clip.num_frames // 200) == 0:
vs.core.log_message(vs.MESSAGE_TYPE_INFORMATION, "Detecting keyframes... {}% done.\n".format(100 * done // clip.num_frames)) progress_set_progress(100 * done / clip.num_frames)
return f return f
deque(clip.std.ModifyFrame(clip, _cb).frames(close=True), 0) deque(clip.std.ModifyFrame(clip, _cb).frames(close=True), 0)
vs.core.log_message(vs.MESSAGE_TYPE_INFORMATION, "Done detecting keyframes.\n") progress_set_progress(100)
return [n for n in range(clip.num_frames) if keyframes[n]] return [n for n in range(clip.num_frames) if keyframes[n]]
@ -224,8 +256,12 @@ class GenKeyframesMode(Enum):
def ask_gen_keyframes(_: str) -> bool: def ask_gen_keyframes(_: str) -> bool:
from tkinter.messagebox import askyesno from tkinter.messagebox import askyesno
return askyesno("Generate Keyframes", \ progress_set_message("Asking whether to generate keyframes")
progress_set_indeterminate()
result = askyesno("Generate Keyframes", \
"No keyframes file was found for this video file.\nShould Aegisub detect keyframes from the video?\nThis will take a while.", default="no") "No keyframes file was found for this video file.\nShould Aegisub detect keyframes from the video?\nThis will take a while.", default="no")
progress_set_message("")
return result
def get_keyframes(filename: str, clip: vs.VideoNode, fallback: str | List[int], def get_keyframes(filename: str, clip: vs.VideoNode, fallback: str | List[int],
@ -244,6 +280,9 @@ def get_keyframes(filename: str, clip: vs.VideoNode, fallback: str | List[int],
generated or not generated or not
Additional keyword arguments are passed on to make_keyframes. Additional keyword arguments are passed on to make_keyframes.
""" """
progress_set_message("Looking for keyframes")
progress_set_indeterminate()
kffilename = make_keyframes_filename(filename) kffilename = make_keyframes_filename(filename)
if not os.path.exists(kffilename): if not os.path.exists(kffilename):
@ -252,7 +291,6 @@ def get_keyframes(filename: str, clip: vs.VideoNode, fallback: str | List[int],
if generate == GenKeyframesMode.ASK and not ask_callback(filename): if generate == GenKeyframesMode.ASK and not ask_callback(filename):
return fallback return fallback
vs.core.log_message(vs.MESSAGE_TYPE_INFORMATION, "No keyframes file found, detecting keyframes...\n")
keyframes = make_keyframes(clip, **kwargs) keyframes = make_keyframes(clip, **kwargs)
save_keyframes(kffilename, keyframes) save_keyframes(kffilename, keyframes)
@ -266,6 +304,8 @@ def check_audio(filename: str, **kwargs: Any) -> bool:
won't crash if it's not installed. won't crash if it's not installed.
Additional keyword arguments are passed on to BestAudioSource. Additional keyword arguments are passed on to BestAudioSource.
""" """
progress_set_message("Checking if the file has an audio track")
progress_set_indeterminate()
try: try:
ensure_plugin("bas", "BestAudioSource", "") ensure_plugin("bas", "BestAudioSource", "")
vs.core.bas.Source(source=filename, **kwargs) vs.core.bas.Source(source=filename, **kwargs)

View file

@ -21,8 +21,10 @@
#include "options.h" #include "options.h"
#include "utils.h" #include "utils.h"
#include <libaegisub/background_runner.h> #include <libaegisub/background_runner.h>
#include <libaegisub/format.h>
#include <libaegisub/fs.h> #include <libaegisub/fs.h>
#include <libaegisub/path.h> #include <libaegisub/path.h>
#include <libaegisub/util.h>
#include <boost/algorithm/string/replace.hpp> #include <boost/algorithm/string/replace.hpp>
@ -67,6 +69,30 @@ int OpenScriptOrVideo(const VSAPI *api, const VSSCRIPTAPI *sapi, VSScript *scrip
} }
void VSLogToProgressSink(int msgType, const char *msg, void *userData) { void VSLogToProgressSink(int msgType, const char *msg, void *userData) {
auto sink = reinterpret_cast<agi::ProgressSink *>(userData);
std::string msgStr(msg);
int commaPos = msgStr.find(',');
if (commaPos) {
std::string command = msgStr.substr(0, commaPos);
std::string tail = msgStr.substr(commaPos + 1, msgStr.length());
// We don't allow setting the title since that should stay as "Executing VapourSynth Script".
if (command == "__aegi_set_message") {
sink->SetMessage(tail);
} else if (command == "__aegi_set_progress") {
double percent;
if (!agi::util::try_parse(tail, &percent)) {
msgType = 2;
msgStr = agi::format("Warning: Invalid argument to __aegi_set_progress: %s\n", tail);
} else {
sink->SetProgress(percent, 100);
}
} else if (command == "__aegi_set_indeterminate") {
sink->SetIndeterminate();
}
}
int loglevel = 0; int loglevel = 0;
std::string loglevel_str = OPT_GET("Provider/Video/VapourSynth/Log Level")->GetString(); std::string loglevel_str = OPT_GET("Provider/Video/VapourSynth/Log Level")->GetString();
if (loglevel_str == "Quiet") if (loglevel_str == "Quiet")
@ -85,7 +111,7 @@ void VSLogToProgressSink(int msgType, const char *msg, void *userData) {
if (msgType < loglevel) if (msgType < loglevel)
return; return;
reinterpret_cast<agi::ProgressSink *>(userData)->Log(msg); sink->Log(msgStr);
} }
void VSCleanCache() { void VSCleanCache() {