/*
This file is part of Ingen.
Copyright 2007-2015 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#include "ingen/client/BlockModel.hpp"
#include "ingen/URIs.hpp"
#include "ingen/World.hpp"
#include "ingen/client/PortModel.hpp"
#include
#include
#include
#include
#include
#include
namespace ingen {
namespace client {
BlockModel::BlockModel(URIs& uris,
const std::shared_ptr& plugin,
const Raul::Path& path)
: ObjectModel(uris, path)
, _plugin_uri(plugin->uri())
, _plugin(plugin)
, _num_values(0)
, _min_values(nullptr)
, _max_values(nullptr)
{
}
BlockModel::BlockModel(URIs& uris, URI plugin_uri, const Raul::Path& path)
: ObjectModel(uris, path)
, _plugin_uri(std::move(plugin_uri))
, _num_values(0)
, _min_values(nullptr)
, _max_values(nullptr)
{
}
BlockModel::BlockModel(const BlockModel& copy)
: ObjectModel(copy)
, _plugin_uri(copy._plugin_uri)
, _num_values(copy._num_values)
, _min_values(static_cast(malloc(sizeof(float) * _num_values)))
, _max_values(static_cast(malloc(sizeof(float) * _num_values)))
{
memcpy(_min_values, copy._min_values, sizeof(float) * _num_values);
memcpy(_max_values, copy._max_values, sizeof(float) * _num_values);
}
BlockModel::~BlockModel()
{
clear();
}
void
BlockModel::remove_port(const std::shared_ptr& port)
{
for (auto i = _ports.begin(); i != _ports.end(); ++i) {
if ((*i) == port) {
_ports.erase(i);
break;
}
}
_signal_removed_port.emit(port);
}
void
BlockModel::remove_port(const Raul::Path& port_path)
{
for (auto i = _ports.begin(); i != _ports.end(); ++i) {
if ((*i)->path() == port_path) {
_ports.erase(i);
break;
}
}
}
void
BlockModel::clear()
{
_ports.clear();
assert(_ports.empty());
delete[] _min_values;
delete[] _max_values;
_min_values = nullptr;
_max_values = nullptr;
}
void
BlockModel::add_child(const std::shared_ptr& c)
{
assert(c->parent().get() == this);
//ObjectModel::add_child(c);
auto pm = std::dynamic_pointer_cast(c);
assert(pm);
add_port(pm);
}
bool
BlockModel::remove_child(const std::shared_ptr& c)
{
assert(c->path().is_child_of(path()));
assert(c->parent().get() == this);
//bool ret = ObjectModel::remove_child(c);
auto pm = std::dynamic_pointer_cast(c);
assert(pm);
remove_port(pm);
//return ret;
return true;
}
void
BlockModel::add_port(const std::shared_ptr& pm)
{
assert(pm);
assert(pm->path().is_child_of(path()));
assert(pm->parent().get() == this);
// Store should have handled this by merging the two
assert(find(_ports.begin(), _ports.end(), pm) == _ports.end());
_ports.push_back(pm);
_signal_new_port.emit(pm);
}
std::shared_ptr
BlockModel::get_port(const Raul::Symbol& symbol) const
{
for (auto p : _ports) {
if (p->symbol() == symbol) {
return p;
}
}
return std::shared_ptr();
}
std::shared_ptr
BlockModel::get_port(uint32_t index) const
{
return _ports[index];
}
ingen::Node*
BlockModel::port(uint32_t index) const
{
assert(index < num_ports());
return const_cast(
dynamic_cast(_ports[index].get()));
}
void
BlockModel::default_port_value_range(
const std::shared_ptr& port,
float& min,
float& max,
uint32_t srate) const
{
// Default control values
min = 0.0;
max = 1.0;
// Get range from client-side LV2 data
if (_plugin && _plugin->lilv_plugin()) {
if (!_min_values) {
_num_values = lilv_plugin_get_num_ports(_plugin->lilv_plugin());
_min_values = new float[_num_values];
_max_values = new float[_num_values];
lilv_plugin_get_port_ranges_float(_plugin->lilv_plugin(),
_min_values, _max_values, nullptr);
}
if (!std::isnan(_min_values[port->index()])) {
min = _min_values[port->index()];
}
if (!std::isnan(_max_values[port->index()])) {
max = _max_values[port->index()];
}
}
if (port->port_property(_uris.lv2_sampleRate)) {
min *= srate;
max *= srate;
}
}
void
BlockModel::port_value_range(const std::shared_ptr& port,
float& min,
float& max,
uint32_t srate) const
{
assert(port->parent().get() == this);
default_port_value_range(port, min, max);
// Possibly overriden
const Atom& min_atom = port->get_property(_uris.lv2_minimum);
const Atom& max_atom = port->get_property(_uris.lv2_maximum);
if (min_atom.type() == _uris.forge.Float) {
min = min_atom.get();
}
if (max_atom.type() == _uris.forge.Float) {
max = max_atom.get();
}
if (max <= min) {
max = min + 1.0f;
}
if (port->port_property(_uris.lv2_sampleRate)) {
min *= srate;
max *= srate;
}
}
std::string
BlockModel::label() const
{
const Atom& name_property = get_property(_uris.lv2_name);
if (name_property.type() == _uris.forge.String) {
return name_property.ptr();
} else if (plugin_model()) {
return plugin_model()->human_name();
} else {
return symbol().c_str();
}
}
std::string
BlockModel::port_label(const std::shared_ptr& port) const
{
const Atom& name = port->get_property(URI(LV2_CORE__name));
if (name.is_valid() && name.type() == _uris.forge.String) {
return name.ptr();
}
if (_plugin && _plugin->lilv_plugin()) {
LilvWorld* w = _plugin->lilv_world();
const LilvPlugin* plug = _plugin->lilv_plugin();
LilvNode* sym = lilv_new_string(w, port->symbol().c_str());
const LilvPort* lport = lilv_plugin_get_port_by_symbol(plug, sym);
if (lport) {
LilvNode* lname = lilv_port_get_name(plug, lport);
if (lname && lilv_node_is_string(lname)) {
std::string ret(lilv_node_as_string(lname));
lilv_node_free(lname);
return ret;
}
lilv_node_free(lname);
}
}
return port->symbol().c_str();
}
void
BlockModel::set(const std::shared_ptr& model)
{
auto block = std::dynamic_pointer_cast(model);
if (block) {
_plugin_uri = block->_plugin_uri;
_plugin = block->_plugin;
}
ObjectModel::set(model);
}
} // namespace client
} // namespace ingen