You've already forked libopenshot
mirror of
https://github.com/OpenShot/libopenshot.git
synced 2026-03-02 08:53:52 -08:00
- Mask: Start/End (seconds) picks the exact part of a mask video to use. - Time Mode: Timeline follows project timing; Source FPS follows mask file timing. - Loop mode then defines playback behavior: Play Once, Repeat, or Ping-Pong.
555 lines
19 KiB
C++
555 lines
19 KiB
C++
/**
|
|
* @file
|
|
* @brief Unit tests for common EffectBase mask dispatch and blur mask modes
|
|
*/
|
|
|
|
// Copyright (c) 2008-2026 OpenShot Studios, LLC
|
|
//
|
|
// SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
#include <cstdlib>
|
|
#include <memory>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <unistd.h>
|
|
|
|
#include <QColor>
|
|
#include <QDir>
|
|
#include <QImage>
|
|
|
|
#include "CacheMemory.h"
|
|
#include "Clip.h"
|
|
#include "DummyReader.h"
|
|
#include "effects/Blur.h"
|
|
#include "effects/Brightness.h"
|
|
#include "effects/Hue.h"
|
|
#include "effects/Pixelate.h"
|
|
#include "effects/Saturation.h"
|
|
#include "effects/Sharpen.h"
|
|
#include "QtImageReader.h"
|
|
#include "openshot_catch.h"
|
|
|
|
using namespace openshot;
|
|
|
|
static std::string temp_png_path(const std::string& base) {
|
|
std::stringstream path;
|
|
path << QDir::tempPath().toStdString() << "/libopenshot_" << base << "_"
|
|
<< getpid() << "_" << rand() << ".png";
|
|
return path.str();
|
|
}
|
|
|
|
static std::string create_source_png(int w, int h, const QColor& color) {
|
|
const std::string path = temp_png_path("source");
|
|
QImage image(w, h, QImage::Format_RGBA8888_Premultiplied);
|
|
image.fill(color);
|
|
REQUIRE(image.save(QString::fromStdString(path)));
|
|
return path;
|
|
}
|
|
|
|
static std::string create_mask_png(const std::vector<int>& gray_values) {
|
|
const std::string path = temp_png_path("mask");
|
|
QImage mask(static_cast<int>(gray_values.size()), 1, QImage::Format_RGBA8888_Premultiplied);
|
|
for (size_t i = 0; i < gray_values.size(); ++i) {
|
|
const int gray = gray_values[i];
|
|
mask.setPixelColor(static_cast<int>(i), 0, QColor(gray, gray, gray, 255));
|
|
}
|
|
REQUIRE(mask.save(QString::fromStdString(path)));
|
|
return path;
|
|
}
|
|
|
|
static std::string create_uniform_mask_png(int width, int height, int gray_value) {
|
|
const std::string path = temp_png_path("mask_uniform");
|
|
QImage mask(width, height, QImage::Format_RGBA8888_Premultiplied);
|
|
mask.fill(QColor(gray_value, gray_value, gray_value, 255));
|
|
REQUIRE(mask.save(QString::fromStdString(path)));
|
|
return path;
|
|
}
|
|
|
|
class TrackingMaskReader : public ReaderBase {
|
|
private:
|
|
bool is_open = false;
|
|
CacheMemory cache;
|
|
int width = 2;
|
|
int height = 1;
|
|
|
|
public:
|
|
std::vector<int64_t> requests;
|
|
|
|
TrackingMaskReader(int fps_num, int fps_den, int64_t length_frames) {
|
|
info.has_video = true;
|
|
info.has_audio = false;
|
|
info.has_single_image = false;
|
|
info.width = width;
|
|
info.height = height;
|
|
info.fps = Fraction(fps_num, fps_den);
|
|
info.video_length = length_frames;
|
|
info.duration = static_cast<float>(length_frames / info.fps.ToDouble());
|
|
info.sample_rate = 48000;
|
|
info.channels = 2;
|
|
info.audio_stream_index = -1;
|
|
}
|
|
|
|
openshot::CacheBase* GetCache() override { return &cache; }
|
|
bool IsOpen() override { return is_open; }
|
|
std::string Name() override { return "TrackingMaskReader"; }
|
|
void Open() override { is_open = true; }
|
|
void Close() override { is_open = false; }
|
|
|
|
std::shared_ptr<openshot::Frame> GetFrame(int64_t number) override {
|
|
requests.push_back(number);
|
|
auto frame = std::make_shared<Frame>(number, width, height, "#00000000");
|
|
frame->GetImage()->fill(QColor(128, 128, 128, 255));
|
|
return frame;
|
|
}
|
|
|
|
std::string Json() const override {
|
|
return JsonValue().toStyledString();
|
|
}
|
|
|
|
Json::Value JsonValue() const override {
|
|
Json::Value root = ReaderBase::JsonValue();
|
|
root["type"] = "TrackingMaskReader";
|
|
root["path"] = "";
|
|
return root;
|
|
}
|
|
|
|
void SetJson(const std::string value) override {
|
|
(void) value;
|
|
}
|
|
|
|
void SetJsonValue(const Json::Value root) override {
|
|
ReaderBase::SetJsonValue(root);
|
|
}
|
|
};
|
|
|
|
static std::shared_ptr<Frame> make_input_frame(int64_t number, int width = 2, int height = 1) {
|
|
auto frame = std::make_shared<Frame>(number, width, height, "#00000000");
|
|
frame->GetImage()->fill(QColor(64, 64, 64, 255));
|
|
return frame;
|
|
}
|
|
|
|
TEST_CASE("EffectBase common mask blend applies to ProcessFrame", "[effect][mask][base]") {
|
|
auto frame = std::make_shared<Frame>(1, 4, 1, "#000000");
|
|
auto image = frame->GetImage();
|
|
image->fill(QColor(80, 80, 80, 255));
|
|
|
|
const std::string mask_path = create_mask_png({255, 255, 0, 0});
|
|
|
|
Brightness effect(Keyframe(0.5), Keyframe(0.0));
|
|
effect.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out = effect.ProcessFrame(frame, 1);
|
|
auto out_image = out->GetImage();
|
|
|
|
CHECK(out_image->pixelColor(0, 0).red() > 80);
|
|
CHECK(out_image->pixelColor(1, 0).red() > 80);
|
|
CHECK(out_image->pixelColor(2, 0).red() == 80);
|
|
CHECK(out_image->pixelColor(3, 0).red() == 80);
|
|
}
|
|
|
|
TEST_CASE("EffectBase mask fields serialize and deserialize", "[effect][mask][json]") {
|
|
const std::string mask_path = create_mask_png({255, 0});
|
|
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
effect.mask_invert = true;
|
|
effect.MaskReader(new QtImageReader(mask_path));
|
|
|
|
const Json::Value json = effect.JsonValue();
|
|
CHECK(json["mask_invert"].asBool());
|
|
REQUIRE(json["mask_reader"].isObject());
|
|
CHECK(json["mask_reader"]["type"].asString() == "QtImageReader");
|
|
|
|
Brightness copy(Keyframe(0.0), Keyframe(0.0));
|
|
copy.SetJsonValue(json);
|
|
CHECK(copy.mask_invert);
|
|
CHECK(copy.MaskReader() != nullptr);
|
|
}
|
|
|
|
TEST_CASE("Blur mask mode drive amount differs from post blend", "[effect][mask][blur]") {
|
|
const int width = 20;
|
|
const int height = 20;
|
|
auto frame_post = std::make_shared<Frame>(1, width, height, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, width, height, "#000000");
|
|
|
|
QImage input(width, height, QImage::Format_RGBA8888_Premultiplied);
|
|
for (int y = 0; y < height; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
const int alpha = std::min(255, x * 12 + y * 3);
|
|
input.setPixelColor(x, y, QColor(255, 180, 40, alpha));
|
|
}
|
|
}
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_uniform_mask_png(width, height, 128);
|
|
|
|
Blur post(Keyframe(3.0), Keyframe(3.0), Keyframe(3.0), Keyframe(1.0));
|
|
post.mask_mode = BLUR_MASK_POST_BLEND;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Blur drive(Keyframe(3.0), Keyframe(3.0), Keyframe(3.0), Keyframe(1.0));
|
|
drive.mask_mode = BLUR_MASK_DRIVE_AMOUNT;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
bool any_diff = false;
|
|
for (int y = 0; y < height && !any_diff; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
if (out_post->GetImage()->pixelColor(x, y) != out_drive->GetImage()->pixelColor(x, y)) {
|
|
any_diff = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!any_diff) {
|
|
for (int x = 0; x < width; ++x) {
|
|
if (out_post->GetImage()->pixelColor(x, height / 2) != out_drive->GetImage()->pixelColor(x, height / 2)) {
|
|
any_diff = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
CHECK(any_diff);
|
|
}
|
|
|
|
TEST_CASE("Saturation mask mode drive amount differs from post blend", "[effect][mask][saturation]") {
|
|
auto frame_post = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
|
|
QImage input(1, 1, QImage::Format_RGBA8888_Premultiplied);
|
|
input.setPixelColor(0, 0, QColor(70, 120, 200, 255));
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_mask_png({128});
|
|
|
|
Saturation post(Keyframe(2.0), Keyframe(1.0), Keyframe(1.0), Keyframe(1.0));
|
|
post.mask_mode = SATURATION_MASK_POST_BLEND;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Saturation drive(Keyframe(2.0), Keyframe(1.0), Keyframe(1.0), Keyframe(1.0));
|
|
drive.mask_mode = SATURATION_MASK_DRIVE_AMOUNT;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
CHECK(out_post->GetImage()->pixelColor(0, 0) != out_drive->GetImage()->pixelColor(0, 0));
|
|
}
|
|
|
|
TEST_CASE("Brightness mask mode vary strength differs from limit-to-area", "[effect][mask][brightness]") {
|
|
auto frame_post = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
|
|
QImage input(1, 1, QImage::Format_RGBA8888_Premultiplied);
|
|
input.setPixelColor(0, 0, QColor(80, 120, 200, 255));
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_mask_png({128});
|
|
|
|
Brightness post(Keyframe(0.6), Keyframe(6.0));
|
|
post.mask_mode = BRIGHTNESS_MASK_LIMIT_TO_AREA;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Brightness drive(Keyframe(0.6), Keyframe(6.0));
|
|
drive.mask_mode = BRIGHTNESS_MASK_VARY_STRENGTH;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
CHECK(out_post->GetImage()->pixelColor(0, 0) != out_drive->GetImage()->pixelColor(0, 0));
|
|
}
|
|
|
|
TEST_CASE("Hue mask mode vary strength differs from limit-to-area", "[effect][mask][hue]") {
|
|
auto frame_post = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, 1, 1, "#000000");
|
|
|
|
QImage input(1, 1, QImage::Format_RGBA8888_Premultiplied);
|
|
input.setPixelColor(0, 0, QColor(200, 80, 40, 255));
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_mask_png({128});
|
|
|
|
Hue post(Keyframe(0.8));
|
|
post.mask_mode = HUE_MASK_LIMIT_TO_AREA;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Hue drive(Keyframe(0.8));
|
|
drive.mask_mode = HUE_MASK_VARY_STRENGTH;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
CHECK(out_post->GetImage()->pixelColor(0, 0) != out_drive->GetImage()->pixelColor(0, 0));
|
|
}
|
|
|
|
TEST_CASE("Pixelate mask mode vary strength differs from limit-to-area", "[effect][mask][pixelate]") {
|
|
const int width = 20;
|
|
const int height = 20;
|
|
auto frame_post = std::make_shared<Frame>(1, width, height, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, width, height, "#000000");
|
|
|
|
QImage input(width, height, QImage::Format_RGBA8888_Premultiplied);
|
|
for (int y = 0; y < height; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
input.setPixelColor(x, y, QColor((x * 13) % 256, (y * 11) % 256, ((x + y) * 7) % 256, 255));
|
|
}
|
|
}
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_uniform_mask_png(width, height, 128);
|
|
|
|
Pixelate post(Keyframe(1.0), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0));
|
|
post.mask_mode = PIXELATE_MASK_LIMIT_TO_AREA;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Pixelate drive(Keyframe(1.0), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0));
|
|
drive.mask_mode = PIXELATE_MASK_VARY_STRENGTH;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
bool any_diff = false;
|
|
for (int y = 0; y < height && !any_diff; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
if (out_post->GetImage()->pixelColor(x, y) != out_drive->GetImage()->pixelColor(x, y)) {
|
|
any_diff = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
CHECK(any_diff);
|
|
}
|
|
|
|
TEST_CASE("Sharpen mask mode vary strength differs from limit-to-area", "[effect][mask][sharpen]") {
|
|
const int width = 20;
|
|
const int height = 20;
|
|
auto frame_post = std::make_shared<Frame>(1, width, height, "#000000");
|
|
auto frame_drive = std::make_shared<Frame>(1, width, height, "#000000");
|
|
|
|
QImage input(width, height, QImage::Format_RGBA8888_Premultiplied);
|
|
for (int y = 0; y < height; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
const int red = (x < width / 2) ? 30 : 220;
|
|
const int green = (y < height / 2) ? 60 : 200;
|
|
const int blue = ((x + y) % 2 == 0) ? 20 : 240;
|
|
input.setPixelColor(x, y, QColor(red, green, blue, 255));
|
|
}
|
|
}
|
|
*frame_post->GetImage() = input;
|
|
*frame_drive->GetImage() = input;
|
|
|
|
const std::string mask_path = create_uniform_mask_png(width, height, 128);
|
|
|
|
Sharpen post(Keyframe(1.6), Keyframe(3.0), Keyframe(0.05));
|
|
post.mask_mode = SHARPEN_MASK_LIMIT_TO_AREA;
|
|
post.MaskReader(new QtImageReader(mask_path));
|
|
|
|
Sharpen drive(Keyframe(1.6), Keyframe(3.0), Keyframe(0.05));
|
|
drive.mask_mode = SHARPEN_MASK_VARY_STRENGTH;
|
|
drive.MaskReader(new QtImageReader(mask_path));
|
|
|
|
auto out_post = post.ProcessFrame(frame_post, 1);
|
|
auto out_drive = drive.ProcessFrame(frame_drive, 1);
|
|
|
|
bool any_diff = false;
|
|
for (int y = 0; y < height && !any_diff; ++y) {
|
|
for (int x = 0; x < width; ++x) {
|
|
if (out_post->GetImage()->pixelColor(x, y) != out_drive->GetImage()->pixelColor(x, y)) {
|
|
any_diff = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
CHECK(any_diff);
|
|
}
|
|
|
|
TEST_CASE("Effect mask_mode roundtrip for supported effects", "[effect][mask][mode][json]") {
|
|
Blur blur(Keyframe(1.0), Keyframe(1.0), Keyframe(3.0), Keyframe(1.0));
|
|
blur.mask_mode = BLUR_MASK_DRIVE_AMOUNT;
|
|
Blur blur_copy;
|
|
blur_copy.SetJsonValue(blur.JsonValue());
|
|
CHECK(blur_copy.mask_mode == BLUR_MASK_DRIVE_AMOUNT);
|
|
|
|
Saturation saturation(Keyframe(1.0), Keyframe(1.0), Keyframe(1.0), Keyframe(1.0));
|
|
saturation.mask_mode = SATURATION_MASK_DRIVE_AMOUNT;
|
|
Saturation saturation_copy;
|
|
saturation_copy.SetJsonValue(saturation.JsonValue());
|
|
CHECK(saturation_copy.mask_mode == SATURATION_MASK_DRIVE_AMOUNT);
|
|
|
|
Brightness brightness(Keyframe(0.4), Keyframe(5.0));
|
|
brightness.mask_mode = BRIGHTNESS_MASK_VARY_STRENGTH;
|
|
Brightness brightness_copy;
|
|
brightness_copy.SetJsonValue(brightness.JsonValue());
|
|
CHECK(brightness_copy.mask_mode == BRIGHTNESS_MASK_VARY_STRENGTH);
|
|
|
|
Hue hue(Keyframe(0.4));
|
|
hue.mask_mode = HUE_MASK_VARY_STRENGTH;
|
|
Hue hue_copy;
|
|
hue_copy.SetJsonValue(hue.JsonValue());
|
|
CHECK(hue_copy.mask_mode == HUE_MASK_VARY_STRENGTH);
|
|
|
|
Pixelate pixelate(Keyframe(0.8), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0), Keyframe(0.0));
|
|
pixelate.mask_mode = PIXELATE_MASK_VARY_STRENGTH;
|
|
Pixelate pixelate_copy;
|
|
pixelate_copy.SetJsonValue(pixelate.JsonValue());
|
|
CHECK(pixelate_copy.mask_mode == PIXELATE_MASK_VARY_STRENGTH);
|
|
|
|
Sharpen sharpen(Keyframe(1.5), Keyframe(2.0), Keyframe(0.1));
|
|
sharpen.mask_mode = SHARPEN_MASK_VARY_STRENGTH;
|
|
Sharpen sharpen_copy;
|
|
sharpen_copy.SetJsonValue(sharpen.JsonValue());
|
|
CHECK(sharpen_copy.mask_mode == SHARPEN_MASK_VARY_STRENGTH);
|
|
}
|
|
|
|
TEST_CASE("EffectBase accepts legacy reader key for mask source", "[effect][mask][json][legacy_reader]") {
|
|
const std::string mask_path = create_mask_png({255, 0});
|
|
QtImageReader reader(mask_path);
|
|
|
|
Saturation effect(Keyframe(2.0), Keyframe(1.0), Keyframe(1.0), Keyframe(1.0));
|
|
Json::Value update;
|
|
update["reader"] = reader.JsonValue();
|
|
effect.SetJsonValue(update);
|
|
|
|
REQUIRE(effect.MaskReader() != nullptr);
|
|
CHECK(effect.JsonValue()["mask_reader"]["type"].asString() == "QtImageReader");
|
|
}
|
|
|
|
TEST_CASE("EffectBase mask timing fields roundtrip and clamp", "[effect][mask][timing][json]") {
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
|
|
Json::Value update;
|
|
update["mask_start"] = -10;
|
|
update["mask_end"] = -20;
|
|
update["mask_time_mode"] = 99;
|
|
update["mask_loop_mode"] = 99;
|
|
effect.SetJsonValue(update);
|
|
|
|
const Json::Value clamped = effect.JsonValue();
|
|
CHECK(clamped["mask_start"].asDouble() == Approx(0.0).margin(0.00001));
|
|
CHECK(clamped["mask_end"].asDouble() == Approx(0.0).margin(0.00001));
|
|
CHECK(clamped["mask_time_mode"].asInt() == 0);
|
|
CHECK(clamped["mask_loop_mode"].asInt() == 0);
|
|
|
|
update["mask_start"] = 0.5;
|
|
update["mask_end"] = 1.2;
|
|
update["mask_time_mode"] = 1;
|
|
update["mask_loop_mode"] = 2;
|
|
effect.SetJsonValue(update);
|
|
const Json::Value roundtrip = effect.JsonValue();
|
|
CHECK(roundtrip["mask_start"].asDouble() == Approx(0.5).margin(0.00001));
|
|
CHECK(roundtrip["mask_end"].asDouble() == Approx(1.2).margin(0.00001));
|
|
CHECK(roundtrip["mask_time_mode"].asInt() == 1);
|
|
CHECK(roundtrip["mask_loop_mode"].asInt() == 2);
|
|
}
|
|
|
|
TEST_CASE("EffectBase mask properties expose timing and loop controls", "[effect][mask][timing][properties]") {
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
Json::Value update;
|
|
update["mask_start"] = 0.2;
|
|
update["mask_end"] = 0.9;
|
|
update["mask_time_mode"] = 1;
|
|
update["mask_loop_mode"] = 2;
|
|
effect.SetJsonValue(update);
|
|
|
|
const Json::Value properties = openshot::stringToJson(effect.PropertiesJSON(1));
|
|
REQUIRE(properties.isObject());
|
|
CHECK(properties["mask_reader"]["name"].asString() == "Mask: Source");
|
|
CHECK(properties["mask_start"]["name"].asString() == "Mask: Start");
|
|
CHECK(properties["mask_end"]["name"].asString() == "Mask: End");
|
|
CHECK(properties["mask_time_mode"]["name"].asString() == "Mask: Time Mode");
|
|
CHECK(properties["mask_loop_mode"]["name"].asString() == "Mask: Loop");
|
|
CHECK(properties["mask_time_mode"]["choices"].size() == 2);
|
|
CHECK(properties["mask_loop_mode"]["choices"].size() == 3);
|
|
}
|
|
|
|
TEST_CASE("EffectBase timeline mode maps one-to-one with repeat loop", "[effect][mask][timing][timeline][repeat]") {
|
|
auto* tracking = new TrackingMaskReader(24, 1, 100);
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
effect.MaskReader(tracking);
|
|
|
|
Json::Value update;
|
|
update["mask_start"] = 1.0 / 24.0;
|
|
update["mask_end"] = 3.0 / 24.0;
|
|
update["mask_time_mode"] = 0; // Timeline
|
|
update["mask_loop_mode"] = 1; // Repeat
|
|
effect.SetJsonValue(update);
|
|
|
|
for (int64_t n = 1; n <= 7; ++n)
|
|
effect.ProcessFrame(make_input_frame(n), n);
|
|
|
|
const std::vector<int64_t> expected = {2, 3, 4, 2, 3, 4, 2};
|
|
CHECK(tracking->requests == expected);
|
|
}
|
|
|
|
TEST_CASE("EffectBase timeline mode supports ping-pong loop", "[effect][mask][timing][timeline][pingpong]") {
|
|
auto* tracking = new TrackingMaskReader(24, 1, 100);
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
effect.MaskReader(tracking);
|
|
|
|
Json::Value update;
|
|
update["mask_start"] = 1.0 / 24.0;
|
|
update["mask_end"] = 3.0 / 24.0;
|
|
update["mask_time_mode"] = 0; // Timeline
|
|
update["mask_loop_mode"] = 2; // Ping-Pong
|
|
effect.SetJsonValue(update);
|
|
|
|
for (int64_t n = 1; n <= 7; ++n)
|
|
effect.ProcessFrame(make_input_frame(n), n);
|
|
|
|
const std::vector<int64_t> expected = {2, 3, 4, 3, 2, 3, 4};
|
|
CHECK(tracking->requests == expected);
|
|
}
|
|
|
|
TEST_CASE("EffectBase timeline mode play-once clamps at end", "[effect][mask][timing][timeline][once]") {
|
|
auto* tracking = new TrackingMaskReader(24, 1, 100);
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
effect.MaskReader(tracking);
|
|
|
|
Json::Value update;
|
|
update["mask_start"] = 1.0 / 24.0;
|
|
update["mask_end"] = 3.0 / 24.0;
|
|
update["mask_time_mode"] = 0; // Timeline
|
|
update["mask_loop_mode"] = 0; // Play Once
|
|
effect.SetJsonValue(update);
|
|
|
|
for (int64_t n = 1; n <= 6; ++n)
|
|
effect.ProcessFrame(make_input_frame(n), n);
|
|
|
|
const std::vector<int64_t> expected = {2, 3, 4, 4, 4, 4};
|
|
CHECK(tracking->requests == expected);
|
|
}
|
|
|
|
TEST_CASE("EffectBase source FPS mode maps using parent clip FPS", "[effect][mask][timing][source_fps]") {
|
|
DummyReader clip_reader(Fraction(30, 1), 320, 240, 48000, 2, 4.0f);
|
|
Clip parent_clip;
|
|
parent_clip.Reader(&clip_reader);
|
|
|
|
auto* tracking = new TrackingMaskReader(15, 1, 100);
|
|
Brightness effect(Keyframe(0.0), Keyframe(0.0));
|
|
effect.ParentClip(&parent_clip);
|
|
effect.MaskReader(tracking);
|
|
|
|
Json::Value update;
|
|
update["mask_start"] = 0.0;
|
|
update["mask_end"] = 0;
|
|
update["mask_time_mode"] = 1; // Source FPS
|
|
update["mask_loop_mode"] = 0; // Play Once
|
|
effect.SetJsonValue(update);
|
|
|
|
for (int64_t n = 1; n <= 5; ++n)
|
|
effect.ProcessFrame(make_input_frame(n), n);
|
|
|
|
const std::vector<int64_t> expected = {1, 2, 2, 3, 3};
|
|
CHECK(tracking->requests == expected);
|
|
}
|