|
- /*
- * Carla State utils
- * Copyright (C) 2012-2014 Filipe Coelho <falktx@falktx.com>
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of
- * the License, or any later version.
- *
- * This program 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 General Public License for more details.
- *
- * For a full copy of the GNU General Public License see the doc/GPL.txt file.
- */
-
- #include "CarlaStateUtils.hpp"
-
- #include "CarlaBackendUtils.hpp"
- #include "CarlaMathUtils.hpp"
- #include "CarlaMIDI.h"
-
- #include <QtCore/QString>
- #include <QtXml/QDomNode>
-
- CARLA_BACKEND_START_NAMESPACE
-
- // -----------------------------------------------------------------------
- // StateParameter
-
- StateParameter::StateParameter() noexcept
- : isInput(true),
- index(-1),
- name(nullptr),
- symbol(nullptr),
- value(0.0f),
- midiChannel(0),
- midiCC(-1) {}
-
- StateParameter::~StateParameter()
- {
- if (name != nullptr)
- {
- delete[] name;
- name = nullptr;
- }
- if (symbol != nullptr)
- {
- delete[] symbol;
- symbol = nullptr;
- }
- }
-
- // -----------------------------------------------------------------------
- // StateCustomData
-
- StateCustomData::StateCustomData() noexcept
- : type(nullptr),
- key(nullptr),
- value(nullptr) {}
-
- StateCustomData::~StateCustomData()
- {
- if (type != nullptr)
- {
- delete[] type;
- type = nullptr;
- }
- if (key != nullptr)
- {
- delete[] key;
- key = nullptr;
- }
- if (value != nullptr)
- {
- delete[] value;
- value = nullptr;
- }
- }
-
- // -----------------------------------------------------------------------
- // SaveState
-
- SaveState::SaveState() noexcept
- : type(nullptr),
- name(nullptr),
- label(nullptr),
- binary(nullptr),
- uniqueId(0),
- active(false),
- dryWet(1.0f),
- volume(1.0f),
- balanceLeft(-1.0f),
- balanceRight(1.0f),
- panning(0.0f),
- ctrlChannel(-1),
- currentProgramIndex(-1),
- currentProgramName(nullptr),
- currentMidiBank(-1),
- currentMidiProgram(-1),
- chunk(nullptr) {}
-
- SaveState::~SaveState()
- {
- reset();
- }
-
- void SaveState::reset()
- {
- if (type != nullptr)
- {
- delete[] type;
- type = nullptr;
- }
- if (name != nullptr)
- {
- delete[] name;
- name = nullptr;
- }
- if (label != nullptr)
- {
- delete[] label;
- label = nullptr;
- }
- if (binary != nullptr)
- {
- delete[] binary;
- binary = nullptr;
- }
- if (currentProgramName != nullptr)
- {
- delete[] currentProgramName;
- currentProgramName = nullptr;
- }
- if (chunk != nullptr)
- {
- delete[] chunk;
- chunk = nullptr;
- }
-
- uniqueId = 0;
- active = false;
- dryWet = 1.0f;
- volume = 1.0f;
- balanceLeft = -1.0f;
- balanceRight = 1.0f;
- panning = 0.0f;
- ctrlChannel = -1;
- currentProgramIndex = -1;
- currentMidiBank = -1;
- currentMidiProgram = -1;
-
- for (StateParameterItenerator it = parameters.begin(); it.valid(); it.next())
- {
- StateParameter* const stateParameter(it.getValue());
- delete stateParameter;
- }
-
- for (StateCustomDataItenerator it = customData.begin(); it.valid(); it.next())
- {
- StateCustomData* const stateCustomData(it.getValue());
- delete stateCustomData;
- }
-
- parameters.clear();
- customData.clear();
- }
-
- // -----------------------------------------------------------------------
- // xmlSafeString
-
- static QString xmlSafeString(const QString& string, const bool toXml)
- {
- QString newString(string);
-
- if (toXml)
- return newString.replace("&","&").replace("<","<").replace(">",">").replace("'","'").replace("\"",""");
- else
- return newString.replace("&","&").replace("<","<").replace(">",">").replace("'","'").replace(""","\"");
- }
-
- static const char* xmlSafeStringCharDup(const QString& string, const bool toXml)
- {
- return carla_strdup(xmlSafeString(string, toXml).toUtf8().constData());
- }
-
- // -----------------------------------------------------------------------
- // fillSaveStateFromXmlNode
-
- void fillSaveStateFromXmlNode(SaveState& saveState, const QDomNode& xmlNode)
- {
- if (xmlNode.isNull())
- return;
-
- for (QDomNode node = xmlNode.firstChild(); ! node.isNull(); node = node.nextSibling())
- {
- QString tagName(node.toElement().tagName());
-
- // ---------------------------------------------------------------
- // Info
-
- if (tagName.compare("info", Qt::CaseInsensitive) == 0)
- {
- for (QDomNode xmlInfo = node.toElement().firstChild(); ! xmlInfo.isNull(); xmlInfo = xmlInfo.nextSibling())
- {
- const QString tag(xmlInfo.toElement().tagName());
- const QString text(xmlInfo.toElement().text().trimmed());
-
- if (tag.compare("type", Qt::CaseInsensitive) == 0)
- {
- saveState.type = xmlSafeStringCharDup(text, false);
- }
- else if (tag.compare("name", Qt::CaseInsensitive) == 0)
- {
- saveState.name = xmlSafeStringCharDup(text, false);
- }
- else if (tag.compare("label", Qt::CaseInsensitive) == 0 || tag.compare("uri", Qt::CaseInsensitive) == 0)
- {
- saveState.label = xmlSafeStringCharDup(text, false);
- }
- else if (tag.compare("binary", Qt::CaseInsensitive) == 0 || tag.compare("bundle", Qt::CaseInsensitive) == 0 || tag.compare("filename", Qt::CaseInsensitive) == 0)
- {
- saveState.binary = xmlSafeStringCharDup(text, false);
- }
- else if (tag.compare("uniqueid", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const qlonglong uniqueId(text.toLongLong(&ok));
- if (ok) saveState.uniqueId = static_cast<int64_t>(uniqueId);
- }
- }
- }
-
- // ---------------------------------------------------------------
- // Data
-
- else if (tagName.compare("data", Qt::CaseInsensitive) == 0)
- {
- for (QDomNode xmlData = node.toElement().firstChild(); ! xmlData.isNull(); xmlData = xmlData.nextSibling())
- {
- const QString tag(xmlData.toElement().tagName());
- const QString text(xmlData.toElement().text().trimmed());
-
- // -------------------------------------------------------
- // Internal Data
-
- if (tag.compare("active", Qt::CaseInsensitive) == 0)
- {
- saveState.active = (text.compare("yes", Qt::CaseInsensitive) == 0 || text.compare("true", Qt::CaseInsensitive) == 0);
- }
- else if (tag.compare("drywet", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(text.toFloat(&ok));
- if (ok) saveState.dryWet = carla_fixValue(0.0f, 1.0f, value);
- }
- else if (tag.compare("volume", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(text.toFloat(&ok));
- if (ok) saveState.volume = carla_fixValue(0.0f, 1.27f, value);
- }
- else if (tag.compare("balanceleft", Qt::CaseInsensitive) == 0 || tag.compare("balance-left", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(text.toFloat(&ok));
- if (ok) saveState.balanceLeft = carla_fixValue(-1.0f, 1.0f, value);
- }
- else if (tag.compare("balanceright", Qt::CaseInsensitive) == 0 || tag.compare("balance-right", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(text.toFloat(&ok));
- if (ok) saveState.balanceRight = carla_fixValue(-1.0f, 1.0f, value);
- }
- else if (tag.compare("panning", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(text.toFloat(&ok));
- if (ok) saveState.panning = carla_fixValue(-1.0f, 1.0f, value);
- }
- else if (tag.compare("controlchannel", Qt::CaseInsensitive) == 0 || tag.compare("control-channel", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const short value(text.toShort(&ok));
- if (ok && value >= 1 && value < MAX_MIDI_CHANNELS)
- saveState.ctrlChannel = static_cast<int8_t>(value-1);
- }
-
- // -------------------------------------------------------
- // Program (current)
-
- else if (tag.compare("currentprogramindex", Qt::CaseInsensitive) == 0 || tag.compare("current-program-index", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const int value(text.toInt(&ok));
- if (ok && value >= 1)
- saveState.currentProgramIndex = value-1;
- }
- else if (tag.compare("currentprogramname", Qt::CaseInsensitive) == 0 || tag.compare("current-program-name", Qt::CaseInsensitive) == 0)
- {
- saveState.currentProgramName = xmlSafeStringCharDup(text, false);
- }
-
- // -------------------------------------------------------
- // Midi Program (current)
-
- else if (tag.compare("currentmidibank", Qt::CaseInsensitive) == 0 || tag.compare("current-midi-bank", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const int value(text.toInt(&ok));
- if (ok && value >= 1)
- saveState.currentMidiBank = value-1;
- }
- else if (tag.compare("currentmidiprogram", Qt::CaseInsensitive) == 0 || tag.compare("current-midi-program", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const int value(text.toInt(&ok));
- if (ok && value >= 1)
- saveState.currentMidiProgram = value-1;
- }
-
- // -------------------------------------------------------
- // Parameters
-
- else if (tag.compare("parameter", Qt::CaseInsensitive) == 0)
- {
- StateParameter* const stateParameter(new StateParameter());
-
- for (QDomNode xmlSubData = xmlData.toElement().firstChild(); ! xmlSubData.isNull(); xmlSubData = xmlSubData.nextSibling())
- {
- const QString pTag(xmlSubData.toElement().tagName());
- const QString pText(xmlSubData.toElement().text().trimmed());
-
- if (pTag.compare("index", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const int index(pText.toInt(&ok));
- if (ok && index >= 0) stateParameter->index = index;
- }
- else if (pTag.compare("name", Qt::CaseInsensitive) == 0)
- {
- stateParameter->name = xmlSafeStringCharDup(pText, false);
- }
- else if (pTag.compare("symbol", Qt::CaseInsensitive) == 0)
- {
- stateParameter->symbol = xmlSafeStringCharDup(pText, false);
- }
- else if (pTag.compare("value", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const float value(pText.toFloat(&ok));
- if (ok) stateParameter->value = value;
- }
- else if (pTag.compare("midichannel", Qt::CaseInsensitive) == 0 || pTag.compare("midi-channel", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const ushort channel(pText.toUShort(&ok));
- if (ok && channel >= 1 && channel < MAX_MIDI_CHANNELS)
- stateParameter->midiChannel = static_cast<uint8_t>(channel-1);
- }
- else if (pTag.compare("midicc", Qt::CaseInsensitive) == 0 || pTag.compare("midi-cc", Qt::CaseInsensitive) == 0)
- {
- bool ok;
- const int cc(pText.toInt(&ok));
- if (ok && cc >= 1 && cc < 0x5F)
- stateParameter->midiCC = static_cast<int16_t>(cc);
- }
- }
-
- saveState.parameters.append(stateParameter);
- }
-
- // -------------------------------------------------------
- // Custom Data
-
- else if (tag.compare("customdata", Qt::CaseInsensitive) == 0 || tag.compare("custom-data", Qt::CaseInsensitive) == 0)
- {
- StateCustomData* const stateCustomData(new StateCustomData());
-
- for (QDomNode xmlSubData = xmlData.toElement().firstChild(); ! xmlSubData.isNull(); xmlSubData = xmlSubData.nextSibling())
- {
- const QString cTag(xmlSubData.toElement().tagName());
- const QString cText(xmlSubData.toElement().text().trimmed());
-
- if (cTag.compare("type", Qt::CaseInsensitive) == 0)
- stateCustomData->type = xmlSafeStringCharDup(cText, false);
- else if (cTag.compare("key", Qt::CaseInsensitive) == 0)
- stateCustomData->key = xmlSafeStringCharDup(cText, false);
- else if (cTag.compare("value", Qt::CaseInsensitive) == 0)
- stateCustomData->value = xmlSafeStringCharDup(cText, false);
- }
-
- saveState.customData.append(stateCustomData);
- }
-
- // -------------------------------------------------------
- // Chunk
-
- else if (tag.compare("chunk", Qt::CaseInsensitive) == 0)
- {
- saveState.chunk = xmlSafeStringCharDup(text, false);
- }
- }
- }
- }
- }
-
- // -----------------------------------------------------------------------
- // fillXmlStringFromSaveState
-
- void fillXmlStringFromSaveState(QString& content, const SaveState& saveState)
- {
- {
- QString info(" <Info>\n");
-
- info += QString(" <Type>%1</Type>\n").arg(saveState.type);
- info += QString(" <Name>%1</Name>\n").arg(xmlSafeString(saveState.name, true));
-
- switch (getPluginTypeFromString(saveState.type))
- {
- case PLUGIN_NONE:
- break;
- case PLUGIN_INTERNAL:
- info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
- break;
- case PLUGIN_LADSPA:
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
- info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueId);
- break;
- case PLUGIN_DSSI:
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
- break;
- case PLUGIN_LV2:
- info += QString(" <Bundle>%1</Bundle>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <URI>%1</URI>\n").arg(xmlSafeString(saveState.label, true));
- break;
- case PLUGIN_VST:
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueId);
- break;
- case PLUGIN_VST3:
- // TODO?
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueId);
- break;
- case PLUGIN_AU:
- // TODO?
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <UniqueID>%1</UniqueID>\n").arg(saveState.uniqueId);
- break;
- case PLUGIN_JACK:
- info += QString(" <Binary>%1</Binary>\n").arg(xmlSafeString(saveState.binary, true));
- break;
- case PLUGIN_REWIRE:
- info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
- break;
- case PLUGIN_FILE_CSD:
- case PLUGIN_FILE_GIG:
- case PLUGIN_FILE_SF2:
- case PLUGIN_FILE_SFZ:
- info += QString(" <Filename>%1</Filename>\n").arg(xmlSafeString(saveState.binary, true));
- info += QString(" <Label>%1</Label>\n").arg(xmlSafeString(saveState.label, true));
- break;
- }
-
- info += " </Info>\n\n";
-
- content += info;
- }
-
- {
- QString data(" <Data>\n");
-
- data += QString(" <Active>%1</Active>\n").arg(saveState.active ? "Yes" : "No");
-
- if (saveState.dryWet != 1.0f)
- data += QString(" <DryWet>%1</DryWet>\n").arg(saveState.dryWet);
- if (saveState.volume != 1.0f)
- data += QString(" <Volume>%1</Volume>\n").arg(saveState.volume);
- if (saveState.balanceLeft != -1.0f)
- data += QString(" <Balance-Left>%1</Balance-Left>\n").arg(saveState.balanceLeft);
- if (saveState.balanceRight != 1.0f)
- data += QString(" <Balance-Right>%1</Balance-Right>\n").arg(saveState.balanceRight);
- if (saveState.panning != 0.0f)
- data += QString(" <Panning>%1</Panning>\n").arg(saveState.panning);
-
- if (saveState.ctrlChannel < 0)
- data += QString(" <ControlChannel>N</ControlChannel>\n");
- else
- data += QString(" <ControlChannel>%1</ControlChannel>\n").arg(saveState.ctrlChannel+1);
-
- content += data;
- }
-
- for (StateParameterItenerator it = saveState.parameters.begin(); it.valid(); it.next())
- {
- StateParameter* const stateParameter(it.getValue());
-
- QString parameter("\n"" <Parameter>\n");
-
- parameter += QString(" <Index>%1</Index>\n").arg(stateParameter->index);
- parameter += QString(" <Name>%1</Name>\n").arg(xmlSafeString(stateParameter->name, true));
-
- if (stateParameter->symbol != nullptr && stateParameter->symbol[0] != '\0')
- parameter += QString(" <Symbol>%1</Symbol>\n").arg(xmlSafeString(stateParameter->symbol, true));
-
- if (stateParameter->isInput)
- parameter += QString(" <Value>%1</Value>\n").arg(stateParameter->value);
-
- if (stateParameter->midiCC > 0)
- {
- parameter += QString(" <MidiCC>%1</MidiCC>\n").arg(stateParameter->midiCC);
- parameter += QString(" <MidiChannel>%1</MidiChannel>\n").arg(stateParameter->midiChannel+1);
- }
-
- parameter += " </Parameter>\n";
-
- content += parameter;
- }
-
- if (saveState.currentProgramIndex >= 0 && saveState.currentProgramName != nullptr && saveState.currentProgramName[0] != '\0')
- {
- // ignore 'default' program
- if (saveState.currentProgramIndex > 0 || QString(saveState.currentProgramName).compare("default", Qt::CaseInsensitive) != 0)
- {
- QString program("\n");
- program += QString(" <CurrentProgramIndex>%1</CurrentProgramIndex>\n").arg(saveState.currentProgramIndex+1);
- program += QString(" <CurrentProgramName>%1</CurrentProgramName>\n").arg(xmlSafeString(saveState.currentProgramName, true));
-
- content += program;
- }
- }
-
- if (saveState.currentMidiBank >= 0 && saveState.currentMidiProgram >= 0)
- {
- QString midiProgram("\n");
- midiProgram += QString(" <CurrentMidiBank>%1</CurrentMidiBank>\n").arg(saveState.currentMidiBank+1);
- midiProgram += QString(" <CurrentMidiProgram>%1</CurrentMidiProgram>\n").arg(saveState.currentMidiProgram+1);
-
- content += midiProgram;
- }
-
- for (StateCustomDataItenerator it = saveState.customData.begin(); it.valid(); it.next())
- {
- StateCustomData* const stateCustomData(it.getValue());
-
- QString customData("\n"" <CustomData>\n");
- customData += QString(" <Type>%1</Type>\n").arg(xmlSafeString(stateCustomData->type, true));
- customData += QString(" <Key>%1</Key>\n").arg(xmlSafeString(stateCustomData->key, true));
-
- if (std::strcmp(stateCustomData->type, CUSTOM_DATA_TYPE_CHUNK) == 0 || std::strlen(stateCustomData->value) >= 128)
- {
- customData += " <Value>\n";
- customData += QString("%1\n").arg(xmlSafeString(stateCustomData->value, true));
- customData += " </Value>\n";
- }
- else
- customData += QString(" <Value>%1</Value>\n").arg(xmlSafeString(stateCustomData->value, true));
-
- customData += " </CustomData>\n";
-
- content += customData;
- }
-
- if (saveState.chunk != nullptr && saveState.chunk[0] != '\0')
- {
- QString chunk("\n"" <Chunk>\n");
- chunk += QString("%1\n").arg(saveState.chunk);
- chunk += " </Chunk>\n";
-
- content += chunk;
- }
-
- content += " </Data>\n";
- }
-
- // -----------------------------------------------------------------------
-
- CARLA_BACKEND_END_NAMESPACE
|