Overhaul Pixmap Recorder

This commit is contained in:
Elias Batek 2024-08-18 02:40:18 +02:00
parent f432e7d744
commit 8efa5c7720
1 changed files with 57 additions and 6 deletions

View File

@ -27,6 +27,8 @@
inclusion a matter of viability for distributors. inclusion a matter of viability for distributors.
) )
### Tips and tricks
$(TIP $(TIP
The FFmpeg binary to be used can be specified by the optional The FFmpeg binary to be used can be specified by the optional
constructor parameter `ffmpegExecutablePath`. constructor parameter `ffmpegExecutablePath`.
@ -51,9 +53,46 @@
[PixmapRecorder.advancedFFmpegAdditionalOutputArgs|additional-output-args property]. [PixmapRecorder.advancedFFmpegAdditionalOutputArgs|additional-output-args property].
) )
## Examples $(TIP
Combining this module with [arsd.pixmappresenter|Pixmap Presenter]
is really straightforward.
### Getting started In the most simplistic case, setup a [PixmapRecorder] before running
the presenter.
Then call
[PixmapRecorder.put|pixmapRecorder.record(presenter.framebuffer)]
at the end of the drawing callback in the eventloop.
---
auto recorder = new PixmapRecorder(60, /* … */);
scope(exit) {
const recorderStatus = recorder.stopRecording();
}
return presenter.eventLoop(delegate() {
// […]
recorder.record(presenter.framebuffer);
return LoopCtrl.redrawIn(16);
}
---
)
$(TIP
To use this module with [arsd.color] (which includes the image file
loading functionality provided by other arsd modules),
convert the
[arsd.color.TrueColorImage|TrueColorImage] or
[arsd.color.MemoryImage|MemoryImage] to a
[arsd.pixmappaint.Pixmap|Pixmap] first by calling
[arsd.pixmappaint.Pixmap.fromTrueColorImage|Pixmap.fromTrueColorImage()]
or
[arsd.pixmappaint.Pixmap.fromMemoryImage|Pixmap.fromMemoryImage()]
respectively.
)
### Examples
#### Getting started
1. Install FFmpeg (the CLI version). 1. Install FFmpeg (the CLI version).
- Debian derivatives (with FFmpeg in their repos): `apt install ffmpeg` - Debian derivatives (with FFmpeg in their repos): `apt install ffmpeg`
@ -106,7 +145,7 @@ import arsd.pixmappaint;
import std.format; import std.format;
import std.path : buildPath; import std.path : buildPath;
import std.process; import std.process;
import std.range : OutputRange; import std.range : isOutputRange, OutputRange;
import std.sumtype; import std.sumtype;
import std.stdio : File; import std.stdio : File;
@ -135,7 +174,7 @@ private @safe {
FFmpeg will render an actual video file from the frame data. FFmpeg will render an actual video file from the frame data.
This uses the CLI version of FFmpeg, no linking required. This uses the CLI version of FFmpeg, no linking required.
+/ +/
final class PixmapRecorder : OutputRange!Pixmap { final class PixmapRecorder : OutputRange!(const(Pixmap)) {
@safe: @safe:
@ -302,6 +341,9 @@ final class PixmapRecorder : OutputRange!Pixmap {
return _input.writeEnd.isOpen; return _input.writeEnd.isOpen;
} }
/// ditto
alias isRecording = isOpen;
private string[] buildFFmpegCommand() pure { private string[] buildFFmpegCommand() pure {
// Build resolution as understood by FFmpeg. // Build resolution as understood by FFmpeg.
const string resolutionString = format!"%sx%s"( const string resolutionString = format!"%sx%s"(
@ -362,7 +404,7 @@ final class PixmapRecorder : OutputRange!Pixmap {
Theres usually no need to call this manually. Theres usually no need to call this manually.
) )
+/ +/
void open(Size resolution) void open(const Size resolution)
in (!this.isOpen) { in (!this.isOpen) {
// Save resolution for sanity checks. // Save resolution for sanity checks.
_resolution = resolution; _resolution = resolution;
@ -418,7 +460,7 @@ final class PixmapRecorder : OutputRange!Pixmap {
This function automatically calls [open|open()] if necessary. This function automatically calls [open|open()] if necessary.
) )
+/ +/
void put(Pixmap frame) { void put(const Pixmap frame) {
if (!this.isOpen) { if (!this.isOpen) {
this.open(frame.size); this.open(frame.size);
} else { } else {
@ -428,6 +470,9 @@ final class PixmapRecorder : OutputRange!Pixmap {
_input.writeEnd.rawWrite(frame.data); _input.writeEnd.rawWrite(frame.data);
} }
/// ditto
alias record = put;
/++ /++
Ends the recording process. Ends the recording process.
@ -455,3 +500,9 @@ final class PixmapRecorder : OutputRange!Pixmap {
/// ditto /// ditto
alias stopRecording = close; alias stopRecording = close;
} }
// self-test
private {
static assert(isOutputRange!(PixmapRecorder, Pixmap));
static assert(isOutputRange!(PixmapRecorder, const(Pixmap)));
}