Show cursor in Cut mode on button press at cutting position
[ardour.git] / gtk2_ardour / patch_change.cc
index b5d9c6c8a7204961f9318132f948b735ecfb4f3c..4006cbdf469b7030ed3bf11d026edf18a4a8dc4a 100644 (file)
 
 #include <iostream>
 
+#include <boost/algorithm/string.hpp>
+
 #include <glibmm/regex.h>
 
 #include "gtkmm2ext/keyboard.h"
-#include "ardour/midi_patch_manager.h"
-#include "ardour_ui.h"
-#include "midi_region_view.h"
-#include "patch_change.h"
+#include "gtkmm2ext/utils.h"
+
+#include "midi++/midnam_patch.h"
+
+#include "canvas/debug.h"
+
 #include "editor.h"
 #include "editor_drag.h"
+#include "midi_region_view.h"
+#include "patch_change.h"
+#include "ui_config.h"
 
 using namespace MIDI::Name;
 using namespace std;
+using Gtkmm2ext::Keyboard;
 
 /** @param x x position in pixels.
  */
-PatchChange::PatchChange(
-               MidiRegionView& region,
-               ArdourCanvas::Group* parent,
-               const string&   text,
-               double          height,
-               double          x,
-               double          y,
-               string&         model_name,
-               string&         custom_device_mode,
-               ARDOUR::MidiModel::PatchChangePtr patch)
+PatchChange::PatchChange(MidiRegionView&                   region,
+                         ArdourCanvas::Container*          parent,
+                         const string&                     text,
+                         double                            height,
+                         double                            x,
+                         double                            y,
+                         ARDOUR::InstrumentInfo&           info,
+                         ARDOUR::MidiModel::PatchChangePtr patch)
        : _region (region)
-       , _model_name(model_name)
-       , _custom_device_mode(custom_device_mode)
+       , _info (info)
        , _patch (patch)
        , _popup_initialized(false)
 {
        _flag = new ArdourCanvas::Flag (
                parent,
                height,
-               ARDOUR_UI::config()->canvasvar_MidiPatchChangeOutline.get(),
-               ARDOUR_UI::config()->canvasvar_MidiPatchChangeFill.get(),
-               ArdourCanvas::Duple (x, y)
-               );
-       
+               UIConfiguration::instance().color ("midi patch change outline"),
+               UIConfiguration::instance().color_mod ("midi patch change fill", "midi patch change fill"),
+               ArdourCanvas::Duple (x, y),
+               true);
+
+       CANVAS_DEBUG_NAME (_flag, text);
+
        _flag->Event.connect (sigc::mem_fun (*this, &PatchChange::event_handler));
