Allow to dynamically un/load Midnam Patches
[ardour.git] / libs / ardour / ardour / midi_patch_manager.h
index 10d0f99a7ad83b6d67fb25c86d150fe950d2d26d..aa4fda108e071b5fe94a0fc0810c3cb730381d4b 100644 (file)
@@ -1,5 +1,5 @@
 /*
-    Copyright (C) 2008 Hans Baier 
+    Copyright (C) 2008 Hans Baier
 
     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
 #ifndef MIDI_PATCH_MANAGER_H_
 #define MIDI_PATCH_MANAGER_H_
 
-#include <midi++/midnam_patch.h>
+#include "midi++/midnam_patch.h"
 
-namespace ARDOUR {
-       class Session;
-}
+#include "pbd/signals.h"
+#include "pbd/search_path.h"
+
+#include "ardour/libardour_visibility.h"
 
 namespace MIDI
 {
@@ -33,117 +34,133 @@ namespace MIDI
 namespace Name
 {
 
-class MidiPatchManager
+class LIBARDOUR_API MidiPatchManager
 {
        /// Singleton
 private:
-       MidiPatchManager() {};
+       MidiPatchManager();
        MidiPatchManager( const MidiPatchManager& );
        MidiPatchManager& operator= (const MidiPatchManager&);
-       
-       static MidiPatchManager* _manager; 
-       
+
+       static MidiPatchManager* _manager;
+
 public:
-       typedef std::map<std::string, boost::shared_ptr<MIDINameDocument> > MidiNameDocuments;
-       
+       typedef std::map<std::string, boost::shared_ptr<MIDINameDocument> >    MidiNameDocuments;
+       typedef std::map<std::string, MIDINameDocument::MasterDeviceNamesList> DeviceNamesByMaker;
+
        virtual ~MidiPatchManager() { _manager = 0; }
-       
-       static MidiPatchManager& instance() { 
+
+       static MidiPatchManager& instance() {
                if (_manager == 0) {
                        _manager = new MidiPatchManager();
                }
-               return *_manager; 
+               return *_manager;
        }
-       
-       void set_session (ARDOUR::Session&);
-       
-       boost::shared_ptr<MIDINameDocument> document_by_model(std::string model_name) 
-               { return _documents[model_name]; }
-       
-       boost::shared_ptr<MasterDeviceNames> master_device_by_model(std::string model_name) 
+
+       PBD::Signal0<void> PatchesChanged;
+
+       bool add_custom_midnam (const std::string& id, const std::string& midnam);
+       bool remove_custom_midnam (const std::string& id);
+
+       void add_search_path (const PBD::Searchpath& search_path);
+
+       void remove_search_path (const PBD::Searchpath& search_path);
+
+       boost::shared_ptr<MIDINameDocument> document_by_model(std::string model_name) const;
+
+       boost::shared_ptr<MasterDeviceNames> master_device_by_model(std::string model_name)
                { return _master_devices_by_model[model_name]; }
-       
+
        boost::shared_ptr<ChannelNameSet> find_channel_name_set(
-                       string model, 
-                       string custom_device_mode, 
+                       std::string model,
+                       std::string custom_device_mode,
                        uint8_t channel) {
                boost::shared_ptr<MIDI::Name::MasterDeviceNames> master_device = master_device_by_model(model);
-               
+
                if (master_device != 0 && custom_device_mode != "") {
-                       return master_device->
-                                channel_name_set_by_device_mode_and_channel(custom_device_mode, channel);
+                       return master_device->channel_name_set_by_channel(custom_device_mode, channel);
                } else {
                        return boost::shared_ptr<ChannelNameSet>();
-               }               
+               }
        }
 
-       
        boost::shared_ptr<Patch> find_patch(
-                       string model, 
-                       string custom_device_mode, 
-                       uint8_t channel, 
+                       std::string model,
+                       std::string custom_device_mode,
+                       uint8_t channel,
                        PatchPrimaryKey patch_key) {
-               
+
                boost::shared_ptr<ChannelNameSet> channel_name_set = find_channel_name_set(model, custom_device_mode, channel);
-               
+
                if (channel_name_set) {
-                       return  channel_name_set->find_patch(patch_key);                        
+                       return  channel_name_set->find_patch(patch_key);
                } else {
                        return boost::shared_ptr<Patch>();
                }
        }
-       
+
        boost::shared_ptr<Patch> previous_patch(
-                       string model, 
-                       string custom_device_mode, 
-                       uint8_t channel, 
+                       std::string model,
+                       std::string custom_device_mode,
+                       uint8_t channel,
                        PatchPrimaryKey patch_key) {
-               
+
                boost::shared_ptr<ChannelNameSet> channel_name_set = find_channel_name_set(model, custom_device_mode, channel);
-               
+
                if (channel_name_set) {
-                       return  channel_name_set->previous_patch(patch_key);                    
+                       return  channel_name_set->previous_patch(patch_key);
                } else {
                        return boost::shared_ptr<Patch>();
                }
        }
-       
+
        boost::shared_ptr<Patch> next_patch(
-                       string model, 
-                       string custom_device_mode, 
-                       uint8_t channel, 
+                       std::string model,
+                       std::string custom_device_mode,
+                       uint8_t channel,
                        PatchPrimaryKey patch_key) {
-               
+
                boost::shared_ptr<ChannelNameSet> channel_name_set = find_channel_name_set(model, custom_device_mode, channel);
-               
+
                if (channel_name_set) {
-                       return  channel_name_set->next_patch(patch_key);                        
+                       return  channel_name_set->next_patch(patch_key);
                } else {
                        return boost::shared_ptr<Patch>();
                }
        }
-       
-       std::list<string> custom_device_mode_names_by_model(std::string model_name) {
+
+       std::list<std::string> custom_device_mode_names_by_model(std::string model_name) {
                if (model_name != "") {
-                       return master_device_by_model(model_name)->custom_device_mode_names();
-               } else {
-                       return std::list<string>();
+                       if (master_device_by_model(model_name)) {
+                               return master_device_by_model(model_name)->custom_device_mode_names();
+                       }
                }
+               return std::list<std::string>();
        }
-       
+
        const MasterDeviceNames::Models& all_models() const { return _all_models; }
-       
+
+       const DeviceNamesByMaker& devices_by_manufacturer() const { return _devices_by_manufacturer; }
+
+private:
+       bool load_midi_name_document(const std::string& file_path);
+       bool add_midi_name_document(boost::shared_ptr<MIDINameDocument>);
+       bool remove_midi_name_document(const std::string& file_path);
+
+       void add_midnam_files_from_directory(const std::string& directory_path);
+       void remove_midnam_files_from_directory(const std::string& directory_path);
+
 private:
-       void drop_session();
-       void refresh();
-       
-       ARDOUR::Session*                        _session;
+       PBD::Searchpath                         _search_path;
+
        MidiNameDocuments                       _documents;
        MIDINameDocument::MasterDeviceNamesList _master_devices_by_model;
+       DeviceNamesByMaker                      _devices_by_manufacturer;
        MasterDeviceNames::Models               _all_models;
 };
 
 } // namespace Name
 
 } // namespace MIDI
+
 #endif /* MIDI_PATCH_MANAGER_H_ */