mirror of https://github.com/adamdruppe/arsd.git
Add Pixmap Recorder module
This commit is contained in:
parent
ad0f6109e6
commit
aba6a49f74
12
dub.json
12
dub.json
|
@ -694,6 +694,18 @@
|
||||||
"dflags-ldc": ["--mv=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"],
|
"dflags-ldc": ["--mv=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"],
|
||||||
"dflags-gdc": ["-fmodule-file=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"]
|
"dflags-gdc": ["-fmodule-file=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "pixmaprecorder",
|
||||||
|
"description": "Video rendering extension for Pixmap Paint. Fancy wrapper for piping frame data to FFmpeg.",
|
||||||
|
"targetType": "library",
|
||||||
|
"sourceFiles": ["pixmaprecorder.d"],
|
||||||
|
"dependencies": {
|
||||||
|
"arsd-official:pixmappaint":"*"
|
||||||
|
},
|
||||||
|
"dflags-dmd": ["-mv=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"],
|
||||||
|
"dflags-ldc": ["--mv=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"],
|
||||||
|
"dflags-gdc": ["-fmodule-file=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "pixmappresenter",
|
"name": "pixmappresenter",
|
||||||
"description": "High-level display library. Designed to blit fully-rendered frames to the screen.",
|
"description": "High-level display library. Designed to blit fully-rendered frames to the screen.",
|
||||||
|
|
|
@ -0,0 +1,411 @@
|
||||||
|
/+
|
||||||
|
== pixmaprecorder ==
|
||||||
|
Copyright Elias Batek (0xEAB) 2024.
|
||||||
|
Distributed under the Boost Software License, Version 1.0.
|
||||||
|
+/
|
||||||
|
/++
|
||||||
|
$(B Pixmap Recorder) is a helper library for rendering video files from
|
||||||
|
[arsd.pixmappaint.Pixmap|Pixmap] frames, by piping them to
|
||||||
|
[FFmpeg](https://ffmpeg.org/about.html).
|
||||||
|
|
||||||
|
$(SIDEBAR
|
||||||
|
Piping frame data to an independent copy of FFmpeg enables this library
|
||||||
|
to be used with a wide range of verions of said third-party program
|
||||||
|
and (hopefully) helps to reduce the chances of breaking changes.
|
||||||
|
|
||||||
|
It also allows end-users to upgrade their possibilities by swapping the
|
||||||
|
accompanying copy FFmpeg.
|
||||||
|
This could be useful in cases where software distributors can only
|
||||||
|
provide limited functionality in their bundled binaries because of
|
||||||
|
legal requirements like patent licenses.
|
||||||
|
)
|
||||||
|
|
||||||
|
$(TIP
|
||||||
|
The value of the `outputFormat` parameter of the constructor overloads
|
||||||
|
is passed to FFmpeg via the `-f` option.
|
||||||
|
|
||||||
|
Run `ffmpeg -formats` to get a list of available formats.
|
||||||
|
)
|
||||||
|
|
||||||
|
$(TIP
|
||||||
|
To pass additional options to FFmpeg, use the
|
||||||
|
[PixmapRecorder.advancedFFmpegAdditionalOutputArgs|additional-output-args property].
|
||||||
|
)
|
||||||
|
|
||||||
|
---
|
||||||
|
import arsd.pixmaprecorder;
|
||||||
|
import arsd.pixmappaint;
|
||||||
|
|
||||||
|
/++
|
||||||
|
This demo renders a 1280×720 video at 30 FPS
|
||||||
|
fading from white (#FFF) to blue (#00F).
|
||||||
|
+/
|
||||||
|
int main() {
|
||||||
|
// Instantiate a recorder.
|
||||||
|
auto recorder = new PixmapRecorder(
|
||||||
|
30, // Video framerate [=FPS]
|
||||||
|
"out.mkv", // Output path to write the video file to.
|
||||||
|
);
|
||||||
|
|
||||||
|
// We will use this framebuffer later on to provide image data
|
||||||
|
// to the encoder.
|
||||||
|
auto frame = Pixmap(1280, 720);
|
||||||
|
|
||||||
|
for (int light = 0xFF; light >= 0; --light) {
|
||||||
|
auto color = Color(light, light, 0xFF);
|
||||||
|
frame.clear(color);
|
||||||
|
|
||||||
|
// Record the current frame.
|
||||||
|
// The video resolution to use is derived from the first frame.
|
||||||
|
recorder.put(frame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// End and finalize the recording process.
|
||||||
|
return recorder.stopRecording();
|
||||||
|
}
|
||||||
|
---
|
||||||
|
+/
|
||||||
|
module arsd.pixmaprecorder;
|
||||||
|
|
||||||
|
import arsd.pixmappaint;
|
||||||
|
|
||||||
|
import std.format;
|
||||||
|
import std.path : buildPath;
|
||||||
|
import std.process;
|
||||||
|
import std.sumtype;
|
||||||
|
import std.stdio : File;
|
||||||
|
|
||||||
|
private @safe {
|
||||||
|
|
||||||
|
auto stderrFauxSafe() @trusted {
|
||||||
|
import std.stdio : stderr;
|
||||||
|
|
||||||
|
return stderr;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto stdoutFauxSafe() @trusted {
|
||||||
|
import std.stdio : stderr;
|
||||||
|
|
||||||
|
return stderr;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto stderr() {
|
||||||
|
return stderrFauxSafe;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto stdout() {
|
||||||
|
return stderrFauxSafe;
|
||||||
|
}
|
||||||
|
|
||||||
|
alias RecorderOutput = SumType!(string, File);
|
||||||
|
}
|
||||||
|
|
||||||
|
final class PixmapRecorder {
|
||||||
|
|
||||||
|
@safe:
|
||||||
|
|
||||||
|
private {
|
||||||
|
string _ffmpegExecutablePath;
|
||||||
|
double _frameRate;
|
||||||
|
string _outputFormat;
|
||||||
|
RecorderOutput _output;
|
||||||
|
File _log;
|
||||||
|
string[] _outputAdditionalArgs;
|
||||||
|
|
||||||
|
Pid _pid;
|
||||||
|
Pipe _input;
|
||||||
|
Size _resolution;
|
||||||
|
bool _outputIsOurs = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private this(
|
||||||
|
string ffmpegExecutablePath,
|
||||||
|
double frameRate,
|
||||||
|
string outputFormat,
|
||||||
|
RecorderOutput output,
|
||||||
|
File log,
|
||||||
|
) {
|
||||||
|
_ffmpegExecutablePath = ffmpegExecutablePath;
|
||||||
|
_frameRate = frameRate;
|
||||||
|
_outputFormat = outputFormat;
|
||||||
|
_output = output;
|
||||||
|
_log = log;
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Prepares a recorder for encoding video frames
|
||||||
|
into the specified file pipe.
|
||||||
|
|
||||||
|
$(WARNING
|
||||||
|
Certain formats cannot be produced in pipes by FFmpeg.
|
||||||
|
Look out for error message like such:
|
||||||
|
|
||||||
|
($BLOCKQUOTE
|
||||||
|
`[mp4 @ 0xdead1337beef] muxer does not support non seekable output`
|
||||||
|
)
|
||||||
|
|
||||||
|
This is not a limitation of this library (but rather one of FFmpeg).
|
||||||
|
Let FFmpeg output the video to file path instead;
|
||||||
|
check out the other overloads of this constructor.
|
||||||
|
)
|
||||||
|
|
||||||
|
Params:
|
||||||
|
frameRate = Framerate of the video output; in frames per second.
|
||||||
|
output = File handle to write the video output to.
|
||||||
|
outputFormat = Video (container) format to output.
|
||||||
|
This is value passed to FFmpeg via the `-f` option.
|
||||||
|
log = Target file for the stderr log output of FFmpeg.
|
||||||
|
This is where error messages are written to.
|
||||||
|
ffmpegExecutablePath = Path to the FFmpeg executable
|
||||||
|
(e.g. `ffmpeg`, `ffmpeg.exe` or `/usr/bin/ffmpeg`).
|
||||||
|
/* Keep this table in sync with the ones of other overloads. */
|
||||||
|
+/
|
||||||
|
public this(
|
||||||
|
double frameRate,
|
||||||
|
File output,
|
||||||
|
string outputFormat,
|
||||||
|
File log = stderr,
|
||||||
|
string ffmpegExecutablePath = "ffmpeg",
|
||||||
|
)
|
||||||
|
in (frameRate > 0)
|
||||||
|
in (output.isOpen)
|
||||||
|
in (outputFormat != "")
|
||||||
|
in (log.isOpen)
|
||||||
|
in (ffmpegExecutablePath != "") {
|
||||||
|
this(
|
||||||
|
ffmpegExecutablePath,
|
||||||
|
frameRate,
|
||||||
|
outputFormat,
|
||||||
|
RecorderOutput(output),
|
||||||
|
log,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Prepares a recorder for encoding video frames
|
||||||
|
into a video file saved to the specified path.
|
||||||
|
|
||||||
|
Params:
|
||||||
|
frameRate = Framerate of the video output; in frames per second.
|
||||||
|
outputPath = File path to write the video output to.
|
||||||
|
Existing files will be overwritten.
|
||||||
|
FFmpeg will use this to autodetect the format
|
||||||
|
when no `outputFormat` is provided.
|
||||||
|
log = Target file for the stderr log output of FFmpeg.
|
||||||
|
This is where error messages are written to.
|
||||||
|
outputFormat = Video (container) format to output.
|
||||||
|
This is value passed to FFmpeg via the `-f` option.
|
||||||
|
If `null`, the format is not provided and FFmpeg
|
||||||
|
will try to autodetect the format from the filename
|
||||||
|
of the `outputPath`.
|
||||||
|
ffmpegExecutablePath = Path to the FFmpeg executable
|
||||||
|
(e.g. `ffmpeg`, `ffmpeg.exe` or `/usr/bin/ffmpeg`).
|
||||||
|
/* Keep this table in sync with the ones of other overloads. */
|
||||||
|
+/
|
||||||
|
public this(
|
||||||
|
double frameRate,
|
||||||
|
string outputPath,
|
||||||
|
File log = stderr,
|
||||||
|
string outputFormat = null,
|
||||||
|
string ffmpegExecutablePath = "ffmpeg",
|
||||||
|
)
|
||||||
|
in (frameRate > 0)
|
||||||
|
in ((outputPath != "") && (outputPath != "-"))
|
||||||
|
in (log.isOpen)
|
||||||
|
in ((outputFormat is null) || outputFormat != "")
|
||||||
|
in (ffmpegExecutablePath != "") {
|
||||||
|
|
||||||
|
// Sanitize output path
|
||||||
|
// if it would get confused with a command-line arg.
|
||||||
|
// Otherwise a relative path like `-my.mkv` would make FFmpeg complain
|
||||||
|
// about an “Unrecognized option 'out.mkv'”.
|
||||||
|
if (outputPath[0] == '-') {
|
||||||
|
outputPath = buildPath(".", outputPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
this(
|
||||||
|
ffmpegExecutablePath,
|
||||||
|
frameRate,
|
||||||
|
null,
|
||||||
|
RecorderOutput(outputPath),
|
||||||
|
log,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
$(I Advanced users only:)
|
||||||
|
Additional command-line arguments passed to FFmpeg.
|
||||||
|
|
||||||
|
$(WARNING
|
||||||
|
The values provided through this property function are not
|
||||||
|
validated and passed verbatim to FFmpeg.
|
||||||
|
)
|
||||||
|
|
||||||
|
$(PITFAL
|
||||||
|
If code makes use of this and FFmpeg errors,
|
||||||
|
check the arguments provided here this first.
|
||||||
|
)
|
||||||
|
+/
|
||||||
|
void advancedFFmpegAdditionalOutputArgs(string[] args) {
|
||||||
|
_outputAdditionalArgs = args;
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Determines whether the recorder is active
|
||||||
|
(which implies that an output file is open).
|
||||||
|
+/
|
||||||
|
bool isOpen() {
|
||||||
|
return _input.writeEnd.isOpen;
|
||||||
|
}
|
||||||
|
|
||||||
|
private string[] buildFFmpegCommand() pure {
|
||||||
|
// Build resolution as understood by FFmpeg.
|
||||||
|
const string resolutionString = format!"%sx%s"(
|
||||||
|
_resolution.width,
|
||||||
|
_resolution.height,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Convert framerate to string.
|
||||||
|
const string frameRateString = format!"%s"(_frameRate);
|
||||||
|
|
||||||
|
// Build command-line argument list.
|
||||||
|
auto cmd = [
|
||||||
|
_ffmpegExecutablePath,
|
||||||
|
"-y",
|
||||||
|
"-r",
|
||||||
|
frameRateString,
|
||||||
|
"-f",
|
||||||
|
"rawvideo",
|
||||||
|
"-pix_fmt",
|
||||||
|
"rgba",
|
||||||
|
"-s",
|
||||||
|
resolutionString,
|
||||||
|
"-i",
|
||||||
|
"-",
|
||||||
|
];
|
||||||
|
|
||||||
|
if (_outputFormat !is null) {
|
||||||
|
cmd ~= "-f";
|
||||||
|
cmd ~= _outputFormat;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_outputAdditionalArgs.length > 0) {
|
||||||
|
cmd = cmd ~ _outputAdditionalArgs;
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd ~= _output.match!(
|
||||||
|
(string filePath) => filePath,
|
||||||
|
(ref File file) => "-",
|
||||||
|
);
|
||||||
|
|
||||||
|
return cmd;
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Starts the video encoding process.
|
||||||
|
Launches FFmpeg.
|
||||||
|
|
||||||
|
This function sets the video resolution for the encoding process.
|
||||||
|
All frames to record must match it.
|
||||||
|
|
||||||
|
$(SIDEBAR
|
||||||
|
Variable/dynamic resolution is neither supported by this library
|
||||||
|
nor most real-world applications.
|
||||||
|
)
|
||||||
|
|
||||||
|
$(NOTE
|
||||||
|
This function is called by [put|put()] automatically.
|
||||||
|
There’s usually no need to call this manually.
|
||||||
|
)
|
||||||
|
+/
|
||||||
|
void open(Size resolution)
|
||||||
|
in (!this.isOpen) {
|
||||||
|
// Save resolution for sanity checks.
|
||||||
|
_resolution = resolution;
|
||||||
|
|
||||||
|
const string[] cmd = buildFFmpegCommand();
|
||||||
|
|
||||||
|
// Prepare arsd → FFmpeg I/O pipe.
|
||||||
|
_input = pipe();
|
||||||
|
|
||||||
|
// Launch FFmpeg.
|
||||||
|
const processConfig = (
|
||||||
|
Config.suppressConsole
|
||||||
|
| Config.newEnv
|
||||||
|
);
|
||||||
|
|
||||||
|
// dfmt off
|
||||||
|
_pid = _output.match!(
|
||||||
|
delegate(string filePath) {
|
||||||
|
auto stdout = pipe();
|
||||||
|
stdout.readEnd.close();
|
||||||
|
return spawnProcess(
|
||||||
|
cmd,
|
||||||
|
_input.readEnd,
|
||||||
|
stdout.writeEnd,
|
||||||
|
_log,
|
||||||
|
null,
|
||||||
|
processConfig,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
delegate(File file) {
|
||||||
|
auto stdout = pipe();
|
||||||
|
stdout.readEnd.close();
|
||||||
|
return spawnProcess(
|
||||||
|
cmd,
|
||||||
|
_input.readEnd,
|
||||||
|
file,
|
||||||
|
_log,
|
||||||
|
null,
|
||||||
|
processConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// dfmt on
|
||||||
|
}
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
alias startRecording = close;
|
||||||
|
|
||||||
|
/++
|
||||||
|
Provides the next video frame to encode.
|
||||||
|
|
||||||
|
$(TIP
|
||||||
|
This function automatically calls [open|open()] if necessary.
|
||||||
|
)
|
||||||
|
+/
|
||||||
|
void put(Pixmap frame) {
|
||||||
|
if (!this.isOpen) {
|
||||||
|
this.open(frame.size);
|
||||||
|
} else {
|
||||||
|
assert(frame.size == _resolution, "Variable resolutions are not supported.");
|
||||||
|
}
|
||||||
|
|
||||||
|
_input.writeEnd.rawWrite(frame.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Ends the recording process.
|
||||||
|
|
||||||
|
$(NOTE
|
||||||
|
Waits for the FFmpeg process to exit in a blocking way.
|
||||||
|
)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The status code provided by the FFmpeg program.
|
||||||
|
+/
|
||||||
|
int close()
|
||||||
|
in (this.isOpen) {
|
||||||
|
|
||||||
|
_input.writeEnd.flush();
|
||||||
|
_input.writeEnd.close();
|
||||||
|
scope (exit) {
|
||||||
|
_input.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
return wait(_pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
alias stopRecording = close;
|
||||||
|
}
|
Loading…
Reference in New Issue