+       _flag->set_font_description (UIConfiguration::instance().get_SmallFont());
        _flag->set_text(text);
 }
 
 PatchChange::~PatchChange()
 {
+       delete _flag;
 }
 
 void
 PatchChange::initialize_popup_menus()
 {
-       boost::shared_ptr<ChannelNameSet> channel_name_set =
-               MidiPatchManager::instance()
-               .find_channel_name_set(_model_name, _custom_device_mode, _patch->channel());
+       using namespace MIDI::Name;
+
+       boost::shared_ptr<ChannelNameSet> channel_name_set = _info.get_patches (_patch->channel());
 
        if (!channel_name_set) {
                return;
@@ -79,40 +88,62 @@ PatchChange::initialize_popup_menus()
 
        const ChannelNameSet::PatchBanks& patch_banks = channel_name_set->patch_banks();
 
-       // fill popup menu:
-       Gtk::Menu::MenuList& patch_bank_menus = _popup.items();
+       if (patch_banks.size() > 1) {
 
-       for (ChannelNameSet::PatchBanks::const_iterator bank = patch_banks.begin();
-            bank != patch_banks.end();
-            ++bank) {
-               Glib::RefPtr<Glib::Regex> underscores = Glib::Regex::create("_");
-               std::string replacement(" ");
+               // fill popup menu:
+               Gtk::Menu::MenuList& patch_bank_menus = _popup.items();
 
-               Gtk::Menu& patch_bank_menu = *manage(new Gtk::Menu());
+               for (ChannelNameSet::PatchBanks::const_iterator bank = patch_banks.begin();
+                    bank != patch_banks.end();
+                    ++bank) {
+                       Glib::RefPtr<Glib::Regex> underscores = Glib::Regex::create("_");
+                       std::string replacement(" ");
 
-               const PatchBank::PatchNameList& patches = (*bank)->patch_name_list();
-               Gtk::Menu::MenuList& patch_menus = patch_bank_menu.items();
+                       Gtk::Menu& patch_bank_menu = *manage(new Gtk::Menu());
 
-               for (PatchBank::PatchNameList::const_iterator patch = patches.begin();
-                    patch != patches.end();
-                    ++patch) {
-                       std::string name = underscores->replace((*patch)->name().c_str(), -1, 0, replacement);
+                       const PatchNameList& patches = (*bank)->patch_name_list();
+                       Gtk::Menu::MenuList& patch_menus = patch_bank_menu.items();
+
+                       for (PatchNameList::const_iterator patch = patches.begin();
+                            patch != patches.end();
+                            ++patch) {
+                               std::string name = underscores->replace((*patch)->name().c_str(), -1, 0, replacement);
+
+                               patch_menus.push_back(
+                                       Gtk::Menu_Helpers::MenuElem(
+                                               name,
+                                               sigc::bind(
+                                                       sigc::mem_fun(*this, &PatchChange::on_patch_menu_selected),
+                                                       (*patch)->patch_primary_key())) );
+                       }
+
+
+                       std::string name = underscores->replace((*bank)->name().c_str(), -1, 0, replacement);
 
-                       patch_menus.push_back(
+                       patch_bank_menus.push_back(
                                Gtk::Menu_Helpers::MenuElem(
                                        name,
-                                       sigc::bind(
-                                               sigc::mem_fun(*this, &PatchChange::on_patch_menu_selected),
-                                               (*patch)->patch_primary_key())) );
+                                       patch_bank_menu) );
                }
 
+       } else {
+               /* only one patch bank, so make it the initial menu */
 
-               std::string name = underscores->replace((*bank)->name().c_str(), -1, 0, replacement);
+               const PatchNameList& patches = patch_banks.front()->patch_name_list();
+               Gtk::Menu::MenuList& patch_menus = _popup.items();
 
-               patch_bank_menus.push_back(
-                       Gtk::Menu_Helpers::MenuElem(
-                               name,
-                               patch_bank_menu) );
+               for (PatchNameList::const_iterator patch = patches.begin();
+                    patch != patches.end();
+                    ++patch) {
+                       std::string name = (*patch)->name();
+                       boost::replace_all (name, "_", " ");
+
+                       patch_menus.push_back (
+                               Gtk::Menu_Helpers::MenuElem (
+                                       name,
+                                       sigc::bind (sigc::mem_fun(*this, &PatchChange::on_patch_menu_selected),
+                                                   (*patch)->patch_primary_key())));
+               }
        }
 }
 
@@ -125,12 +156,16 @@ PatchChange::on_patch_menu_selected(const PatchPrimaryKey& key)
 bool
 PatchChange::event_handler (GdkEvent* ev)
 {
+       /* XXX: icky dcast */
+       Editor* e = dynamic_cast<Editor*> (&_region.get_time_axis_view().editor());
+
+       if (!e->internal_editing()) {
+               return false;
+       }
+
        switch (ev->type) {
        case GDK_BUTTON_PRESS:
-       {
-               /* XXX: icky dcast */
-               Editor* e = dynamic_cast<Editor*> (&_region.get_time_axis_view().editor());
-               if (e->current_mouse_mode() == Editing::MouseObject && e->internal_editing()) {
+               if (e->current_mouse_mode() == Editing::MouseContent) {
 
                        if (Gtkmm2ext::Keyboard::is_delete_event (&ev->button)) {
 
@@ -141,14 +176,14 @@ PatchChange::event_handler (GdkEvent* ev)
 
                                _region.edit_patch_change (this);
                                return true;
-                               
+
                        } else if (ev->button.button == 1) {
                                e->drags()->set (new PatchChangeDrag (e, this, &_region), ev);
                                return true;
                        }
                }
 
-               if (ev->button.button == 3) {
+               if (Gtkmm2ext::Keyboard::is_context_menu_event (&ev->button)) {
                        if (!_popup_initialized) {
                                initialize_popup_menus();
                                _popup_initialized = true;
@@ -157,20 +192,31 @@ PatchChange::event_handler (GdkEvent* ev)
                        return true;
                }
                break;
-       }
 
        case GDK_KEY_PRESS:
                switch (ev->key.keyval) {
                case GDK_Up:
                case GDK_KP_Up:
                case GDK_uparrow:
-                       _region.previous_patch (*this);
-                       break;
+                       _region.step_patch(
+                               *this, Keyboard::modifier_state_contains(ev->key.state, Keyboard::TertiaryModifier), 1);
+                       return true;
                case GDK_Down:
                case GDK_KP_Down:
                case GDK_downarrow:
-                       _region.next_patch (*this);
+                       _region.step_patch(
+                               *this, Keyboard::modifier_state_contains(ev->key.state, Keyboard::TertiaryModifier), -1);
+                       return true;
+               default:
                        break;
+               }
+               break;
+
+       case GDK_KEY_RELEASE:
+               switch (ev->key.keyval) {
+               case GDK_BackSpace:
+               case GDK_Delete:
+                       _region.delete_patch_change (this);
                default:
                        break;
                }
@@ -178,10 +224,12 @@ PatchChange::event_handler (GdkEvent* ev)
 
        case GDK_SCROLL:
                if (ev->scroll.direction == GDK_SCROLL_UP) {
-                       _region.previous_patch (*this);
+                       _region.step_patch(
+                               *this, Keyboard::modifier_state_contains(ev->scroll.state, Keyboard::TertiaryModifier), 1);
                        return true;
                } else if (ev->scroll.direction == GDK_SCROLL_DOWN) {
-                       _region.next_patch (*this);
+                       _region.step_patch(
+                               *this, Keyboard::modifier_state_contains(ev->scroll.state, Keyboard::TertiaryModifier), -1);
                        return true;
                }
                break;