| 
							- /*
 -  * Carla State utils
 -  * Copyright (C) 2012-2013 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.
 -  */
 - 
 - #ifndef CARLA_STATE_UTILS_HPP_INCLUDED
 - #define CARLA_STATE_UTILS_HPP_INCLUDED
 - 
 - #include "CarlaBackendUtils.hpp"
 - #include "CarlaMIDI.h"
 - #include "List.hpp"
 - 
 - #include <QtXml/QDomNode>
 - 
 - CARLA_BACKEND_START_NAMESPACE
 - 
 - // -----------------------------------------------------------------------
 - 
 - struct StateParameter {
 -     uint32_t    index;
 -     const char* name;
 -     const char* symbol;
 -     float       value;
 -     uint8_t     midiChannel;
 -     int16_t     midiCC;
 - 
 -     StateParameter() noexcept
 -         : index(0),
 -           name(nullptr),
 -           symbol(nullptr),
 -           value(0.0f),
 -           midiChannel(0),
 -           midiCC(-1) {}
 - 
 -     ~StateParameter()
 -     {
 -         if (name != nullptr)
 -         {
 -             delete[] name;
 -             name = nullptr;
 -         }
 -         if (symbol != nullptr)
 -         {
 -             delete[] symbol;
 -             symbol = nullptr;
 -         }
 -     }
 - 
 -     CARLA_DECLARE_NON_COPY_STRUCT(StateParameter)
 - };
 - 
 - struct StateCustomData {
 -     const char* type;
 -     const char* key;
 -     const char* value;
 - 
 -     StateCustomData() noexcept
 -         : type(nullptr),
 -           key(nullptr),
 -           value(nullptr) {}
 - 
 -     ~StateCustomData()
 -     {
 -         if (type != nullptr)
 -         {
 -             delete[] type;
 -             type = nullptr;
 -         }
 -         if (key != nullptr)
 -         {
 -             delete[] key;
 -             key = nullptr;
 -         }
 -         if (value != nullptr)
 -         {
 -             delete[] value;
 -             value = nullptr;
 -         }
 -     }
 - 
 -     CARLA_DECLARE_NON_COPY_STRUCT(StateCustomData)
 - };
 - 
 - typedef List<StateParameter*> StateParameterList;
 - typedef List<StateCustomData*> StateCustomDataList;
 - 
 - typedef List<StateParameter*>::Itenerator StateParameterItenerator;
 - typedef List<StateCustomData*>::Itenerator StateCustomDataItenerator;
 - 
 - struct SaveState {
 -     const char* type;
 -     const char* name;
 -     const char* label;
 -     const char* binary;
 -     long        uniqueID;
 - 
 -     bool  active;
 -     float dryWet;
 -     float volume;
 -     float balanceLeft;
 -     float balanceRight;
 -     float panning;
 -     int8_t ctrlChannel;
 - 
 -     int32_t     currentProgramIndex;
 -     const char* currentProgramName;
 -     int32_t     currentMidiBank;
 -     int32_t     currentMidiProgram;
 -     const char* chunk;
 - 
 -     StateParameterList parameters;
 -     StateCustomDataList customData;
 - 
 -     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()
 -     {
 -         reset();
 -     }
 - 
 -     void 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();
 -     }
 - 
 -     CARLA_DECLARE_NON_COPY_STRUCT(SaveState)
 - };
 - 
 - // -----------------------------------------------------------------------
 - 
 - static inline
 - 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 inline
 - const char* xmlSafeStringCharDup(const QString& string, const bool toXml)
 - {
 -     return carla_strdup(xmlSafeString(string, toXml).toUtf8().constData());
 - }
 - 
 - // -----------------------------------------------------------------------
 - 
 - static inline
 - 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("filename", Qt::CaseInsensitive) == 0)
 -                 {
 -                     saveState.binary = xmlSafeStringCharDup(text, false);
 -                 }
 -                 else if (tag.compare("uniqueid", Qt::CaseInsensitive) == 0)
 -                 {
 -                     bool ok;
 -                     const long uniqueID(text.toLong(&ok));
 -                     if (ok) saveState.uniqueID = 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 uint index(pText.toUInt(&ok));
 -                             if (ok) 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);
 -                 }
 -             }
 -         }
 -     }
 - }
 - 
 - // -----------------------------------------------------------------------
 - 
 - static inline
 - 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("   <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_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_CSOUND:
 -         case PLUGIN_GIG:
 -         case PLUGIN_SF2:
 -         case PLUGIN_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));
 - 
 -         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
 - 
 - #endif // CARLA_STATE_UTILS_HPP_INCLUDED
 
 
  |