blob: b8e8dec1901daf052eebd784c184e90be2a28dac [file] [log] [blame]
#include "rive/file.hpp"
#include "rive/animation/linear_animation.hpp"
#include "rive/animation/linear_animation_instance.hpp"
#include "rive/animation/state_machine_instance.hpp"
#include "rive/viewmodel/viewmodel.hpp"
#include "rive/viewmodel/viewmodel_instance_boolean.hpp"
#include "rive/viewmodel/viewmodel_instance_color.hpp"
#include "rive/viewmodel/viewmodel_instance_enum.hpp"
#include "rive/viewmodel/viewmodel_instance_list.hpp"
#include "rive/viewmodel/viewmodel_instance_number.hpp"
#include "rive/viewmodel/viewmodel_instance_trigger.hpp"
#include "rive/viewmodel/viewmodel_instance_string.hpp"
#include "rive/viewmodel/viewmodel_instance_viewmodel.hpp"
#include "rive/math/random.hpp"
#include "utils/serializing_factory.hpp"
#include "rive_file_reader.hpp"
#include <catch.hpp>
#include <cstdio>
#include <cstring>
using namespace rive;
TEST_CASE("juice silver", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/juice.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto walkAnimation = artboard->animationNamed("walk");
REQUIRE(walkAnimation != nullptr);
auto renderer = silver.makeRenderer();
// Draw first frame.
walkAnimation->advanceAndApply(0.0f);
artboard->draw(renderer.get());
int frames = (int)(walkAnimation->durationSeconds() / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
walkAnimation->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("juice"));
}
TEST_CASE("hide silver", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/hide_test.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(1, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("hide_test"));
}
TEST_CASE("n-slice silver", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/n_slice_triangle.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
artboard->advance(0.0f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
CHECK(silver.matches("n_slice_triangle"));
}
TEST_CASE("lock icon listener silver", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/lock_icon_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
// Click in the middle of the state machine.
stateMachine->pointerDown(
rive::Vec2D(artboard->width() / 2.0f, artboard->height() / 2.0f));
// Advance and apply twice to take the transition and apply the next state.
stateMachine->advanceAndApply(0.1f);
stateMachine->advanceAndApply(1.0f);
artboard->draw(renderer.get());
silver.addFrame();
// Do it again to lock the icon.
stateMachine->pointerDown(
rive::Vec2D(artboard->width() / 2.0f, artboard->height() / 2.0f));
// Advance and apply twice to take the transition and apply the next state.
stateMachine->advanceAndApply(0.1f);
stateMachine->advanceAndApply(1.0f);
artboard->draw(renderer.get());
CHECK(silver.matches("lock_icon_demo"));
}
TEST_CASE("validate text run listener works", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/text_listener_simpler.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
// Click in the middle of the state machine.
stateMachine->pointerDown(
rive::Vec2D(artboard->width() * 0.8, artboard->height() / 2.0f));
// Advance and apply twice to take the transition and apply the next state.
stateMachine->advanceAndApply(0.1f);
stateMachine->advanceAndApply(1.0f);
artboard->draw(renderer.get());
CHECK(silver.matches("text_listener_simpler"));
}
TEST_CASE("validate text with modifiers and dashes render correctly",
"[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/text_stroke_test.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
// Draw first frame.
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("text_stroke_test"));
}
TEST_CASE("superbowl data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/superbowl.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(10.0f / 1.0f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(1.0f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("superbowl"));
}
TEST_CASE("data viz demo data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/data_viz_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
auto vmiItem1 = vmi->propertyValue("item1");
auto vmiItem1ViewModelInstance = vmiItem1->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
if (vmiItem1ViewModelInstance != nullptr)
{
auto valueProp = vmiItem1ViewModelInstance->propertyValue("value");
if (valueProp != nullptr)
{
valueProp->as<ViewModelInstanceNumber>()->propertyValue(20);
}
}
int frames = (int)(0.8f / 0.064f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.064f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("data_viz_demo"));
}
TEST_CASE("bank card data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/bankcard.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(2.0f / 0.1f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("bankcard"));
}
TEST_CASE("ai assistant data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/ai_assitant.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
auto leftNumber = vmi->propertyValue("left");
auto bottomNumber = vmi->propertyValue("bottom");
auto topNumber = vmi->propertyValue("top");
auto rightNumber = vmi->propertyValue("right");
int frames = (int)(1.0f / 0.33f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
leftNumber->as<ViewModelInstanceNumber>()->propertyValue(i * 10);
bottomNumber->as<ViewModelInstanceNumber>()->propertyValue(i * 5);
topNumber->as<ViewModelInstanceNumber>()->propertyValue(i * 3);
rightNumber->as<ViewModelInstanceNumber>()->propertyValue(i * 2);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("ai_assitant"));
}
TEST_CASE("echo show demo data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/echo_show_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("echo_show_demo"));
}
TEST_CASE("rewards demo data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/rewards_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
auto buttonProp = vmi->propertyValue("Button");
if (buttonProp != nullptr)
{
auto buttonVm = buttonProp->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
if (buttonVm != nullptr)
{
auto trigger = buttonVm->propertyValue("Pressed");
trigger->as<ViewModelInstanceTrigger>()->trigger();
}
}
int frames = (int)(5.0f / 0.25f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("rewards_demo"));
}
TEST_CASE("spotify kids demo data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/spotify_kids_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("spotify_kids_demo"));
}
TEST_CASE("spotify kids app icon data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/spotify_kids_app_icon.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("spotify_kids_app_icon"));
}
TEST_CASE("hunter x demo data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/hunter_x_demo.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("hunter_x_demo"));
}
TEST_CASE("health tracker data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/db_health_tracker.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("db_health_tracker"));
}
TEST_CASE("car widgets data binding", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/car_widgets_v01.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
auto vmiItem1 = vmi->propertyValue("COMPASS");
auto vmiItem1ViewModelInstance = vmiItem1->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
if (vmiItem1ViewModelInstance != nullptr)
{
auto valueProp = vmiItem1ViewModelInstance->propertyValue("Rotation");
if (valueProp != nullptr)
{
valueProp->as<ViewModelInstanceNumber>()->propertyValue(20);
}
}
auto vmiItem2 = vmi->propertyValue("TIRE PSI");
auto vmiItem2ViewModelInstance = vmiItem2->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
if (vmiItem2ViewModelInstance != nullptr)
{
auto valueProp = vmiItem2ViewModelInstance->propertyValue("FL Tyre");
if (valueProp != nullptr)
{
valueProp->as<ViewModelInstanceNumber>()->propertyValue(10);
}
}
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("car_widgets_v01"));
}
TEST_CASE("Vertical align on text with ellipsis", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/vertical_align_ellipsis.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
CHECK(silver.matches("vertical_align_ellipsis"));
}
TEST_CASE("Event triggers another event", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/event_trigger_event.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
// Click in a square that fires a trigger. This trigger will cause a
// transition that fires an event. There is a listener on that event that
// fires a second event.
stateMachine->pointerDown(rive::Vec2D(475, 25));
stateMachine->pointerUp(rive::Vec2D(475, 25));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("event_trigger_event"));
}
TEST_CASE("Collapsed data binds from layouts with display hidden", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/collapse_data_binds.riv", &silver);
auto artboard = file->artboardNamed("test-1");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->advanceAndApply(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.016f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("collapse_data_binds-test_1"));
}
TEST_CASE("Collapsed data binds from property groups in solos", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/collapse_data_binds.riv", &silver);
auto artboard = file->artboardNamed("test-2");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->advanceAndApply(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.016f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("collapse_data_binds-test_2"));
}
TEST_CASE("Collapsed data bound layout styles still update", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/collapse_data_binds.riv", &silver);
auto artboard = file->artboardNamed("test-3");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto display1 =
vmi->propertyValue("display_1")->as<ViewModelInstanceEnum>();
auto display2 =
vmi->propertyValue("display_2")->as<ViewModelInstanceEnum>();
stateMachine->advanceAndApply(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.016f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
display2->value(1); // Hide inner layout
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
silver.addFrame();
display1->value(1); // Hide outer layout
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
silver.addFrame();
display2->value(0); // Show inner layout (nothing changes)
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
silver.addFrame();
display1->value(0); // Show outer layout (nothing changes)
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
CHECK(silver.matches("collapse_data_binds-test_3"));
}
TEST_CASE("Reset randomization on value change", "[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/formula_random.riv", &silver);
auto artboard = file->artboardNamed("source_change");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto numProp = vmi->propertyValue("n1")->as<ViewModelInstanceNumber>();
RandomProvider::addRandomValue(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
REQUIRE(RandomProvider::totalCalls() == 1);
silver.addFrame();
RandomProvider::addRandomValue(1.0f);
numProp->propertyValue(500);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
// Random generation hasn't been called again
REQUIRE(RandomProvider::totalCalls() == 2);
RandomProvider::clearRandoms();
CHECK(silver.matches("formula_random-source_change"));
}
TEST_CASE("Reset randomization only once", "[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/formula_random.riv", &silver);
auto artboard = file->artboardNamed("once");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto numProp = vmi->propertyValue("n1")->as<ViewModelInstanceNumber>();
RandomProvider::addRandomValue(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
REQUIRE(RandomProvider::totalCalls() == 1);
silver.addFrame();
RandomProvider::addRandomValue(1.0f);
numProp->propertyValue(500);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
// Random generation has been called again
REQUIRE(RandomProvider::totalCalls() == 1);
RandomProvider::clearRandoms();
CHECK(silver.matches("formula_random-once"));
}
TEST_CASE("Reset randomization on every change", "[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/formula_random.riv", &silver);
auto artboard = file->artboardNamed("always");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto numProp = vmi->propertyValue("n1")->as<ViewModelInstanceNumber>();
RandomProvider::addRandomValue(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
REQUIRE(RandomProvider::totalCalls() == 1);
silver.addFrame();
numProp->propertyValue(500);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
// Random generation hasn't been called on every advance
REQUIRE(RandomProvider::totalCalls() == 64);
RandomProvider::clearRandoms();
CHECK(silver.matches("formula_random-always"));
}
TEST_CASE("Target to source with different data types on source and target",
"[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/saturation.riv", &silver);
auto artboard = file->artboardNamed("main");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->advanceAndApply(0.0f);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.016f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("saturation"));
}
TEST_CASE("interactive and non interactive scrolling", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/interactive_scrolling.riv", &silver);
auto artboard = file->artboardDefault();
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto boolProp =
vmi->propertyValue("isInteractive")->as<ViewModelInstanceBoolean>();
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
double yPos = artboard->height() - 20;
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
while (yPos > 120)
{
silver.addFrame();
stateMachine->pointerMove(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
yPos -= 20;
}
silver.addFrame();
stateMachine->pointerUp(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Change to interactive
boolProp->propertyValue(true);
stateMachine->advanceAndApply(0.1f);
yPos = artboard->height() - 20;
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
while (yPos > 120)
{
silver.addFrame();
stateMachine->pointerMove(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
yPos -= 20;
}
silver.addFrame();
stateMachine->pointerUp(rive::Vec2D(artboard->width() / 2.0f, yPos));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("interactive_scrolling"));
}
TEST_CASE("Interpolator returns advance status as true until it settles",
"[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/interpolate_to_end.riv", &silver);
auto artboard = file->artboardNamed("child");
silver.frameSize(artboard->width(), artboard->height());
REQUIRE(artboard != nullptr);
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
auto numProp = vmi->propertyValue("num")->as<ViewModelInstanceNumber>();
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
numProp->propertyValue(1000);
stateMachine->advanceAndApply(0.001f);
auto shouldAdvance = true;
while (shouldAdvance)
{
silver.addFrame();
shouldAdvance = stateMachine->advanceAndApply(0.25f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("interpolate_to_end"));
}
TEST_CASE("View models keep reference to their dependents and parents",
"[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/replace_vm_instance.riv", &silver);
auto artboardMain1 = file->artboardNamed("main-1");
auto artboardMain2 = file->artboardNamed("main-2");
auto artboardCommon = file->artboardNamed("common");
REQUIRE(artboardMain1 != nullptr);
REQUIRE(artboardMain2 != nullptr);
REQUIRE(artboardCommon != nullptr);
silver.frameSize(artboardMain1->width(), artboardMain1->height());
std::vector<StateMachineInstance*> stateMachines;
auto stateMachineMain1 = artboardMain1->stateMachineAt(0);
auto stateMachineMain2 = artboardMain2->stateMachineAt(0);
auto stateMachineCommon = artboardCommon->stateMachineAt(0);
stateMachines.push_back(stateMachineMain1.get());
stateMachines.push_back(stateMachineMain2.get());
stateMachines.push_back(stateMachineCommon.get());
// Each view model instance has its own independent tree
auto vmiMain1 =
file->createViewModelInstance(artboardMain1.get()->viewModelId(), 0);
auto vmiMain2 =
file->createViewModelInstance(artboardMain2.get()->viewModelId(), 0);
auto vmiCommon =
file->createViewModelInstance(artboardCommon.get()->viewModelId(), 0);
stateMachineMain1->bindViewModelInstance(vmiMain1);
stateMachineMain1->advanceAndApply(0.1f);
stateMachineMain2->bindViewModelInstance(vmiMain2);
stateMachineMain2->advanceAndApply(0.1f);
stateMachineCommon->bindViewModelInstance(vmiCommon);
stateMachineCommon->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboardMain1->draw(renderer.get());
silver.addFrame();
artboardMain2->draw(renderer.get());
silver.addFrame();
artboardCommon->draw(renderer.get());
for (auto& sm : stateMachines)
{
silver.addFrame();
sm->advanceAndApply(0.1f);
sm->artboard()->draw(renderer.get());
}
// Changing the text value of one instance only affects that instance
auto vmCommon1Prop = vmiMain1->propertyValue("common-1")
->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
auto vmChildProp = vmCommon1Prop->propertyValue("ch")
->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance();
auto stringProp =
vmChildProp.get()->propertyValue("str")->as<ViewModelInstanceString>();
stringProp->propertyValue("update-1");
for (auto& sm : stateMachines)
{
silver.addFrame();
sm->advanceAndApply(0.1f);
sm->artboard()->draw(renderer.get());
}
// Replace the nested child view model of main2 with the same instance of
// main1 Instance 2 should update
auto vmCommonMain2Prop = vmiMain2->propertyValue("common-2")
->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
vmCommonMain2Prop->replaceViewModelByName("ch", vmChildProp);
for (auto& sm : stateMachines)
{
silver.addFrame();
sm->advanceAndApply(0.1f);
sm->artboard()->draw(renderer.get());
}
// Updating the value, changes it on both artboard
stringProp->propertyValue("update-2");
for (auto& sm : stateMachines)
{
silver.addFrame();
sm->advanceAndApply(0.1f);
sm->artboard()->draw(renderer.get());
}
// Replace the child view model of common with the same instance of
// main1
// common should update now too
auto currentChild =
vmiCommon->propertyValue("ch")->as<ViewModelInstanceViewModel>();
auto referenceViewModel = currentChild->referenceViewModelInstance();
REQUIRE(referenceViewModel->parents().size() == 1);
vmiCommon->replaceViewModelByName("ch", vmChildProp);
REQUIRE(referenceViewModel->parents().size() == 0);
REQUIRE(vmiMain1->dependents().size() == 1);
REQUIRE(vmiMain2->dependents().size() == 1);
stateMachineMain2->bindViewModelInstance(vmiMain1);
REQUIRE(vmiMain1->dependents().size() == 2);
REQUIRE(vmiMain2->dependents().size() == 0);
for (auto& sm : stateMachines)
{
silver.addFrame();
sm->advanceAndApply(0.1f);
sm->artboard()->draw(renderer.get());
}
CHECK(silver.matches("replace_vm_instance"));
}
TEST_CASE("Replace view model instance in list", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/replace_vm_instance.riv", &silver);
auto artboard = file->artboardNamed("main-list");
silver.frameSize(artboard->width(), artboard->height());
auto renderer = silver.makeRenderer();
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(artboard.get()->viewModelId(), 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Replace an instance of a list and expect the binding to be automatically
// called
silver.addFrame();
// First create a instance
auto grandChildInstance =
file->createViewModelInstance("main-list-grandchild");
auto labelProp = grandChildInstance->propertyValue("label")
->as<ViewModelInstanceString>();
labelProp->propertyValue("modified");
// Second retrieve the item from the list that will be replaced
auto lis = vmi->propertyValue("lis")->as<ViewModelInstanceList>();
{
auto listItem = lis->item(1);
auto listItemViewModelInstance = listItem->viewModelInstance();
// Third replace the property "grandchild" with the new instance created
listItemViewModelInstance->replaceViewModelByName("grandchild",
grandChildInstance);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
// Create a new child instance and append it to the list, then replace its
// child
{
silver.addFrame();
// First create instance to be added and set its label value
auto childInstance = file->createViewModelInstance("main-list-child");
auto childGrandChild = childInstance->propertyValue("grandchild")
->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
auto childGrandChildLabel = childGrandChild->propertyValue("label")
->as<ViewModelInstanceString>();
childGrandChildLabel->propertyValue("new label");
// Second append it to the list
auto listItem = make_rcp<ViewModelInstanceListItem>();
listItem->viewModelInstance(childInstance);
lis->addItem(listItem);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Replace the view model property and expect the instance to be updated
silver.addFrame();
auto grandChildInstance =
file->createViewModelInstance("main-list-grandchild");
auto labelProp = grandChildInstance->propertyValue("label")
->as<ViewModelInstanceString>();
labelProp->propertyValue("modified 2");
childInstance->replaceViewModelByName("grandchild", grandChildInstance);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("replace_vm_instance-list"));
}
TEST_CASE("Replace view model instance in nested list", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/replace_vm_instance.riv", &silver);
auto artboard = file->artboardNamed("main-double-nest");
silver.frameSize(artboard->width(), artboard->height());
auto renderer = silver.makeRenderer();
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(artboard.get()->viewModelId(), 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Replace an instance of a list and expect the binding to be automatically
// called
silver.addFrame();
// First create a instance
auto grandChildInstance =
file->createViewModelInstance("main-list-grandchild");
auto labelProp = grandChildInstance->propertyValue("label")
->as<ViewModelInstanceString>();
labelProp->propertyValue("modified");
auto mainList = vmi->propertyValue("lis")->as<ViewModelInstanceList>();
auto mainListChild = mainList->item(0);
auto mainListChildInstance = mainListChild->viewModelInstance();
// Second retrieve the item from the list that will be replaced
auto lis = mainListChildInstance->propertyValue("lis")
->as<ViewModelInstanceList>();
{
auto listItem = lis->item(1);
auto listItemViewModelInstance = listItem->viewModelInstance();
// Third replace the property "grandchild" with the new instance created
listItemViewModelInstance->replaceViewModelByName("grandchild",
grandChildInstance);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
// Create a new child instance and append it to the list, then replace its
// child
{
silver.addFrame();
// First create instance to be added and set its label value
auto childInstance = file->createViewModelInstance("main-list-child");
auto childGrandChild = childInstance->propertyValue("grandchild")
->as<ViewModelInstanceViewModel>()
->referenceViewModelInstance()
.get();
auto childGrandChildLabel = childGrandChild->propertyValue("label")
->as<ViewModelInstanceString>();
childGrandChildLabel->propertyValue("new label");
// Second append it to the list
auto listItem = make_rcp<ViewModelInstanceListItem>();
listItem->viewModelInstance(childInstance);
lis->addItem(listItem);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Replace the view model property and expect the instance to be updated
silver.addFrame();
auto grandChildInstance =
file->createViewModelInstance("main-list-grandchild");
auto labelProp = grandChildInstance->propertyValue("label")
->as<ViewModelInstanceString>();
labelProp->propertyValue("modified 2");
childInstance->replaceViewModelByName("grandchild", grandChildInstance);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("replace_vm_instance-double-nest"));
}
TEST_CASE("Pointer drag event", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/drag_event.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(artboard.get()->viewModelId(), 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
// Clicking on a square without moving will trigger the click on the nested
// artboard
stateMachine->pointerDown(rive::Vec2D(250, 250));
stateMachine->pointerUp(rive::Vec2D(250, 250));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
auto coord = 250.0f;
stateMachine->pointerDown(rive::Vec2D(coord, coord));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
// Drag gesture works and click is cancelled
while (coord > 50)
{
silver.addFrame();
stateMachine->pointerMove(rive::Vec2D(coord, coord));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
coord -= 10;
}
stateMachine->pointerUp(rive::Vec2D(coord, coord));
silver.addFrame();
// Clicking again on a square without moving will trigger the click on the
// nested artboard
stateMachine->pointerDown(rive::Vec2D(coord, coord));
stateMachine->advanceAndApply(0.1f);
stateMachine->pointerUp(rive::Vec2D(coord, coord));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("drag_event"));
}
TEST_CASE("Recursive data binding artboards are skipped", "[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/recursive_data_bind.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(artboard.get()->viewModelId(), 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(249.0f, 430.0f));
stateMachine->pointerUp(rive::Vec2D(249.0f, 430.0f));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(391.0f, 430.0f));
stateMachine->pointerUp(rive::Vec2D(391.0f, 430.0f));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
// This action will bind to a recursive artboard and should be skipped
stateMachine->pointerDown(rive::Vec2D(107.0f, 430.0f));
stateMachine->pointerUp(rive::Vec2D(107.0f, 430.0f));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(249.0f, 430.0f));
stateMachine->pointerUp(rive::Vec2D(249.0f, 430.0f));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->pointerDown(rive::Vec2D(391.0f, 430.0f));
stateMachine->pointerUp(rive::Vec2D(391.0f, 430.0f));
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("recursive_data_bind"));
}
TEST_CASE("Collapsable data binds get added when object is uncollapsed",
"[silver]")
{
SerializingFactory silver;
auto file = ReadRiveFile("assets/collapsable_data_binding.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
auto vmi = file->createViewModelInstance(artboard.get()->viewModelId(), 0);
auto soloIndexProp =
vmi->propertyValue("soloIndex")->as<ViewModelInstanceNumber>();
auto colorProp = vmi->propertyValue("col")->as<ViewModelInstanceColor>();
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.1f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
auto redColor = (255 << 24) | (255 << 16);
// Setting the red color should update both data binds although one is
// soloed
colorProp->propertyValue(redColor);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
soloIndexProp->propertyValue(1);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
auto greenColor = (255 << 24) | (255 << 8);
colorProp->propertyValue(greenColor);
soloIndexProp->propertyValue(0);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
silver.addFrame();
soloIndexProp->propertyValue(1);
stateMachine->advanceAndApply(0.1f);
artboard->draw(renderer.get());
CHECK(silver.matches("collapsable_data_binding"));
}
// Note to this test for future reference: This test is passing because when an
// artboard list is initialized, it populates its nested artboards only after
// option C has advanced a second time. This causes tryChangeState to run before
// advancing, and that allows the state to be available for the first advance to
// mix its blend values. That's not the case for nested artboard, so if any of
// these premises changes in the future, this test would catch the change.
TEST_CASE(
"Virtualized list with blended animations as initial state correctly render their mixed values",
"[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/virtualize_blendmode.riv", &silver);
auto artboard = file->artboardNamed("main");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.016f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
int frames = (int)(4.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("virtualize_blendmode"));
}
TEST_CASE("Advance two consecutive blend modes and apply inputs", "[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/advance_blend_mode.riv", &silver);
auto artboard = file->artboardNamed("main-inputs");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.0f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
CHECK(silver.matches("advance_blend_mode-inputs"));
}
TEST_CASE("Advance two consecutive blend modes and apply view model",
"[silver]")
{
RandomProvider::clearRandoms();
REQUIRE(RandomProvider::totalCalls() == 0);
SerializingFactory silver;
auto file = ReadRiveFile("assets/advance_blend_mode.riv", &silver);
auto artboard = file->artboardNamed("main-vms");
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.0f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
CHECK(silver.matches("advance_blend_mode-vms"));
}
TEST_CASE("Test State machine transition conditions based on artboards",
"[silver]")
{
SerializingFactory silver;
auto file =
ReadRiveFile("assets/transition_artboard_condition_test.riv", &silver);
auto artboard = file->artboardDefault();
REQUIRE(artboard != nullptr);
silver.frameSize(artboard->width(), artboard->height());
auto stateMachine = artboard->stateMachineAt(0);
int viewModelId = artboard.get()->viewModelId();
auto vmi = viewModelId == -1
? file->createViewModelInstance(artboard.get())
: file->createViewModelInstance(viewModelId, 0);
stateMachine->bindViewModelInstance(vmi);
stateMachine->advanceAndApply(0.0f);
auto renderer = silver.makeRenderer();
artboard->draw(renderer.get());
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
int frames = (int)(1.0f / 0.016f);
for (int i = 0; i < frames; i++)
{
silver.addFrame();
stateMachine->advanceAndApply(0.016f);
artboard->draw(renderer.get());
}
CHECK(silver.matches("transition_artboard_condition_test"));
}