fix crash when copy'ing latent plugins
[ardour.git] / libs / ardour / event_type_map.cc
index 2d724a26efdafb799f724e8baa624ccf91f3f022..28f688df8124a0f656decbfb853c1a403987e60b 100644 (file)
@@ -1,6 +1,6 @@
 /*
     Copyright (C) 2008 Paul Davis
-    Author: Dave Robillard
+    Author: David Robillard
 
     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
 
 */
 
-#include <ardour/types.h>
-#include <ardour/event_type_map.h>
-#include <evoral/Parameter.hpp>
-#include <evoral/midi_events.h>
+#include <ctype.h>
+#include <cstdio>
+#include "ardour/types.h"
+#include "ardour/event_type_map.h"
+#include "ardour/parameter_descriptor.h"
+#include "ardour/parameter_types.h"
+#include "ardour/uri_map.h"
+#include "evoral/Parameter.hpp"
+#include "evoral/ParameterDescriptor.hpp"
+#include "evoral/midi_events.h"
+#include "pbd/error.h"
+#include "pbd/compose.h"
+
+using namespace std;
 
 namespace ARDOUR {
 
-EventTypeMap EventTypeMap::event_type_map;
+EventTypeMap* EventTypeMap::event_type_map;
+
+EventTypeMap&
+EventTypeMap::instance()
+{
+       if (!EventTypeMap::event_type_map) {
+#ifdef LV2_SUPPORT
+               EventTypeMap::event_type_map = new EventTypeMap(&URIMap::instance());
+#else
+               EventTypeMap::event_type_map = new EventTypeMap(NULL);
+#endif
+       }
+       return *EventTypeMap::event_type_map;
+}
 
 bool
 EventTypeMap::type_is_midi(uint32_t type) const
 {
-       return (type >= MidiCCAutomation) && (type <= MidiChannelPressureAutomation);
+       return ARDOUR::parameter_is_midi((AutomationType)type);
 }
 
 uint8_t
 EventTypeMap::parameter_midi_type(const Evoral::Parameter& param) const
 {
-       switch (param.type()) {
-       case MidiCCAutomation:              return MIDI_CMD_CONTROL; break; 
-       case MidiPgmChangeAutomation:       return MIDI_CMD_PGM_CHANGE; break; 
-       case MidiChannelPressureAutomation: return MIDI_CMD_CHANNEL_PRESSURE; break; 
-       case MidiPitchBenderAutomation:     return MIDI_CMD_BENDER; break; 
-       default: return 0;
-       }
+       return ARDOUR::parameter_midi_type((AutomationType)param.type());
 }
 
 uint32_t
 EventTypeMap::midi_event_type(uint8_t status) const
 {
-       switch (status & 0xF0) {
-       case MIDI_CMD_CONTROL:          return MidiCCAutomation; break;
-       case MIDI_CMD_PGM_CHANGE:       return MidiPgmChangeAutomation; break;
-       case MIDI_CMD_CHANNEL_PRESSURE: return MidiChannelPressureAutomation; break;
-       case MIDI_CMD_BENDER:           return MidiPitchBenderAutomation; break;
-       default: return 0;
+       return (uint32_t)ARDOUR::midi_parameter_type(status);
+}
+
+Evoral::ControlList::InterpolationStyle
+EventTypeMap::interpolation_of(const Evoral::Parameter& param)
+{
+       switch (param.type()) {
+       case MidiCCAutomation:
+               switch (param.id()) {
+               case MIDI_CTL_LSB_BANK:
+               case MIDI_CTL_MSB_BANK:
+               case MIDI_CTL_LSB_EFFECT1:
+               case MIDI_CTL_LSB_EFFECT2:
+               case MIDI_CTL_MSB_EFFECT1:
+               case MIDI_CTL_MSB_EFFECT2:
+               case MIDI_CTL_MSB_GENERAL_PURPOSE1:
+               case MIDI_CTL_MSB_GENERAL_PURPOSE2:
+               case MIDI_CTL_MSB_GENERAL_PURPOSE3:
+               case MIDI_CTL_MSB_GENERAL_PURPOSE4:
+               case MIDI_CTL_SUSTAIN:
+               case MIDI_CTL_PORTAMENTO:
+               case MIDI_CTL_SOSTENUTO:
+               case MIDI_CTL_SOFT_PEDAL:
+               case MIDI_CTL_LEGATO_FOOTSWITCH:
+               case MIDI_CTL_HOLD2:
+               case MIDI_CTL_GENERAL_PURPOSE5:
+               case MIDI_CTL_GENERAL_PURPOSE6:
+               case MIDI_CTL_GENERAL_PURPOSE7:
+               case MIDI_CTL_GENERAL_PURPOSE8:
+               case MIDI_CTL_DATA_INCREMENT:
+               case MIDI_CTL_DATA_DECREMENT:
+               case MIDI_CTL_NONREG_PARM_NUM_LSB:
+               case MIDI_CTL_NONREG_PARM_NUM_MSB:
+               case MIDI_CTL_REGIST_PARM_NUM_LSB:
+               case MIDI_CTL_REGIST_PARM_NUM_MSB:
+               case MIDI_CTL_ALL_SOUNDS_OFF:
+               case MIDI_CTL_RESET_CONTROLLERS:
+               case MIDI_CTL_LOCAL_CONTROL_SWITCH:
+               case MIDI_CTL_ALL_NOTES_OFF:
+               case MIDI_CTL_OMNI_OFF:
+               case MIDI_CTL_OMNI_ON:
+               case MIDI_CTL_MONO:
+               case MIDI_CTL_POLY:
+                       return Evoral::ControlList::Discrete; break;
+               default:
+                       return Evoral::ControlList::Linear; break;
+               }
+               break;
+       case MidiPgmChangeAutomation:       return Evoral::ControlList::Discrete; break;
+       case MidiChannelPressureAutomation: return Evoral::ControlList::Linear; break;
+       case MidiPitchBenderAutomation:     return Evoral::ControlList::Linear; break;
+       default: assert(false);
        }
+       return Evoral::ControlList::Linear; // Not reached, suppress warnings
 }
 
-bool
-EventTypeMap::is_integer(const Evoral::Parameter& param) const
+Evoral::Parameter
+EventTypeMap::from_symbol(const string& str) const
+{
+       AutomationType p_type    = NullAutomation;
+       uint8_t        p_channel = 0;
+       uint32_t       p_id      = 0;
+
+       if (str == "gain") {
+               p_type = GainAutomation;
+       } else if (str == "trim") {
+               p_type = TrimAutomation;
+       } else if (str == "solo") {
+               p_type = SoloAutomation;
+       } else if (str == "mute") {
+               p_type = MuteAutomation;
+       } else if (str == "fadein") {
+               p_type = FadeInAutomation;
+       } else if (str == "fadeout") {
+               p_type = FadeOutAutomation;
+       } else if (str == "envelope") {
+               p_type = EnvelopeAutomation;
+       } else if (str == "pan-azimuth") {
+               p_type = PanAzimuthAutomation;
+       } else if (str == "pan-width") {
+               p_type = PanWidthAutomation;
+       } else if (str == "pan-elevation") {
+               p_type = PanElevationAutomation;
+       } else if (str == "pan-frontback") {
+               p_type = PanFrontBackAutomation;
+       } else if (str == "pan-lfe") {
+               p_type = PanLFEAutomation;
+       } else if (str.length() > 10 && str.substr(0, 10) == "parameter-") {
+               p_type = PluginAutomation;
+               p_id = atoi(str.c_str()+10);
+#ifdef LV2_SUPPORT
+       } else if (str.length() > 9 && str.substr(0, 9) == "property-") {
+               p_type = PluginPropertyAutomation;
+               const char* name = str.c_str() + 9;
+               if (isdigit(str.c_str()[0])) {
+                       p_id = atoi(name);
+               } else {
+                       p_id = _uri_map->uri_to_id(name);
+               }
+#endif
+       } else if (str.length() > 7 && str.substr(0, 7) == "midicc-") {
+               p_type = MidiCCAutomation;
+               uint32_t channel = 0;
+               sscanf(str.c_str(), "midicc-%d-%d", &channel, &p_id);
+               assert(channel < 16);
+               p_channel = channel;
+       } else if (str.length() > 16 && str.substr(0, 16) == "midi-pgm-change-") {
+               p_type = MidiPgmChangeAutomation;
+               uint32_t channel = 0;
+               sscanf(str.c_str(), "midi-pgm-change-%d", &channel);
+               assert(channel < 16);
+               p_id = 0;
+               p_channel = channel;
+       } else if (str.length() > 18 && str.substr(0, 18) == "midi-pitch-bender-") {
+               p_type = MidiPitchBenderAutomation;
+               uint32_t channel = 0;
+               sscanf(str.c_str(), "midi-pitch-bender-%d", &channel);
+               assert(channel < 16);
+               p_id = 0;
+               p_channel = channel;
+       } else if (str.length() > 22 && str.substr(0, 22) == "midi-channel-pressure-") {
+               p_type = MidiChannelPressureAutomation;
+               uint32_t channel = 0;
+               sscanf(str.c_str(), "midi-channel-pressure-%d", &channel);
+               assert(channel < 16);
+               p_id = 0;
+               p_channel = channel;
+       } else {
+               PBD::warning << "Unknown Parameter '" << str << "'" << endmsg;
+       }
+
+       return Evoral::Parameter(p_type, p_channel, p_id);
+}
+
+/** Unique string representation, suitable as an XML property value.
+ * e.g. <AutomationList automation-id="whatthisreturns">
+ */
+std::string
+EventTypeMap::to_symbol(const Evoral::Parameter& param) const
+{
+       AutomationType t = (AutomationType)param.type();
+
+       if (t == GainAutomation) {
+               return "gain";
+       } else if (t == TrimAutomation) {
+                return "trim";
+       } else if (t == PanAzimuthAutomation) {
+                return "pan-azimuth";
+       } else if (t == PanElevationAutomation) {
+                return "pan-elevation";
+       } else if (t == PanWidthAutomation) {
+                return "pan-width";
+       } else if (t == PanFrontBackAutomation) {
+                return "pan-frontback";
+       } else if (t == PanLFEAutomation) {
+                return "pan-lfe";
+       } else if (t == SoloAutomation) {
+               return "solo";
+       } else if (t == MuteAutomation) {
+               return "mute";
+       } else if (t == FadeInAutomation) {
+               return "fadein";
+       } else if (t == FadeOutAutomation) {
+               return "fadeout";
+       } else if (t == EnvelopeAutomation) {
+               return "envelope";
+       } else if (t == PluginAutomation) {
+               return string_compose("parameter-%1", param.id());
+#ifdef LV2_SUPPORT
+       } else if (t == PluginPropertyAutomation) {
+               const char* uri = _uri_map->id_to_uri(param.id());
+               if (uri) {
+                       return string_compose("property-%1", uri);
+               } else {
+                       return string_compose("property-%1", param.id());
+               }
+#endif
+       } else if (t == MidiCCAutomation) {
+               return string_compose("midicc-%1-%2", int(param.channel()), param.id());
+       } else if (t == MidiPgmChangeAutomation) {
+               return string_compose("midi-pgm-change-%1", int(param.channel()));
+       } else if (t == MidiPitchBenderAutomation) {
+               return string_compose("midi-pitch-bender-%1", int(param.channel()));
+       } else if (t == MidiChannelPressureAutomation) {
+               return string_compose("midi-channel-pressure-%1", int(param.channel()));
+       } else {
+               PBD::warning << "Uninitialized Parameter symbol() called." << endmsg;
+               return "";
+       }
+}
+
+Evoral::ParameterDescriptor
+EventTypeMap::descriptor(const Evoral::Parameter& param) const
+{
+       // Found an existing (perhaps custom) descriptor
+       Descriptors::const_iterator d = _descriptors.find(param);
+       if (d != _descriptors.end()) {
+               return d->second;
+       }
+
+       // Add default descriptor and return that
+       return ARDOUR::ParameterDescriptor(param);
+}
+
+void
+EventTypeMap::set_descriptor(const Evoral::Parameter&           param,
+                             const Evoral::ParameterDescriptor& desc)
 {
-       return (   param.type() >= MidiCCAutomation
-                       && param.type() <= MidiChannelPressureAutomation);
+       _descriptors.insert(std::make_pair(param, desc));
 }
 
 } // namespace ARDOUR