3e6320f4d58444360484371733452d666388b794
[ardour.git] / gtk2_ardour / plugin_selector.cc
1 /*
2     Copyright (C) 2000 Paul Davis 
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <cstdio>
21 #include <lrdf.h>
22
23 #include <gtkmm/table.h>
24 #include <gtkmm/stock.h>
25 #include <gtkmm/button.h>
26 #include <gtkmm/notebook.h>
27
28 #include <ardour/plugin_manager.h>
29 #include <ardour/plugin.h>
30 #include <ardour/configuration.h>
31
32 #include "ardour_ui.h"
33 #include "plugin_selector.h"
34 #include "gui_thread.h"
35
36 #include "i18n.h"
37
38 using namespace ARDOUR;
39 using namespace Gtk;
40
41 PluginSelector::PluginSelector (PluginManager *mgr)
42         : ArdourDialog (_("ardour: plugins"), true, false)
43 {
44         set_position (Gtk::WIN_POS_MOUSE);
45         set_name ("PluginSelectorWindow");
46         add_events (Gdk::KEY_PRESS_MASK|Gdk::KEY_RELEASE_MASK);
47
48         manager = mgr;
49         session = 0;
50         o_selected_plug = -1;
51         i_selected_plug = 0;
52
53         lmodel = Gtk::ListStore::create(lcols);
54         ladspa_display.set_model (lmodel);
55         ladspa_display.append_column (_("Available LADSPA plugins"), lcols.name);
56         ladspa_display.append_column (_("Type"), lcols.type);
57         ladspa_display.append_column (_("# Inputs"),lcols.ins);
58         ladspa_display.append_column (_("# Outputs"), lcols.outs);
59         ladspa_display.set_headers_visible (true);
60         ladspa_display.set_reorderable (false);
61         lscroller.set_border_width(10);
62         lscroller.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
63         lscroller.add(ladspa_display);
64
65         amodel = Gtk::ListStore::create(acols);
66         added_list.set_model (amodel);
67         added_list.append_column (_("To be added"), acols.text);
68         added_list.set_headers_visible (true);
69         added_list.set_reorderable (false);
70
71         for (int i = 0; i <=3; i++) {
72                 Gtk::TreeView::Column* column = ladspa_display.get_column(i);
73                 column->set_sort_column(i);
74         }
75
76 #ifdef VST_SUPPORT
77         vmodel = ListStore::create(vcols);
78         vst_display.set_model (vmodel);
79         vst_display.append_column (_("Available plugins"), vcols.name);
80         vst_display.append_column (_("# Inputs"), vcols.ins);
81         vst_display.append_column (_("# Outputs"), vcols.outs);
82         vst_display.set_headers_visible (true);
83         vst_display.set_reorderable (false);
84         vscroller.set_border_width(10);
85         vscroller.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
86         vscroller.add(vst_display);
87
88         for (int i = 0; i <=2; i++) {
89                 column = vst_display.get_column(i);
90                 column->set_sort_column(i);
91         }
92 #endif
93
94         Gtk::Button *btn_add = manage(new Gtk::Button(_("Add")));
95         ARDOUR_UI::instance()->tooltips().set_tip(*btn_add, _("Add a plugin to the effect list"));
96         Gtk::Button *btn_remove = manage(new Gtk::Button(_("Remove")));
97         ARDOUR_UI::instance()->tooltips().set_tip(*btn_remove, _("Remove a plugin from the effect list"));
98         Gtk::Button *btn_update = manage(new Gtk::Button(_("Update")));
99         ARDOUR_UI::instance()->tooltips().set_tip(*btn_update, _("Update available plugins"));
100
101         btn_add->set_name("PluginSelectorButton");
102         btn_remove->set_name("PluginSelectorButton");
103
104         Gtk::Table* table = manage(new Gtk::Table(7, 10));
105         table->set_size_request(750, 500);
106         table->attach(notebook, 0, 7, 0, 5);
107
108         table->attach(*btn_add, 1, 2, 5, 6, Gtk::FILL, Gtk::FILL, 5, 5);
109         table->attach(*btn_remove, 3, 4, 5, 6, Gtk::FILL, Gtk::FILL, 5, 5);
110         table->attach(*btn_update, 5, 6, 5, 6, Gtk::FILL, Gtk::FILL, 5, 5);
111
112         table->attach(added_list, 0, 7, 7, 9);
113
114         add_button (Stock::OK, RESPONSE_ACCEPT);
115         add_button (Stock::CANCEL, RESPONSE_CANCEL);
116
117         get_vbox()->pack_start (*table);
118
119         using namespace Gtk::Notebook_Helpers;
120         notebook.pages().push_back (TabElem (lscroller, _("LADSPA")));
121 #ifdef VST_SUPPORT
122         if (Config->get_use_vst()) {
123                 notebook.pages().push_back (TabElem (vscroller, _("VST")));
124         }
125 #endif
126
127         table->set_name("PluginSelectorTable");
128         //ladspa_display.set_name("PluginSelectorDisplay");
129         ladspa_display.set_name("PluginSelectorList");
130         added_list.set_name("PluginSelectorList");
131
132         ladspa_display.signal_button_press_event().connect_notify (mem_fun(*this, &PluginSelector::row_clicked));
133 #ifdef VST_SUPPORT
134         if (Config->get_use_vst()) {
135                 vst_display.signal_button_press_event().connect_notify (mem_fun(*this, &PluginSelector::row_clicked));
136         }
137 #endif
138         
139         btn_update->signal_clicked().connect (mem_fun(*this, &PluginSelector::btn_update_clicked));
140         btn_add->signal_clicked().connect(mem_fun(*this, &PluginSelector::btn_add_clicked));
141         btn_remove->signal_clicked().connect(mem_fun(*this, &PluginSelector::btn_remove_clicked));
142
143         input_refiller ();
144 }
145
146 void
147 PluginSelector::row_clicked(GdkEventButton* event)
148 {
149         if (event->type == GDK_2BUTTON_PRESS)
150                 btn_add_clicked();
151 }
152
153 void
154 PluginSelector::set_session (Session* s)
155 {
156         ENSURE_GUI_THREAD(bind (mem_fun(*this, &PluginSelector::set_session), s));
157         
158         session = s;
159
160         if (session) {
161                 session->going_away.connect (bind (mem_fun(*this, &PluginSelector::set_session), static_cast<Session*> (0)));
162         }
163 }
164
165 void
166 PluginSelector::_input_refiller (void *arg)
167 {
168
169         ((PluginSelector *) arg)->input_refiller ();
170 }
171
172 int compare(const void *left, const void *right)
173 {
174   return strcmp(*((char**)left), *((char**)right));
175 }
176
177 void
178 PluginSelector::input_refiller ()
179 {
180         guint row;
181         list<PluginInfo *> &plugs = manager->ladspa_plugin_info ();
182         list<PluginInfo *>::iterator i;
183         char ibuf[16], obuf[16];
184         lmodel->clear();
185 #ifdef VST_SUPPORT
186         vmodel->clear();
187 #endif
188         // Insert into GTK list
189         for (row = 0, i=plugs.begin(); i != plugs.end(); ++i, ++row) {
190                 snprintf (ibuf, sizeof(ibuf)-1, "%d", (*i)->n_inputs);
191                 snprintf (obuf, sizeof(obuf)-1, "%d", (*i)->n_outputs);         
192                 
193                 Gtk::TreeModel::Row newrow = *(lmodel->append());
194                 newrow[lcols.name] = (*i)->name.c_str();
195                 newrow[lcols.type] = (*i)->category.c_str();
196                 newrow[lcols.ins] = ibuf;
197                 newrow[lcols.outs] = obuf;
198                 newrow[lcols.plugin] = *i;
199         }
200
201         lmodel->set_sort_column (0, Gtk::SORT_ASCENDING);
202 }
203
204 #ifdef VST_SUPPORT
205
206 void
207 PluginSelector::_vst_refiller (void *arg)
208 {
209         ((PluginSelector *) arg)->vst_refiller ();
210 }
211
212 void
213 PluginSelector::vst_refiller ()
214 {
215         guint row;
216         list<PluginInfo *> &plugs = manager->vst_plugin_info ();
217         list<PluginInfo *>::iterator i;
218         char ibuf[16], obuf[16];
219         
220         // Insert into GTK list
221         for (row = 0, i=plugs.begin(); i != plugs.end(); ++i, ++row) {
222
223                 snprintf (ibuf, sizeof(ibuf)-1, "%d", (*i)->n_inputs);
224                 snprintf (obuf, sizeof(obuf)-1, "%d", (*i)->n_outputs);         
225                 
226                 Gtk::TreeModel::Row newrow = *(vmodel->append());
227                 newrow[vcols.name] = (*i)->name.c_str();
228                 newrow[vcols.ins] = ibuf;
229                 newrow[vcols.outs] = obuf;
230                 newrow[vcols.plugin] = i;
231         }
232         vmodel->set_sort_column (0, Gtk::SORT_ASCENDING);
233 }
234 #endif
235
236 void
237 PluginSelector::use_plugin (PluginInfo* pi)
238 {
239         list<PluginInfo *>::iterator i;
240
241         if (pi == 0 || session == 0) {
242                 return;
243         }
244
245         Plugin *plugin = manager->load (*session, pi);
246
247         if (plugin) {
248                 PluginCreated (plugin);
249         }
250 }
251
252 void
253 PluginSelector::btn_add_clicked()
254 {
255         bool vst = notebook.get_current_page(); // 0 = LADSPA, 1 = VST
256         std::string name;
257         ARDOUR::PluginInfo *pi;
258         Gtk::TreeModel::Row newrow = *(amodel->append());
259         
260         if (vst) {
261 #ifdef VST_SUPPORT
262                 Gtk::TreeModel::Row row = *(vst_display.get_selection()->get_selected());
263                 name = row[vcols.name];
264                 pi = row[vcols.plugin];
265                 added_plugins.push_back (row[vcols.plugin]);
266 #endif
267         } else {
268                 Gtk::TreeModel::Row row = *(ladspa_display.get_selection()->get_selected());
269                 name = row[lcols.name];
270                 pi = row[lcols.plugin];
271                 added_plugins.push_back (row[lcols.plugin]);
272         }
273         newrow[acols.text] = name;
274         newrow[acols.plugin] = pi;
275 }
276
277 void
278 PluginSelector::btn_remove_clicked()
279 {
280         if (o_selected_plug > -1){
281                 gint row = 0;
282                 list<PluginInfo*>::iterator i = added_plugins.begin();
283                 while(row < o_selected_plug){
284                         i++;
285                         row++;
286                 }
287                 added_plugins.erase(i);
288                 o_selected_plug = -1;
289         }
290 }
291
292 void
293 PluginSelector::btn_update_clicked()
294 {
295         manager->refresh ();
296         input_refiller ();
297 }
298
299 int
300 PluginSelector::run ()
301 {
302         ResponseType r;
303         list<PluginInfo*>::iterator i;
304
305         r = (ResponseType) Dialog::run ();
306
307         switch (r) {
308         case RESPONSE_ACCEPT:
309                 for (i = added_plugins.begin(); i != added_plugins.end(); ++i){
310                         use_plugin (*i);
311                 }
312                 break;
313
314         default:
315                 break;
316         }
317
318         cleanup ();
319
320         return (int) r;
321 }
322
323 void
324 PluginSelector::cleanup ()
325 {
326         hide();
327         added_plugins.clear();
328 }
329