blob: 5ee9df3db05d26c4e7160d7a3d047d255af57554 [file] [log] [blame]
#ifdef WITH_RIVE_SCRIPTING
#include "rive/lua/rive_lua_libs.hpp"
#endif
#include "rive/assets/script_asset.hpp"
#include "rive/component_dirt.hpp"
#include "rive/data_bind/data_bind.hpp"
#include "rive/data_bind/data_bind_context.hpp"
#include "rive/data_bind/data_values/data_type.hpp"
#include "rive/data_bind/data_values/data_value.hpp"
#include "rive/data_bind/data_values/data_value_boolean.hpp"
#include "rive/data_bind/data_values/data_value_color.hpp"
#include "rive/data_bind/data_values/data_value_number.hpp"
#include "rive/data_bind/data_values/data_value_string.hpp"
#include "rive/scripted/scripted_data_converter.hpp"
using namespace rive;
ScriptedDataConverter::~ScriptedDataConverter()
{
disposeScriptInputs();
if (m_dataValue)
{
delete m_dataValue;
}
}
void ScriptedDataConverter::disposeScriptInputs()
{
auto props = m_customProperties;
ScriptedObject::disposeScriptInputs();
for (auto prop : props)
{
auto scriptInput = ScriptInput::from(prop);
if (scriptInput != nullptr)
{
// ScriptedDataConverters need to delete their own inputs
// because they are not components
delete scriptInput;
}
}
}
#ifdef WITH_RIVE_SCRIPTING
bool ScriptedDataConverter::scriptInit(LuaState* state)
{
ScriptedObject::scriptInit(state);
addScriptedDirt(ComponentDirt::Bindings);
return true;
}
DataValue* ScriptedDataConverter::applyConversion(DataValue* value,
const std::string& method)
{
if (m_state == nullptr)
{
return value;
}
auto state = m_state->state;
rive_lua_pushRef(state, m_self);
if (static_cast<lua_Type>(lua_getfield(state, -1, method.c_str())) !=
LUA_TFUNCTION)
{
fprintf(stderr, "expected %s to be a function\n", method.c_str());
}
else
{
if (value->is<DataValueNumber>())
{
lua_pushvalue(state, -2);
lua_newrive<ScriptedDataValueNumber>(
state,
state,
value->as<DataValueNumber>()->value());
}
else if (value->is<DataValueString>())
{
lua_pushvalue(state, -2);
lua_newrive<ScriptedDataValueString>(
state,
state,
value->as<DataValueString>()->value());
}
else if (value->is<DataValueBoolean>())
{
lua_pushvalue(state, -2);
lua_newrive<ScriptedDataValueBoolean>(
state,
state,
value->as<DataValueBoolean>()->value());
}
else if (value->is<DataValueColor>())
{
lua_pushvalue(state, -2);
lua_newrive<ScriptedDataValueColor>(
state,
state,
value->as<DataValueColor>()->value());
}
if (static_cast<lua_Status>(rive_lua_pcall(state, 2, 1)) != LUA_OK)
{
rive_lua_pop(state, 1);
}
else
{
auto result = (ScriptedDataValue*)lua_touserdata(state, -1);
if (result->isNumber())
{
storeData<DataValueNumber>(result->dataValue());
}
else if (result->isString())
{
storeData<DataValueString>(result->dataValue());
}
else if (result->isBoolean())
{
storeData<DataValueBoolean>(result->dataValue());
}
else if (result->isColor())
{
storeData<DataValueColor>(result->dataValue());
}
rive_lua_pop(state, 2);
}
}
if (!m_dataValue)
{
m_dataValue = new DataValue();
}
return m_dataValue;
}
DataValue* ScriptedDataConverter::convert(DataValue* value, DataBind* dataBind)
{
return applyConversion(value, "convert");
}
DataValue* ScriptedDataConverter::reverseConvert(DataValue* value,
DataBind* dataBind)
{
return applyConversion(value, "reverseConvert");
}
#endif
void ScriptedDataConverter::bindFromContext(DataContext* dataContext,
DataBind* dataBind)
{
m_dataContext = dataContext;
Super::bindFromContext(dataContext, dataBind);
reinit();
for (auto prop : m_customProperties)
{
auto input = ScriptInput::from(prop);
if (input != nullptr)
{
if (input->dataBind() != nullptr)
{
input->dataBind()->as<DataBindContext>()->bindFromContext(
dataContext);
}
}
}
}
bool ScriptedDataConverter::advanceComponent(float elapsedSeconds,
AdvanceFlags flags)
{
if (elapsedSeconds == 0)
{
return false;
}
if ((flags & AdvanceFlags::AdvanceNested) == 0)
{
elapsedSeconds = 0;
}
return scriptAdvance(elapsedSeconds);
}
void ScriptedDataConverter::addProperty(CustomProperty* prop)
{
auto scriptInput = ScriptInput::from(prop);
if (scriptInput != nullptr)
{
scriptInput->scriptedObject(this);
}
CustomPropertyContainer::addProperty(prop);
}
StatusCode ScriptedDataConverter::import(ImportStack& importStack)
{
auto result = registerReferencer(importStack);
if (result != StatusCode::Ok)
{
return result;
}
return Super::import(importStack);
}
Core* ScriptedDataConverter::clone() const
{
ScriptedDataConverter* twin =
ScriptedDataConverterBase::clone()->as<ScriptedDataConverter>();
if (m_fileAsset != nullptr)
{
twin->setAsset(m_fileAsset);
}
for (auto prop : m_customProperties)
{
auto clonedValue = prop->clone()->as<CustomProperty>();
twin->addProperty(clonedValue);
}
return twin;
}