/[svn]/gigedit/trunk/src/gigedit/mainwindow.cpp
ViewVC logotype

Diff of /gigedit/trunk/src/gigedit/mainwindow.cpp

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 2446 by persson, Sun Apr 28 15:40:43 2013 UTC revision 3197 by schoenebeck, Sat May 20 17:15:36 2017 UTC
# Line 1  Line 1 
1  /*  /*
2   * Copyright (C) 2006-2013 Andreas Persson   * Copyright (C) 2006-2017 Andreas Persson
3   *   *
4   * This program is free software; you can redistribute it and/or   * This program is free software; you can redistribute it and/or
5   * modify it under the terms of the GNU General Public License as   * modify it under the terms of the GNU General Public License as
# Line 20  Line 20 
20  #include <iostream>  #include <iostream>
21  #include <cstring>  #include <cstring>
22    
23    #include <glibmmconfig.h>
24    // threads.h must be included first to be able to build with
25    // G_DISABLE_DEPRECATED
26    #if (GLIBMM_MAJOR_VERSION == 2 && GLIBMM_MINOR_VERSION == 31 && GLIBMM_MICRO_VERSION >= 2) || \
27        (GLIBMM_MAJOR_VERSION == 2 && GLIBMM_MINOR_VERSION > 31) || GLIBMM_MAJOR_VERSION > 2
28    #include <glibmm/threads.h>
29    #endif
30    
31  #include <glibmm/convert.h>  #include <glibmm/convert.h>
32  #include <glibmm/dispatcher.h>  #include <glibmm/dispatcher.h>
33  #include <glibmm/miscutils.h>  #include <glibmm/miscutils.h>
# Line 31  Line 39 
39  #include <gtkmm/targetentry.h>  #include <gtkmm/targetentry.h>
40  #include <gtkmm/main.h>  #include <gtkmm/main.h>
41  #include <gtkmm/toggleaction.h>  #include <gtkmm/toggleaction.h>
42    #include <gtkmm/accelmap.h>
43  #if GTKMM_MAJOR_VERSION < 3  #if GTKMM_MAJOR_VERSION < 3
44  #include "wrapLabel.hh"  #include "wrapLabel.hh"
45  #endif  #endif
# Line 40  Line 49 
49    
50  #include <stdio.h>  #include <stdio.h>
51  #include <sndfile.h>  #include <sndfile.h>
52    #include <assert.h>
53    
54  #include "mainwindow.h"  #include "mainwindow.h"
55    #include "Settings.h"
56    #include "CombineInstrumentsDialog.h"
57    #include "scripteditor.h"
58    #include "scriptslots.h"
59    #include "ReferencesView.h"
60  #include "../../gfx/status_attached.xpm"  #include "../../gfx/status_attached.xpm"
61  #include "../../gfx/status_detached.xpm"  #include "../../gfx/status_detached.xpm"
62    #include "gfx/builtinpix.h"
63    #include "MacroEditor.h"
64    #include "MacrosSetup.h"
65    
66  MainWindow::MainWindow() :  MainWindow::MainWindow() :
67        m_DimRegionChooser(*this),
68      dimreg_label(_("Changes apply to:")),      dimreg_label(_("Changes apply to:")),
69      dimreg_all_regions(_("all regions")),      dimreg_all_regions(_("all regions")),
70      dimreg_all_dimregs(_("all dimension splits")),      dimreg_all_dimregs(_("all dimension splits")),
71      dimreg_stereo(_("both channels"))      dimreg_stereo(_("both channels")),
72        labelLegend(_("Legend:")),
73        labelNoSample(_(" No Sample")),
74        labelMissingSample(_(" Missing some Sample(s)")),
75        labelLooped(_(" Looped")),
76        labelSomeLoops(_(" Some Loop(s)"))
77  {  {
78        loadBuiltInPix();
79    
80  //    set_border_width(5);  //    set_border_width(5);
81  //    set_default_size(400, 200);  //    set_default_size(400, 200);
82    
   
83      add(m_VBox);      add(m_VBox);
84    
85      // Handle selection      // Handle selection
# Line 76  MainWindow::MainWindow() : Line 99  MainWindow::MainWindow() :
99      m_ScrolledWindowSamples.add(m_TreeViewSamples);      m_ScrolledWindowSamples.add(m_TreeViewSamples);
100      m_ScrolledWindowSamples.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);      m_ScrolledWindowSamples.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
101    
102        m_ScrolledWindowScripts.add(m_TreeViewScripts);
103        m_ScrolledWindowScripts.set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
104    
105    
106      m_TreeViewNotebook.set_size_request(300);      m_TreeViewNotebook.set_size_request(300);
107    
# Line 87  MainWindow::MainWindow() : Line 113  MainWindow::MainWindow() :
113      dimreg_hbox.add(dimreg_stereo);      dimreg_hbox.add(dimreg_stereo);
114      dimreg_vbox.add(dimreg_edit);      dimreg_vbox.add(dimreg_edit);
115      dimreg_vbox.pack_start(dimreg_hbox, Gtk::PACK_SHRINK);      dimreg_vbox.pack_start(dimreg_hbox, Gtk::PACK_SHRINK);
116        {
117            legend_hbox.add(labelLegend);
118    
119            imageNoSample.set(redDot);
120            imageNoSample.set_alignment(Gtk::ALIGN_END);
121            labelNoSample.set_alignment(Gtk::ALIGN_START);
122            legend_hbox.add(imageNoSample);
123            legend_hbox.add(labelNoSample);
124    
125            imageMissingSample.set(yellowDot);
126            imageMissingSample.set_alignment(Gtk::ALIGN_END);
127            labelMissingSample.set_alignment(Gtk::ALIGN_START);
128            legend_hbox.add(imageMissingSample);
129            legend_hbox.add(labelMissingSample);
130    
131            imageLooped.set(blackLoop);
132            imageLooped.set_alignment(Gtk::ALIGN_END);
133            labelLooped.set_alignment(Gtk::ALIGN_START);
134            legend_hbox.add(imageLooped);
135            legend_hbox.add(labelLooped);
136    
137            imageSomeLoops.set(grayLoop);
138            imageSomeLoops.set_alignment(Gtk::ALIGN_END);
139            labelSomeLoops.set_alignment(Gtk::ALIGN_START);
140            legend_hbox.add(imageSomeLoops);
141            legend_hbox.add(labelSomeLoops);
142    
143            legend_hbox.show_all_children();
144        }
145        dimreg_vbox.pack_start(legend_hbox, Gtk::PACK_SHRINK);
146      m_HPaned.add2(dimreg_vbox);      m_HPaned.add2(dimreg_vbox);
147    
148        dimreg_label.set_tooltip_text(_("To automatically apply your changes above globally to the entire instrument, check all 3 check boxes on the right."));
149        dimreg_all_regions.set_tooltip_text(_("If checked: all changes you perform above will automatically be applied to all regions of this instrument as well."));
150        dimreg_all_dimregs.set_tooltip_text(_("If checked: all changes you perform above will automatically be applied as well to all dimension splits of the region selected below."));
151        dimreg_stereo.set_tooltip_text(_("If checked: all changes you perform above will automatically be applied to both audio channel splits (only if a \"stereo\" dimension is defined below)."));
152    
153      m_TreeViewNotebook.append_page(m_ScrolledWindowSamples, _("Samples"));      m_TreeViewNotebook.append_page(m_ScrolledWindowSamples, _("Samples"));
154      m_TreeViewNotebook.append_page(m_ScrolledWindow, _("Instruments"));      m_TreeViewNotebook.append_page(m_ScrolledWindow, _("Instruments"));
155        m_TreeViewNotebook.append_page(m_ScrolledWindowScripts, _("Scripts"));
156    
157      actionGroup = Gtk::ActionGroup::create();      actionGroup = Gtk::ActionGroup::create();
158    
# Line 123  MainWindow::MainWindow() : Line 183  MainWindow::MainWindow() :
183                                           Gtk::Stock::PROPERTIES),                                           Gtk::Stock::PROPERTIES),
184                       sigc::mem_fun(                       sigc::mem_fun(
185                           *this, &MainWindow::show_instr_props));                           *this, &MainWindow::show_instr_props));
186        actionGroup->add(Gtk::Action::create("MidiRules",
187                                             _("_Midi Rules...")),
188                         sigc::mem_fun(
189                             *this, &MainWindow::show_midi_rules));
190        actionGroup->add(Gtk::Action::create("ScriptSlots",
191                                             _("_Script Slots...")),
192                         sigc::mem_fun(
193                             *this, &MainWindow::show_script_slots));
194      actionGroup->add(Gtk::Action::create("Quit", Gtk::Stock::QUIT),      actionGroup->add(Gtk::Action::create("Quit", Gtk::Stock::QUIT),
195                       sigc::mem_fun(                       sigc::mem_fun(
196                           *this, &MainWindow::on_action_quit));                           *this, &MainWindow::on_action_quit));
197      actionGroup->add(Gtk::Action::create("MenuInstrument", _("_Instrument")));      actionGroup->add(
198            Gtk::Action::create("MenuSample", _("_Sample")),
199            sigc::mem_fun(*this, &MainWindow::show_samples_tab)
200        );
201        actionGroup->add(
202            Gtk::Action::create("MenuInstrument", _("_Instrument")),
203            sigc::mem_fun(*this, &MainWindow::show_intruments_tab)
204        );
205        actionGroup->add(
206            Gtk::Action::create("MenuScript", _("Scr_ipt")),
207            sigc::mem_fun(*this, &MainWindow::show_scripts_tab)
208        );
209        actionGroup->add(Gtk::Action::create("AllInstruments", _("_Select")));
210    
211        actionGroup->add(Gtk::Action::create("MenuEdit", _("_Edit")));
212    
213        const Gdk::ModifierType primaryModifierKey =
214    #if defined(__APPLE__)
215        Gdk::META_MASK; // Cmd key on Mac
216    #else
217        Gdk::CONTROL_MASK; // Ctrl key on all other OSs
218    #endif
219    
220        actionGroup->add(Gtk::Action::create("CopyDimRgn",
221                                             _("Copy selected dimension region")),
222                         Gtk::AccelKey(GDK_KEY_c, Gdk::MOD1_MASK),
223                         sigc::mem_fun(*this, &MainWindow::copy_selected_dimrgn));
224    
225        actionGroup->add(Gtk::Action::create("PasteDimRgn",
226                                             _("Paste dimension region")),
227                         Gtk::AccelKey(GDK_KEY_v, Gdk::MOD1_MASK),
228                         sigc::mem_fun(*this, &MainWindow::paste_copied_dimrgn));
229    
230        actionGroup->add(Gtk::Action::create("AdjustClipboard",
231                                             _("Adjust Clipboard Content")),
232                         Gtk::AccelKey(GDK_KEY_x, Gdk::MOD1_MASK),
233                         sigc::mem_fun(*this, &MainWindow::adjust_clipboard_content));
234    
235        actionGroup->add(Gtk::Action::create("SelectPrevRegion",
236                                             _("Select Previous Region")),
237                         Gtk::AccelKey(GDK_KEY_Left, primaryModifierKey),
238                         sigc::mem_fun(*this, &MainWindow::select_prev_region));
239    
240        actionGroup->add(Gtk::Action::create("SelectNextRegion",
241                                             _("Select Next Region")),
242                         Gtk::AccelKey(GDK_KEY_Right, primaryModifierKey),
243                         sigc::mem_fun(*this, &MainWindow::select_next_region));
244    
245        actionGroup->add(Gtk::Action::create("SelectPrevDimRgnZone",
246                                             _("Select Previous Dimension Region Zone")),
247                         Gtk::AccelKey(GDK_KEY_Left, Gdk::MOD1_MASK),
248                         sigc::mem_fun(*this, &MainWindow::select_prev_dim_rgn_zone));
249    
250        actionGroup->add(Gtk::Action::create("SelectNextDimRgnZone",
251                                             _("Select Next Dimension Region Zone")),
252                         Gtk::AccelKey(GDK_KEY_Right, Gdk::MOD1_MASK),
253                         sigc::mem_fun(*this, &MainWindow::select_next_dim_rgn_zone));
254    
255        actionGroup->add(Gtk::Action::create("SelectPrevDimension",
256                                             _("Select Previous Dimension")),
257                         Gtk::AccelKey(GDK_KEY_Up, Gdk::MOD1_MASK),
258                         sigc::mem_fun(*this, &MainWindow::select_prev_dimension));
259    
260        actionGroup->add(Gtk::Action::create("SelectNextDimension",
261                                             _("Select Next Dimension")),
262                         Gtk::AccelKey(GDK_KEY_Down, Gdk::MOD1_MASK),
263                         sigc::mem_fun(*this, &MainWindow::select_next_dimension));
264    
265        actionGroup->add(Gtk::Action::create("SelectAddPrevDimRgnZone",
266                                             _("Add Previous Dimension Region Zone to Selection")),
267                         Gtk::AccelKey(GDK_KEY_Left, Gdk::MOD1_MASK | Gdk::SHIFT_MASK),
268                         sigc::mem_fun(*this, &MainWindow::select_add_prev_dim_rgn_zone));
269    
270        actionGroup->add(Gtk::Action::create("SelectAddNextDimRgnZone",
271                                             _("Add Next Dimension Region Zone to Selection")),
272                         Gtk::AccelKey(GDK_KEY_Right, Gdk::MOD1_MASK | Gdk::SHIFT_MASK),
273                         sigc::mem_fun(*this, &MainWindow::select_add_next_dim_rgn_zone));
274    
     actionGroup->add(Gtk::Action::create("MenuView", _("_View")));  
275      Glib::RefPtr<Gtk::ToggleAction> toggle_action =      Glib::RefPtr<Gtk::ToggleAction> toggle_action =
276            Gtk::ToggleAction::create("CopySampleUnity", _("Copy Sample's _Unity Note"));
277        toggle_action->set_active(true);
278        actionGroup->add(toggle_action);
279    
280        toggle_action =
281            Gtk::ToggleAction::create("CopySampleTune", _("Copy Sample's _Fine Tune"));
282        toggle_action->set_active(true);
283        actionGroup->add(toggle_action);
284    
285        toggle_action =
286            Gtk::ToggleAction::create("CopySampleLoop", _("Copy Sample's _Loop Points"));
287        toggle_action->set_active(true);
288        actionGroup->add(toggle_action);
289    
290    
291        actionGroup->add(Gtk::Action::create("MenuMacro", _("_Macro")));
292    
293    
294        actionGroup->add(Gtk::Action::create("MenuView", _("Vie_w")));
295        toggle_action =
296          Gtk::ToggleAction::create("Statusbar", _("_Statusbar"));          Gtk::ToggleAction::create("Statusbar", _("_Statusbar"));
297      toggle_action->set_active(true);      toggle_action->set_active(true);
298      actionGroup->add(toggle_action,      actionGroup->add(toggle_action,
299                       sigc::mem_fun(                       sigc::mem_fun(
300                           *this, &MainWindow::on_action_view_status_bar));                           *this, &MainWindow::on_action_view_status_bar));
301    
302        toggle_action =
303            Gtk::ToggleAction::create("AutoRestoreWinDim", _("_Auto Restore Window Dimension"));
304        toggle_action->set_active(Settings::singleton()->autoRestoreWindowDimension);
305        actionGroup->add(toggle_action,
306                         sigc::mem_fun(
307                             *this, &MainWindow::on_auto_restore_win_dim));
308    
309        toggle_action =
310            Gtk::ToggleAction::create("SaveWithTemporaryFile", _("Save with _temporary file"));
311        toggle_action->set_active(Settings::singleton()->saveWithTemporaryFile);
312        actionGroup->add(toggle_action,
313                         sigc::mem_fun(
314                             *this, &MainWindow::on_save_with_temporary_file));
315    
316        actionGroup->add(
317            Gtk::Action::create("RefreshAll", _("_Refresh All")),
318            sigc::mem_fun(*this, &MainWindow::on_action_refresh_all)
319        );                
320    
321      action = Gtk::Action::create("MenuHelp", Gtk::Stock::HELP);      action = Gtk::Action::create("MenuHelp", Gtk::Stock::HELP);
322      actionGroup->add(Gtk::Action::create("MenuHelp",      actionGroup->add(Gtk::Action::create("MenuHelp",
323                                           action->property_label()));                                           action->property_label()));
# Line 155  MainWindow::MainWindow() : Line 337  MainWindow::MainWindow() :
337          sigc::mem_fun(*this, &MainWindow::on_action_remove_instrument)          sigc::mem_fun(*this, &MainWindow::on_action_remove_instrument)
338      );      );
339    
340    
341        actionGroup->add(Gtk::Action::create("MenuSettings", _("_Settings")));
342        
343        toggle_action =
344            Gtk::ToggleAction::create("WarnUserOnExtensions", _("Show warning on format _extensions"));
345        toggle_action->set_active(Settings::singleton()->warnUserOnExtensions);
346        actionGroup->add(
347            toggle_action,
348            sigc::mem_fun(*this, &MainWindow::on_action_warn_user_on_extensions)
349        );
350    
351        toggle_action =
352            Gtk::ToggleAction::create("SyncSamplerInstrumentSelection", _("Synchronize sampler's instrument selection"));
353        toggle_action->set_active(Settings::singleton()->syncSamplerInstrumentSelection);
354        actionGroup->add(
355            toggle_action,
356            sigc::mem_fun(*this, &MainWindow::on_action_sync_sampler_instrument_selection)
357        );
358    
359        toggle_action =
360            Gtk::ToggleAction::create("MoveRootNoteWithRegionMoved", _("Move root note with region moved"));
361        toggle_action->set_active(Settings::singleton()->moveRootNoteWithRegionMoved);
362        actionGroup->add(
363            toggle_action,
364            sigc::mem_fun(*this, &MainWindow::on_action_move_root_note_with_region_moved)
365        );
366    
367    
368        actionGroup->add(Gtk::Action::create("MenuTools", _("_Tools")));
369    
370        actionGroup->add(
371            Gtk::Action::create("CombineInstruments", _("_Combine Instruments...")),
372            sigc::mem_fun(*this, &MainWindow::on_action_combine_instruments)
373        );
374    
375        actionGroup->add(
376            Gtk::Action::create("MergeFiles", _("_Merge Files...")),
377            sigc::mem_fun(*this, &MainWindow::on_action_merge_files)
378        );
379    
380    
381      // sample right-click popup actions      // sample right-click popup actions
382      actionGroup->add(      actionGroup->add(
383          Gtk::Action::create("SampleProperties", Gtk::Stock::PROPERTIES),          Gtk::Action::create("SampleProperties", Gtk::Stock::PROPERTIES),
# Line 173  MainWindow::MainWindow() : Line 396  MainWindow::MainWindow() :
396          sigc::mem_fun(*this, &MainWindow::on_action_remove_sample)          sigc::mem_fun(*this, &MainWindow::on_action_remove_sample)
397      );      );
398      actionGroup->add(      actionGroup->add(
399            Gtk::Action::create("RemoveUnusedSamples", _("Remove _Unused Samples")),
400            sigc::mem_fun(*this, &MainWindow::on_action_remove_unused_samples)
401        );
402        actionGroup->add(
403            Gtk::Action::create("ShowSampleRefs", _("Show References...")),
404            sigc::mem_fun(*this, &MainWindow::on_action_view_references)
405        );
406        actionGroup->add(
407            Gtk::Action::create("ReplaceSample",
408                                _("Replace Sample...")),
409            sigc::mem_fun(*this, &MainWindow::on_action_replace_sample)
410        );
411        actionGroup->add(
412          Gtk::Action::create("ReplaceAllSamplesInAllGroups",          Gtk::Action::create("ReplaceAllSamplesInAllGroups",
413                              _("Replace All Samples in All Groups...")),                              _("Replace All Samples in All Groups...")),
414          sigc::mem_fun(*this, &MainWindow::on_action_replace_all_samples_in_all_groups)          sigc::mem_fun(*this, &MainWindow::on_action_replace_all_samples_in_all_groups)
415      );      );
416        
417        // script right-click popup actions
418        actionGroup->add(
419            Gtk::Action::create("AddScriptGroup", _("Add _Group")),
420            sigc::mem_fun(*this, &MainWindow::on_action_add_script_group)
421        );
422        actionGroup->add(
423            Gtk::Action::create("AddScript", _("Add _Script")),
424            sigc::mem_fun(*this, &MainWindow::on_action_add_script)
425        );
426        actionGroup->add(
427            Gtk::Action::create("EditScript", _("_Edit Script...")),
428            sigc::mem_fun(*this, &MainWindow::on_action_edit_script)
429        );
430        actionGroup->add(
431            Gtk::Action::create("RemoveScript", Gtk::Stock::REMOVE),
432            sigc::mem_fun(*this, &MainWindow::on_action_remove_script)
433        );
434    
435      uiManager = Gtk::UIManager::create();      uiManager = Gtk::UIManager::create();
436      uiManager->insert_action_group(actionGroup);      uiManager->insert_action_group(actionGroup);
# Line 196  MainWindow::MainWindow() : Line 450  MainWindow::MainWindow() :
450          "      <separator/>"          "      <separator/>"
451          "      <menuitem action='Quit'/>"          "      <menuitem action='Quit'/>"
452          "    </menu>"          "    </menu>"
453            "    <menu action='MenuEdit'>"
454            "      <menuitem action='CopyDimRgn'/>"
455            "      <menuitem action='AdjustClipboard'/>"
456            "      <menuitem action='PasteDimRgn'/>"
457            "      <separator/>"
458            "      <menuitem action='SelectPrevRegion'/>"
459            "      <menuitem action='SelectNextRegion'/>"
460            "      <separator/>"
461            "      <menuitem action='SelectPrevDimension'/>"
462            "      <menuitem action='SelectNextDimension'/>"
463            "      <menuitem action='SelectPrevDimRgnZone'/>"
464            "      <menuitem action='SelectNextDimRgnZone'/>"
465            "      <menuitem action='SelectAddPrevDimRgnZone'/>"
466            "      <menuitem action='SelectAddNextDimRgnZone'/>"
467            "      <separator/>"
468            "      <menuitem action='CopySampleUnity'/>"
469            "      <menuitem action='CopySampleTune'/>"
470            "      <menuitem action='CopySampleLoop'/>"
471            "    </menu>"
472            "    <menu action='MenuMacro'>"
473            "    </menu>"
474            "    <menu action='MenuSample'>"
475            "      <menuitem action='SampleProperties'/>"
476            "      <menuitem action='AddGroup'/>"
477            "      <menuitem action='AddSample'/>"
478            "      <menuitem action='ShowSampleRefs'/>"
479            "      <menuitem action='ReplaceSample' />"
480            "      <menuitem action='ReplaceAllSamplesInAllGroups' />"
481            "      <separator/>"
482            "      <menuitem action='RemoveSample'/>"
483            "      <menuitem action='RemoveUnusedSamples'/>"
484            "    </menu>"
485          "    <menu action='MenuInstrument'>"          "    <menu action='MenuInstrument'>"
486            "      <menu action='AllInstruments'>"
487            "      </menu>"
488            "      <separator/>"
489            "      <menuitem action='InstrProperties'/>"
490            "      <menuitem action='MidiRules'/>"
491            "      <menuitem action='ScriptSlots'/>"
492            "      <menuitem action='AddInstrument'/>"
493            "      <menuitem action='DupInstrument'/>"
494            "      <separator/>"
495            "      <menuitem action='RemoveInstrument'/>"
496            "    </menu>"
497            "    <menu action='MenuScript'>"
498            "      <menuitem action='AddScriptGroup'/>"
499            "      <menuitem action='AddScript'/>"
500            "      <menuitem action='EditScript'/>"
501            "      <separator/>"
502            "      <menuitem action='RemoveScript'/>"
503          "    </menu>"          "    </menu>"
504          "    <menu action='MenuView'>"          "    <menu action='MenuView'>"
505          "      <menuitem action='Statusbar'/>"          "      <menuitem action='Statusbar'/>"
506            "      <menuitem action='AutoRestoreWinDim'/>"
507            "      <separator/>"
508            "      <menuitem action='RefreshAll'/>"
509            "    </menu>"
510            "    <menu action='MenuTools'>"
511            "      <menuitem action='CombineInstruments'/>"
512            "      <menuitem action='MergeFiles'/>"
513            "    </menu>"
514            "    <menu action='MenuSettings'>"
515            "      <menuitem action='WarnUserOnExtensions'/>"
516            "      <menuitem action='SyncSamplerInstrumentSelection'/>"
517            "      <menuitem action='MoveRootNoteWithRegionMoved'/>"
518            "      <menuitem action='SaveWithTemporaryFile'/>"
519          "    </menu>"          "    </menu>"
520          "    <menu action='MenuHelp'>"          "    <menu action='MenuHelp'>"
521          "      <menuitem action='About'/>"          "      <menuitem action='About'/>"
# Line 207  MainWindow::MainWindow() : Line 523  MainWindow::MainWindow() :
523          "  </menubar>"          "  </menubar>"
524          "  <popup name='PopupMenu'>"          "  <popup name='PopupMenu'>"
525          "    <menuitem action='InstrProperties'/>"          "    <menuitem action='InstrProperties'/>"
526            "    <menuitem action='MidiRules'/>"
527            "    <menuitem action='ScriptSlots'/>"
528          "    <menuitem action='AddInstrument'/>"          "    <menuitem action='AddInstrument'/>"
529          "    <menuitem action='DupInstrument'/>"          "    <menuitem action='DupInstrument'/>"
530          "    <separator/>"          "    <separator/>"
# Line 216  MainWindow::MainWindow() : Line 534  MainWindow::MainWindow() :
534          "    <menuitem action='SampleProperties'/>"          "    <menuitem action='SampleProperties'/>"
535          "    <menuitem action='AddGroup'/>"          "    <menuitem action='AddGroup'/>"
536          "    <menuitem action='AddSample'/>"          "    <menuitem action='AddSample'/>"
537            "    <menuitem action='ShowSampleRefs'/>"
538            "    <menuitem action='ReplaceSample' />"
539          "    <menuitem action='ReplaceAllSamplesInAllGroups' />"          "    <menuitem action='ReplaceAllSamplesInAllGroups' />"
540          "    <separator/>"          "    <separator/>"
541          "    <menuitem action='RemoveSample'/>"          "    <menuitem action='RemoveSample'/>"
542            "    <menuitem action='RemoveUnusedSamples'/>"
543            "  </popup>"
544            "  <popup name='ScriptPopupMenu'>"
545            "    <menuitem action='AddScriptGroup'/>"
546            "    <menuitem action='AddScript'/>"
547            "    <menuitem action='EditScript'/>"
548            "    <separator/>"
549            "    <menuitem action='RemoveScript'/>"
550          "  </popup>"          "  </popup>"
551          "</ui>";          "</ui>";
552      uiManager->add_ui_from_string(ui_info);      uiManager->add_ui_from_string(ui_info);
553    
554      popup_menu = dynamic_cast<Gtk::Menu*>(uiManager->get_widget("/PopupMenu"));      popup_menu = dynamic_cast<Gtk::Menu*>(uiManager->get_widget("/PopupMenu"));
555        
556        // Set tooltips for menu items (for some reason, setting a tooltip on the
557        // respective Gtk::Action objects above will simply be ignored, no matter
558        // if using Gtk::Action::set_tooltip() or passing the tooltip string on
559        // Gtk::Action::create()).
560        {
561            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
562                uiManager->get_widget("/MenuBar/MenuEdit/CopySampleUnity"));
563            item->set_tooltip_text(_("Used when dragging a sample to a region's sample reference field. You may disable this for example if you want to replace an existing sample in a region with a new sample, but don't want that the region's current unity note setting will be altered by this action."));
564        }
565        {
566            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
567                uiManager->get_widget("/MenuBar/MenuEdit/CopySampleTune"));
568            item->set_tooltip_text(_("Used when dragging a sample to a region's sample reference field. You may disable this for example if you want to replace an existing sample in a region with a new sample, but don't want that the region's current sample playback tuning will be altered by this action."));
569        }
570        {
571            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
572                uiManager->get_widget("/MenuBar/MenuEdit/CopySampleLoop"));
573            item->set_tooltip_text(_("Used when dragging a sample to a region's sample reference field. You may disable this for example if you want to replace an existing sample in a region with a new sample, but don't want that the region's current loop informations to be altered by this action."));
574        }
575        {
576            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
577                uiManager->get_widget("/MenuBar/MenuSettings/WarnUserOnExtensions"));
578            item->set_tooltip_text(_("If checked, a warning will be shown whenever you try to use a feature which is based on a LinuxSampler extension ontop of the original gig format, which would not work with the Gigasampler/GigaStudio application."));
579        }
580        {
581            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
582                uiManager->get_widget("/MenuBar/MenuSettings/SyncSamplerInstrumentSelection"));
583            item->set_tooltip_text(_("If checked, the sampler's current instrument will automatically be switched whenever another instrument was selected in gigedit (only available in live-mode)."));
584        }
585        {
586            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
587                uiManager->get_widget("/MenuBar/MenuSettings/MoveRootNoteWithRegionMoved"));
588            item->set_tooltip_text(_("If checked, and when a region is moved by dragging it around on the virtual keyboard, the keybord position dependent pitch will move exactly with the amount of semi tones the region was moved around."));
589        }
590        {
591            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
592                uiManager->get_widget("/MenuBar/MenuSample/RemoveUnusedSamples"));
593            item->set_tooltip_text(_("Removes all samples that are not referenced by any instrument (i.e. red ones)."));
594            // copy tooltip to popup menu
595            Gtk::MenuItem* item2 = dynamic_cast<Gtk::MenuItem*>(
596                uiManager->get_widget("/SamplePopupMenu/RemoveUnusedSamples"));
597            item2->set_tooltip_text(item->get_tooltip_text());
598        }
599        {
600            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
601                uiManager->get_widget("/MenuBar/MenuView/RefreshAll"));
602            item->set_tooltip_text(_("Reloads the currently open gig file and updates the entire graphical user interface."));
603        }
604        {
605            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
606                uiManager->get_widget("/MenuBar/MenuView/AutoRestoreWinDim"));
607            item->set_tooltip_text(_("If checked, size and position of all windows will be saved and automatically restored next time."));
608        }
609        {
610            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
611                uiManager->get_widget("/MenuBar/MenuTools/CombineInstruments"));
612            item->set_tooltip_text(_("Create combi sounds out of individual sounds of this .gig file."));
613        }
614        {
615            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
616                uiManager->get_widget("/MenuBar/MenuTools/MergeFiles"));
617            item->set_tooltip_text(_("Add instruments and samples of other .gig files to this .gig file."));
618        }
619    
620    
621      instrument_menu = static_cast<Gtk::MenuItem*>(      instrument_menu = static_cast<Gtk::MenuItem*>(
622          uiManager->get_widget("/MenuBar/MenuInstrument"))->get_submenu();          uiManager->get_widget("/MenuBar/MenuInstrument/AllInstruments"))->get_submenu();
623    
624      Gtk::Widget* menuBar = uiManager->get_widget("/MenuBar");      Gtk::Widget* menuBar = uiManager->get_widget("/MenuBar");
625      m_VBox.pack_start(*menuBar, Gtk::PACK_SHRINK);      m_VBox.pack_start(*menuBar, Gtk::PACK_SHRINK);
# Line 252  MainWindow::MainWindow() : Line 645  MainWindow::MainWindow() :
645      // Create the Tree model:      // Create the Tree model:
646      m_refTreeModel = Gtk::ListStore::create(m_Columns);      m_refTreeModel = Gtk::ListStore::create(m_Columns);
647      m_TreeView.set_model(m_refTreeModel);      m_TreeView.set_model(m_refTreeModel);
648        m_TreeView.get_selection()->set_mode(Gtk::SELECTION_MULTIPLE);
649        m_TreeView.set_tooltip_text(_("Right click here for actions on instruments & MIDI Rules. Drag & drop to change the order of instruments."));
650      instrument_name_connection = m_refTreeModel->signal_row_changed().connect(      instrument_name_connection = m_refTreeModel->signal_row_changed().connect(
651          sigc::mem_fun(*this, &MainWindow::instrument_name_changed)          sigc::mem_fun(*this, &MainWindow::instrument_name_changed)
652      );      );
653    
654      // Add the TreeView's view columns:      // Add the TreeView's view columns:
655      m_TreeView.append_column_editable("Instrument", m_Columns.m_col_name);      m_TreeView.append_column(_("Nr"), m_Columns.m_col_nr);
656      m_TreeView.set_headers_visible(false);      m_TreeView.append_column_editable(_("Instrument"), m_Columns.m_col_name);
657        m_TreeView.append_column(_("Scripts"), m_Columns.m_col_scripts);
658        m_TreeView.set_headers_visible(true);
659        
660        // establish drag&drop within the instrument tree view, allowing to reorder
661        // the sequence of instruments within the gig file
662        {
663            std::vector<Gtk::TargetEntry> drag_target_instrument;
664            drag_target_instrument.push_back(Gtk::TargetEntry("gig::Instrument"));
665            m_TreeView.drag_source_set(drag_target_instrument);
666            m_TreeView.drag_dest_set(drag_target_instrument);
667            m_TreeView.signal_drag_begin().connect(
668                sigc::mem_fun(*this, &MainWindow::on_instruments_treeview_drag_begin)
669            );
670            m_TreeView.signal_drag_data_get().connect(
671                sigc::mem_fun(*this, &MainWindow::on_instruments_treeview_drag_data_get)
672            );
673            m_TreeView.signal_drag_data_received().connect(
674                sigc::mem_fun(*this, &MainWindow::on_instruments_treeview_drop_drag_data_received)
675            );
676        }
677    
678      // create samples treeview (including its data model)      // create samples treeview (including its data model)
679      m_refSamplesTreeModel = SamplesTreeStore::create(m_SamplesModel);      m_refSamplesTreeModel = SamplesTreeStore::create(m_SamplesModel);
680      m_TreeViewSamples.set_model(m_refSamplesTreeModel);      m_TreeViewSamples.set_model(m_refSamplesTreeModel);
681        m_TreeViewSamples.get_selection()->set_mode(Gtk::SELECTION_MULTIPLE);
682        m_TreeViewSamples.set_tooltip_text(_("To actually use a sample, drag it from this list view to \"Sample\" -> \"Sample:\" on the region's settings pane on the right.\n\nRight click here for more actions on samples."));
683      // m_TreeViewSamples.set_reorderable();      // m_TreeViewSamples.set_reorderable();
684      m_TreeViewSamples.append_column_editable("Samples", m_SamplesModel.m_col_name);      m_TreeViewSamples.append_column_editable(_("Name"), m_SamplesModel.m_col_name);
685      m_TreeViewSamples.set_headers_visible(false);      m_TreeViewSamples.append_column(_("Referenced"), m_SamplesModel.m_col_refcount);
686        {
687            Gtk::TreeViewColumn* column = m_TreeViewSamples.get_column(0);
688            Gtk::CellRendererText* cellrenderer =
689                dynamic_cast<Gtk::CellRendererText*>(column->get_first_cell());
690            column->add_attribute(
691                cellrenderer->property_foreground(), m_SamplesModel.m_color
692            );
693        }
694        {
695            Gtk::TreeViewColumn* column = m_TreeViewSamples.get_column(1);
696            Gtk::CellRendererText* cellrenderer =
697                dynamic_cast<Gtk::CellRendererText*>(column->get_first_cell());
698            column->add_attribute(
699                cellrenderer->property_foreground(), m_SamplesModel.m_color
700            );
701        }
702        m_TreeViewSamples.set_headers_visible(true);
703      m_TreeViewSamples.signal_button_press_event().connect_notify(      m_TreeViewSamples.signal_button_press_event().connect_notify(
704          sigc::mem_fun(*this, &MainWindow::on_sample_treeview_button_release)          sigc::mem_fun(*this, &MainWindow::on_sample_treeview_button_release)
705      );      );
# Line 273  MainWindow::MainWindow() : Line 707  MainWindow::MainWindow() :
707          sigc::mem_fun(*this, &MainWindow::sample_name_changed)          sigc::mem_fun(*this, &MainWindow::sample_name_changed)
708      );      );
709    
710        // create scripts treeview (including its data model)
711        m_refScriptsTreeModel = ScriptsTreeStore::create(m_ScriptsModel);
712        m_TreeViewScripts.set_model(m_refScriptsTreeModel);
713        m_TreeViewScripts.set_tooltip_text(_(
714            "Use CTRL + double click for editing a script."
715            "\n\n"
716            "Note: instrument scripts are a LinuxSampler extension of the gig "
717            "format. This feature will not work with the GigaStudio software!"
718        ));
719        // m_TreeViewScripts.set_reorderable();
720        m_TreeViewScripts.append_column_editable("Samples", m_ScriptsModel.m_col_name);
721        m_TreeViewScripts.set_headers_visible(false);
722        m_TreeViewScripts.signal_button_press_event().connect_notify(
723            sigc::mem_fun(*this, &MainWindow::on_script_treeview_button_release)
724        );
725        //FIXME: why the heck does this double click signal_row_activated() only fire while CTRL key is pressed ?
726        m_TreeViewScripts.signal_row_activated().connect(
727            sigc::mem_fun(*this, &MainWindow::script_double_clicked)
728        );
729        m_refScriptsTreeModel->signal_row_changed().connect(
730            sigc::mem_fun(*this, &MainWindow::script_name_changed)
731        );
732    
733        // establish drag&drop between scripts tree view and ScriptSlots window
734        std::vector<Gtk::TargetEntry> drag_target_gig_script;
735        drag_target_gig_script.push_back(Gtk::TargetEntry("gig::Script"));
736        m_TreeViewScripts.drag_source_set(drag_target_gig_script);
737        m_TreeViewScripts.signal_drag_begin().connect(
738            sigc::mem_fun(*this, &MainWindow::on_scripts_treeview_drag_begin)
739        );
740        m_TreeViewScripts.signal_drag_data_get().connect(
741            sigc::mem_fun(*this, &MainWindow::on_scripts_treeview_drag_data_get)
742        );
743    
744      // establish drag&drop between samples tree view and dimension region 'Sample' text entry      // establish drag&drop between samples tree view and dimension region 'Sample' text entry
745      std::vector<Gtk::TargetEntry> drag_target_gig_sample;      std::vector<Gtk::TargetEntry> drag_target_gig_sample;
746      drag_target_gig_sample.push_back(Gtk::TargetEntry("gig::Sample"));      drag_target_gig_sample.push_back(Gtk::TargetEntry("gig::Sample"));
# Line 291  MainWindow::MainWindow() : Line 759  MainWindow::MainWindow() :
759          sigc::hide(sigc::mem_fun(*this, &MainWindow::file_changed)));          sigc::hide(sigc::mem_fun(*this, &MainWindow::file_changed)));
760      m_RegionChooser.signal_instrument_changed().connect(      m_RegionChooser.signal_instrument_changed().connect(
761          sigc::mem_fun(*this, &MainWindow::file_changed));          sigc::mem_fun(*this, &MainWindow::file_changed));
762        m_RegionChooser.signal_instrument_changed().connect(
763            sigc::mem_fun(*this, &MainWindow::region_changed));
764      m_DimRegionChooser.signal_region_changed().connect(      m_DimRegionChooser.signal_region_changed().connect(
765          sigc::mem_fun(*this, &MainWindow::file_changed));          sigc::mem_fun(*this, &MainWindow::file_changed));
766      instrumentProps.signal_changed().connect(      instrumentProps.signal_changed().connect(
767          sigc::mem_fun(*this, &MainWindow::file_changed));          sigc::mem_fun(*this, &MainWindow::file_changed));
768      propDialog.signal_changed().connect(      propDialog.signal_changed().connect(
769          sigc::mem_fun(*this, &MainWindow::file_changed));          sigc::mem_fun(*this, &MainWindow::file_changed));
770        midiRules.signal_changed().connect(
771            sigc::mem_fun(*this, &MainWindow::file_changed));
772    
773      dimreg_edit.signal_dimreg_to_be_changed().connect(      dimreg_edit.signal_dimreg_to_be_changed().connect(
774          dimreg_to_be_changed_signal.make_slot());          dimreg_to_be_changed_signal.make_slot());
# Line 304  MainWindow::MainWindow() : Line 776  MainWindow::MainWindow() :
776          dimreg_changed_signal.make_slot());          dimreg_changed_signal.make_slot());
777      dimreg_edit.signal_sample_ref_changed().connect(      dimreg_edit.signal_sample_ref_changed().connect(
778          sample_ref_changed_signal.make_slot());          sample_ref_changed_signal.make_slot());
779        sample_ref_changed_signal.connect(
780            sigc::mem_fun(*this, &MainWindow::on_sample_ref_changed)
781        );
782        samples_to_be_removed_signal.connect(
783            sigc::mem_fun(*this, &MainWindow::on_samples_to_be_removed)
784        );
785    
786        dimreg_edit.signal_select_sample().connect(
787            sigc::mem_fun(*this, &MainWindow::select_sample)
788        );
789    
790      m_RegionChooser.signal_instrument_struct_to_be_changed().connect(      m_RegionChooser.signal_instrument_struct_to_be_changed().connect(
791          sigc::hide(          sigc::hide(
# Line 345  MainWindow::MainWindow() : Line 827  MainWindow::MainWindow() :
827    
828      // start with a new gig file by default      // start with a new gig file by default
829      on_action_file_new();      on_action_file_new();
830    
831        // select 'Instruments' tab by default
832        // (gtk allows this only if the tab childs are visible, thats why it's here)
833        m_TreeViewNotebook.set_current_page(1);
834    
835        Gtk::Clipboard::get()->signal_owner_change().connect(
836            sigc::mem_fun(*this, &MainWindow::on_clipboard_owner_change)
837        );
838        updateClipboardPasteAvailable();
839        updateClipboardCopyAvailable();
840    
841        // setup macros and their keyboard accelerators
842        {
843            Gtk::Menu* menuMacro = dynamic_cast<Gtk::MenuItem*>(
844                uiManager->get_widget("/MenuBar/MenuMacro")
845            )->get_submenu();
846    
847            const Gdk::ModifierType primaryModifierKey =
848    #if defined(__APPLE__)
849                Gdk::META_MASK; // Cmd key on Mac
850    #else
851                Gdk::CONTROL_MASK; // Ctrl key on all other OSs
852    #endif
853    
854            const Gdk::ModifierType noModifier = (Gdk::ModifierType)0;
855            Gtk::AccelMap::add_entry("<Macros>/macro_0", GDK_KEY_F1, noModifier);
856            Gtk::AccelMap::add_entry("<Macros>/macro_1", GDK_KEY_F2, noModifier);
857            Gtk::AccelMap::add_entry("<Macros>/macro_2", GDK_KEY_F3, noModifier);
858            Gtk::AccelMap::add_entry("<Macros>/macro_3", GDK_KEY_F4, noModifier);
859            Gtk::AccelMap::add_entry("<Macros>/macro_4", GDK_KEY_F5, noModifier);
860            Gtk::AccelMap::add_entry("<Macros>/macro_5", GDK_KEY_F6, noModifier);
861            Gtk::AccelMap::add_entry("<Macros>/macro_6", GDK_KEY_F7, noModifier);
862            Gtk::AccelMap::add_entry("<Macros>/macro_7", GDK_KEY_F8, noModifier);
863            Gtk::AccelMap::add_entry("<Macros>/macro_8", GDK_KEY_F9, noModifier);
864            Gtk::AccelMap::add_entry("<Macros>/macro_9", GDK_KEY_F10, noModifier);
865            Gtk::AccelMap::add_entry("<Macros>/macro_10", GDK_KEY_F11, noModifier);
866            Gtk::AccelMap::add_entry("<Macros>/macro_11", GDK_KEY_F12, noModifier);
867            Gtk::AccelMap::add_entry("<Macros>/SetupMacros", 'm', primaryModifierKey);
868    
869            Glib::RefPtr<Gtk::AccelGroup> accelGroup = this->get_accel_group();
870            menuMacro->set_accel_group(accelGroup);
871    
872            updateMacroMenu();
873        }
874  }  }
875    
876  MainWindow::~MainWindow()  MainWindow::~MainWindow()
877  {  {
878  }  }
879    
880    void MainWindow::updateMacroMenu() {
881        Gtk::Menu* menuMacro = dynamic_cast<Gtk::MenuItem*>(
882            uiManager->get_widget("/MenuBar/MenuMacro")
883        )->get_submenu();
884    
885        // remove all entries from "Macro" menu
886        {
887            const std::vector<Gtk::Widget*> children = menuMacro->get_children();
888            for (int i = 0; i < children.size(); ++i) {
889                Gtk::Widget* child = children[i];
890                menuMacro->remove(*child);
891                delete child;
892            }
893        }
894    
895        // (re)load all macros from config file
896        try {
897            Settings::singleton()->loadMacros(m_macros);
898        } catch (Serialization::Exception e) {
899            std::cerr << "Exception while loading macros: " << e.Message << std::endl;
900        } catch (...) {
901            std::cerr << "Unknown exception while loading macros!" << std::endl;
902        }
903    
904        // add all configured macros as menu items to the "Macro" menu
905        for (int iMacro = 0; iMacro < m_macros.size(); ++iMacro) {
906            const Serialization::Archive& macro = m_macros[iMacro];
907            std::string name =
908                macro.name().empty() ?
909                    (std::string(_("Unnamed Macro")) + " " + ToString(iMacro+1)) : macro.name();
910            Gtk::MenuItem* item = new Gtk::MenuItem(name);
911            item->signal_activate().connect(
912                sigc::bind(
913                    sigc::mem_fun(*this, &MainWindow::onMacroSelected), iMacro
914                )
915            );
916            menuMacro->append(*item);
917            item->set_accel_path("<Macros>/macro_" + ToString(iMacro));
918            Glib::ustring comment = macro.comment();
919            if (!comment.empty())
920                item->set_tooltip_text(comment);
921        }
922        // if there are no macros configured at all, then show a dummy entry instead
923        if (m_macros.empty()) {
924            Gtk::MenuItem* item = new Gtk::MenuItem(_("No Macros"));
925            item->set_sensitive(false);
926            menuMacro->append(*item);
927        }
928    
929        // add separator line to menu
930        menuMacro->append(*new Gtk::SeparatorMenuItem);
931    
932        {
933            Gtk::MenuItem* item = new Gtk::MenuItem(_("Setup Macros ..."));
934            item->signal_activate().connect(
935                sigc::mem_fun(*this, &MainWindow::setupMacros)
936            );
937            menuMacro->append(*item);
938            item->set_accel_path("<Macros>/SetupMacros");
939        }
940    
941        menuMacro->show_all_children();
942    }
943    
944    void MainWindow::onMacroSelected(int iMacro) {
945        printf("onMacroSelected(%d)\n", iMacro);
946        if (iMacro < 0 || iMacro >= m_macros.size()) return;
947        Glib::ustring errorText;
948        try {
949            applyMacro(m_macros[iMacro]);
950        } catch (Serialization::Exception e) {
951            errorText = e.Message;
952        } catch (...) {
953            errorText = _("Unknown exception while applying macro");
954        }
955        if (!errorText.empty()) {
956            Glib::ustring txt = _("Applying macro failed:\n") + errorText;
957            Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
958            msg.run();
959        }
960    }
961    
962    void MainWindow::setupMacros() {
963        MacrosSetup* setup = new MacrosSetup();
964        gig::DimensionRegion* pDimRgn = m_DimRegionChooser.get_main_dimregion();
965        setup->setMacros(m_macros, &m_serializationArchive, pDimRgn);
966        setup->signal_macros_changed().connect(
967            sigc::mem_fun(*this, &MainWindow::onMacrosSetupChanged)
968        );
969        setup->show();
970    }
971    
972    void MainWindow::onMacrosSetupChanged(const std::vector<Serialization::Archive>& macros) {
973        m_macros = macros;
974        Settings::singleton()->saveMacros(m_macros);
975        updateMacroMenu();
976    }
977    
978  bool MainWindow::on_delete_event(GdkEventAny* event)  bool MainWindow::on_delete_event(GdkEventAny* event)
979  {  {
980      return !file_is_shared && file_is_changed && !close_confirmation_dialog();      return !file_is_shared && file_is_changed && !close_confirmation_dialog();
# Line 370  void MainWindow::region_changed() Line 994  void MainWindow::region_changed()
994  gig::Instrument* MainWindow::get_instrument()  gig::Instrument* MainWindow::get_instrument()
995  {  {
996      gig::Instrument* instrument = 0;      gig::Instrument* instrument = 0;
997      Gtk::TreeModel::const_iterator it =      std::vector<Gtk::TreeModel::Path> rows = m_TreeView.get_selection()->get_selected_rows();
998          m_TreeView.get_selection()->get_selected();      if (rows.empty()) return NULL;
999        Gtk::TreeModel::const_iterator it = m_refTreeModel->get_iter(rows[0]);
1000      if (it) {      if (it) {
1001          Gtk::TreeModel::Row row = *it;          Gtk::TreeModel::Row row = *it;
1002          instrument = row[m_Columns.m_col_instr];          instrument = row[m_Columns.m_col_instr];
# Line 414  void MainWindow::update_dimregs() Line 1039  void MainWindow::update_dimregs()
1039              add_region_to_dimregs(region, stereo, all_dimregs);              add_region_to_dimregs(region, stereo, all_dimregs);
1040          }          }
1041      }      }
1042    
1043        m_RegionChooser.setModifyAllRegions(all_regions);
1044        m_DimRegionChooser.setModifyAllRegions(all_regions);
1045        m_DimRegionChooser.setModifyAllDimensionRegions(all_dimregs);
1046        m_DimRegionChooser.setModifyBothChannels(stereo);
1047    
1048        updateClipboardCopyAvailable();
1049  }  }
1050    
1051  void MainWindow::dimreg_all_dimregs_toggled()  void MainWindow::dimreg_all_dimregs_toggled()
# Line 425  void MainWindow::dimreg_all_dimregs_togg Line 1057  void MainWindow::dimreg_all_dimregs_togg
1057  void MainWindow::dimreg_changed()  void MainWindow::dimreg_changed()
1058  {  {
1059      update_dimregs();      update_dimregs();
1060      dimreg_edit.set_dim_region(m_DimRegionChooser.get_dimregion());      dimreg_edit.set_dim_region(m_DimRegionChooser.get_main_dimregion());
1061  }  }
1062    
1063  void MainWindow::on_sel_change()  void MainWindow::on_sel_change()
1064  {  {
1065      // select item in instrument menu      // select item in instrument menu
1066      Gtk::TreeModel::iterator it = m_TreeView.get_selection()->get_selected();      std::vector<Gtk::TreeModel::Path> rows = m_TreeView.get_selection()->get_selected_rows();
1067      if (it) {      if (!rows.empty()) {
1068          Gtk::TreePath path(it);          Gtk::TreeModel::iterator it = m_refTreeModel->get_iter(rows[0]);
1069          int index = path[0];          if (it) {
1070          const std::vector<Gtk::Widget*> children =              Gtk::TreePath path(it);
1071              instrument_menu->get_children();              int index = path[0];
1072          static_cast<Gtk::RadioMenuItem*>(children[index])->set_active();              const std::vector<Gtk::Widget*> children =
1073                    instrument_menu->get_children();
1074                static_cast<Gtk::RadioMenuItem*>(children[index])->set_active();
1075            }
1076      }      }
1077    
1078      m_RegionChooser.set_instrument(get_instrument());      m_RegionChooser.set_instrument(get_instrument());
1079    
1080        if (Settings::singleton()->syncSamplerInstrumentSelection) {
1081            switch_sampler_instrument_signal.emit(get_instrument());
1082        }
1083  }  }
1084    
1085  void loader_progress_callback(gig::progress_t* progress)  void loader_progress_callback(gig::progress_t* progress)
# Line 458  void Loader::progress_callback(float fra Line 1097  void Loader::progress_callback(float fra
1097      progress_dispatcher();      progress_dispatcher();
1098  }  }
1099    
1100    #if defined(WIN32) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 2))
1101    // make sure stack is 16-byte aligned for SSE instructions
1102    __attribute__((force_align_arg_pointer))
1103    #endif
1104  void Loader::thread_function()  void Loader::thread_function()
1105  {  {
1106      printf("thread_function self=%x\n", Glib::Threads::Thread::self());      printf("thread_function self=%p\n",
1107      printf("Start %s\n", filename);             static_cast<void*>(Glib::Threads::Thread::self()));
1108      RIFF::File* riff = new RIFF::File(filename);      printf("Start %s\n", filename.c_str());
1109      gig = new gig::File(riff);      try {
1110      gig::progress_t progress;          RIFF::File* riff = new RIFF::File(filename);
1111      progress.callback = loader_progress_callback;          gig = new gig::File(riff);
1112      progress.custom = this;          gig::progress_t progress;
1113            progress.callback = loader_progress_callback;
1114      gig->GetInstrument(0, &progress);          progress.custom = this;
1115      printf("End\n");  
1116      finished_dispatcher();          gig->GetInstrument(0, &progress);
1117            printf("End\n");
1118            finished_dispatcher();
1119        } catch (RIFF::Exception e) {
1120            error_message = e.Message;
1121            error_dispatcher.emit();
1122        } catch (...) {
1123            error_message = _("Unknown exception occurred");
1124            error_dispatcher.emit();
1125        }
1126  }  }
1127    
1128  Loader::Loader(const char* filename)  Loader::Loader(const char* filename)
1129      : filename(filename), thread(0)      : filename(filename), gig(0), thread(0), progress(0.f)
1130  {  {
1131  }  }
1132    
# Line 485  void Loader::launch() Line 1137  void Loader::launch()
1137  #else  #else
1138      thread = Glib::Threads::Thread::create(sigc::mem_fun(*this, &Loader::thread_function));      thread = Glib::Threads::Thread::create(sigc::mem_fun(*this, &Loader::thread_function));
1139  #endif  #endif
1140      printf("launch thread=%x\n", thread);      printf("launch thread=%p\n", static_cast<void*>(thread));
1141  }  }
1142    
1143  float Loader::get_progress()  float Loader::get_progress()
# Line 508  Glib::Dispatcher& Loader::signal_finishe Line 1160  Glib::Dispatcher& Loader::signal_finishe
1160      return finished_dispatcher;      return finished_dispatcher;
1161  }  }
1162    
1163  LoadDialog::LoadDialog(const Glib::ustring& title, Gtk::Window& parent)  Glib::Dispatcher& Loader::signal_error()
1164    {
1165        return error_dispatcher;
1166    }
1167    
1168    void saver_progress_callback(gig::progress_t* progress)
1169    {
1170        Saver* saver = static_cast<Saver*>(progress->custom);
1171        saver->progress_callback(progress->factor);
1172    }
1173    
1174    void Saver::progress_callback(float fraction)
1175    {
1176        {
1177            Glib::Threads::Mutex::Lock lock(progressMutex);
1178            progress = fraction;
1179        }
1180        progress_dispatcher.emit();
1181    }
1182    
1183    #if defined(WIN32) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 2))
1184    // make sure stack is 16-byte aligned for SSE instructions
1185    __attribute__((force_align_arg_pointer))
1186    #endif
1187    void Saver::thread_function()
1188    {
1189        printf("thread_function self=%p\n",
1190               static_cast<void*>(Glib::Threads::Thread::self()));
1191        printf("Start %s\n", filename.c_str());
1192        try {
1193            gig::progress_t progress;
1194            progress.callback = saver_progress_callback;
1195            progress.custom = this;
1196    
1197            // if no filename was provided, that means "save", if filename was provided means "save as"
1198            if (filename.empty()) {
1199                if (!Settings::singleton()->saveWithTemporaryFile) {
1200                    // save directly over the existing .gig file
1201                    // (requires less disk space than solution below
1202                    // but may be slower)
1203                    gig->Save(&progress);
1204                } else {
1205                    // save the file as separate temporary file first,
1206                    // then move the saved file over the old file
1207                    // (may result in performance speedup during save)
1208                    gig::String tmpname = filename + ".TMP";
1209                    gig->Save(tmpname, &progress);
1210                    #if defined(WIN32)
1211                    if (!DeleteFile(filename.c_str())) {
1212                        throw RIFF::Exception("Could not replace original file with temporary file (unable to remove original file).");
1213                    }
1214                    #else // POSIX ...
1215                    if (unlink(filename.c_str())) {
1216                        throw RIFF::Exception("Could not replace original file with temporary file (unable to remove original file): " + gig::String(strerror(errno)));
1217                    }
1218                    #endif
1219                    if (rename(tmpname.c_str(), filename.c_str())) {
1220                        #if defined(WIN32)
1221                        throw RIFF::Exception("Could not replace original file with temporary file (unable to rename temp file).");
1222                        #else
1223                        throw RIFF::Exception("Could not replace original file with temporary file (unable to rename temp file): " + gig::String(strerror(errno)));
1224                        #endif
1225                    }
1226                }
1227            } else {
1228                gig->Save(filename, &progress);
1229            }
1230    
1231            printf("End\n");
1232            finished_dispatcher.emit();
1233        } catch (RIFF::Exception e) {
1234            error_message = e.Message;
1235            error_dispatcher.emit();
1236        } catch (...) {
1237            error_message = _("Unknown exception occurred");
1238            error_dispatcher.emit();
1239        }
1240    }
1241    
1242    Saver::Saver(gig::File* file, Glib::ustring filename)
1243        : gig(file), filename(filename), thread(0), progress(0.f)
1244    {
1245    }
1246    
1247    void Saver::launch()
1248    {
1249    #ifdef OLD_THREADS
1250        thread = Glib::Thread::create(sigc::mem_fun(*this, &Saver::thread_function), true);
1251    #else
1252        thread = Glib::Threads::Thread::create(sigc::mem_fun(*this, &Saver::thread_function));
1253    #endif
1254        printf("launch thread=%p\n", static_cast<void*>(thread));
1255    }
1256    
1257    float Saver::get_progress()
1258    {
1259        float res;
1260        {
1261            Glib::Threads::Mutex::Lock lock(progressMutex);
1262            res = progress;
1263        }
1264        return res;
1265    }
1266    
1267    Glib::Dispatcher& Saver::signal_progress()
1268    {
1269        return progress_dispatcher;
1270    }
1271    
1272    Glib::Dispatcher& Saver::signal_finished()
1273    {
1274        return finished_dispatcher;
1275    }
1276    
1277    Glib::Dispatcher& Saver::signal_error()
1278    {
1279        return error_dispatcher;
1280    }
1281    
1282    ProgressDialog::ProgressDialog(const Glib::ustring& title, Gtk::Window& parent)
1283      : Gtk::Dialog(title, parent, true)      : Gtk::Dialog(title, parent, true)
1284  {  {
1285      get_vbox()->pack_start(progressBar);      get_vbox()->pack_start(progressBar);
1286      show_all_children();      show_all_children();
1287        resize(600,50);
1288  }  }
1289    
1290  // Clear all GUI elements / controls. This method is typically called  // Clear all GUI elements / controls. This method is typically called
# Line 523  void MainWindow::__clear() { Line 1295  void MainWindow::__clear() {
1295      // clear the samples and instruments tree views      // clear the samples and instruments tree views
1296      m_refTreeModel->clear();      m_refTreeModel->clear();
1297      m_refSamplesTreeModel->clear();      m_refSamplesTreeModel->clear();
1298        m_refScriptsTreeModel->clear();
1299      // remove all entries from "Instrument" menu      // remove all entries from "Instrument" menu
1300      while (!instrument_menu->get_children().empty()) {      while (!instrument_menu->get_children().empty()) {
1301          remove_instrument_from_menu(0);          remove_instrument_from_menu(0);
# Line 533  void MainWindow::__clear() { Line 1306  void MainWindow::__clear() {
1306      set_file_is_shared(false);      set_file_is_shared(false);
1307  }  }
1308    
1309    void MainWindow::__refreshEntireGUI() {
1310        // clear the samples and instruments tree views
1311        m_refTreeModel->clear();
1312        m_refSamplesTreeModel->clear();
1313        m_refScriptsTreeModel->clear();
1314        // remove all entries from "Instrument" menu
1315        while (!instrument_menu->get_children().empty()) {
1316            remove_instrument_from_menu(0);
1317        }
1318    
1319        if (!this->file) return;
1320    
1321        load_gig(
1322            this->file, this->file->pInfo->Name.c_str(), this->file_is_shared
1323        );
1324    }
1325    
1326  void MainWindow::on_action_file_new()  void MainWindow::on_action_file_new()
1327  {  {
1328      if (!file_is_shared && file_is_changed && !close_confirmation_dialog()) return;      if (!file_is_shared && file_is_changed && !close_confirmation_dialog()) return;
# Line 563  bool MainWindow::close_confirmation_dial Line 1353  bool MainWindow::close_confirmation_dial
1353      dialog.set_default_response(Gtk::RESPONSE_YES);      dialog.set_default_response(Gtk::RESPONSE_YES);
1354      int response = dialog.run();      int response = dialog.run();
1355      dialog.hide();      dialog.hide();
1356      if (response == Gtk::RESPONSE_YES) return file_save();  
1357      return response != Gtk::RESPONSE_CANCEL;      // user decided to exit app without saving
1358        if (response == Gtk::RESPONSE_NO) return true;
1359    
1360        // user cancelled dialog, thus don't close app
1361        if (response == Gtk::RESPONSE_CANCEL) return false;
1362    
1363        // TODO: the following return valid is disabled and hard coded instead for
1364        // now, due to the fact that saving with progress bar is now implemented
1365        // asynchronously, as a result the app does not close automatically anymore
1366        // after saving the file has completed
1367        //
1368        //   if (response == Gtk::RESPONSE_YES) return file_save();
1369        //   return response != Gtk::RESPONSE_CANCEL;
1370        //
1371        if (response == Gtk::RESPONSE_YES) file_save();
1372        return false; // always prevent closing the app for now (see comment above)
1373  }  }
1374    
1375  bool MainWindow::leaving_shared_mode_dialog() {  bool MainWindow::leaving_shared_mode_dialog() {
# Line 606  void MainWindow::on_action_file_open() Line 1411  void MainWindow::on_action_file_open()
1411      if (dialog.run() == Gtk::RESPONSE_OK) {      if (dialog.run() == Gtk::RESPONSE_OK) {
1412          std::string filename = dialog.get_filename();          std::string filename = dialog.get_filename();
1413          printf("filename=%s\n", filename.c_str());          printf("filename=%s\n", filename.c_str());
1414          printf("on_action_file_open self=%x\n", Glib::Threads::Thread::self());          printf("on_action_file_open self=%p\n",
1415                   static_cast<void*>(Glib::Threads::Thread::self()));
1416          load_file(filename.c_str());          load_file(filename.c_str());
1417          current_gig_dir = Glib::path_get_dirname(filename);          current_gig_dir = Glib::path_get_dirname(filename);
1418      }      }
# Line 615  void MainWindow::on_action_file_open() Line 1421  void MainWindow::on_action_file_open()
1421  void MainWindow::load_file(const char* name)  void MainWindow::load_file(const char* name)
1422  {  {
1423      __clear();      __clear();
1424      load_dialog = new LoadDialog(_("Loading..."), *this);  
1425      load_dialog->show_all();      progress_dialog = new ProgressDialog( //FIXME: memory leak!
1426      loader = new Loader(strdup(name));          _("Loading") +  Glib::ustring(" '") +
1427            Glib::filename_display_basename(name) + "' ...",
1428            *this
1429        );
1430        progress_dialog->show_all();
1431        loader = new Loader(name); //FIXME: memory leak!
1432      loader->signal_progress().connect(      loader->signal_progress().connect(
1433          sigc::mem_fun(*this, &MainWindow::on_loader_progress));          sigc::mem_fun(*this, &MainWindow::on_loader_progress));
1434      loader->signal_finished().connect(      loader->signal_finished().connect(
1435          sigc::mem_fun(*this, &MainWindow::on_loader_finished));          sigc::mem_fun(*this, &MainWindow::on_loader_finished));
1436        loader->signal_error().connect(
1437            sigc::mem_fun(*this, &MainWindow::on_loader_error));
1438      loader->launch();      loader->launch();
1439  }  }
1440    
# Line 637  void MainWindow::load_instrument(gig::In Line 1450  void MainWindow::load_instrument(gig::In
1450      // load the instrument      // load the instrument
1451      gig::File* pFile = (gig::File*) instr->GetParent();      gig::File* pFile = (gig::File*) instr->GetParent();
1452      load_gig(pFile, 0 /*file name*/, true /*shared instrument*/);      load_gig(pFile, 0 /*file name*/, true /*shared instrument*/);
1453      //TODO: automatically select the given instrument      // automatically select the given instrument
1454        int i = 0;
1455        for (gig::Instrument* instrument = pFile->GetFirstInstrument(); instrument;
1456             instrument = pFile->GetNextInstrument(), ++i)
1457        {
1458            if (instrument == instr) {
1459                // select item in "instruments" tree view
1460                m_TreeView.get_selection()->select(Gtk::TreePath(ToString(i)));
1461                // make sure the selected item in the "instruments" tree view is
1462                // visible (scroll to it)
1463                m_TreeView.scroll_to_row(Gtk::TreePath(ToString(i)));
1464                // select item in instrument menu
1465                {
1466                    const std::vector<Gtk::Widget*> children =
1467                        instrument_menu->get_children();
1468                    static_cast<Gtk::RadioMenuItem*>(children[i])->set_active();
1469                }
1470                // update region chooser and dimension region chooser
1471                m_RegionChooser.set_instrument(instr);
1472                break;
1473            }
1474        }
1475  }  }
1476    
1477  void MainWindow::on_loader_progress()  void MainWindow::on_loader_progress()
1478  {  {
1479      load_dialog->set_fraction(loader->get_progress());      progress_dialog->set_fraction(loader->get_progress());
1480  }  }
1481    
1482  void MainWindow::on_loader_finished()  void MainWindow::on_loader_finished()
1483  {  {
1484      printf("Loader finished!\n");      printf("Loader finished!\n");
1485      printf("on_loader_finished self=%x\n", Glib::Threads::Thread::self());      printf("on_loader_finished self=%p\n",
1486      load_gig(loader->gig, loader->filename);             static_cast<void*>(Glib::Threads::Thread::self()));
1487      load_dialog->hide();      load_gig(loader->gig, loader->filename.c_str());
1488        progress_dialog->hide();
1489    }
1490    
1491    void MainWindow::on_loader_error()
1492    {
1493        Glib::ustring txt = _("Could not load file: ") + loader->error_message;
1494        Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
1495        msg.run();
1496        progress_dialog->hide();
1497  }  }
1498    
1499  void MainWindow::on_action_file_save()  void MainWindow::on_action_file_save()
# Line 689  bool MainWindow::file_save() Line 1532  bool MainWindow::file_save()
1532    
1533      std::cout << "Saving file\n" << std::flush;      std::cout << "Saving file\n" << std::flush;
1534      file_structure_to_be_changed_signal.emit(this->file);      file_structure_to_be_changed_signal.emit(this->file);
1535      try {  
1536          file->Save();      progress_dialog = new ProgressDialog( //FIXME: memory leak!
1537          if (file_is_changed) {          _("Saving") +  Glib::ustring(" '") +
1538              set_title(get_title().substr(1));          Glib::filename_display_basename(this->filename) + "' ...",
1539              file_is_changed = false;          *this
1540          }      );
1541      } catch (RIFF::Exception e) {      progress_dialog->show_all();
1542          file_structure_changed_signal.emit(this->file);      saver = new Saver(this->file); //FIXME: memory leak!
1543          Glib::ustring txt = _("Could not save file: ") + e.Message;      saver->signal_progress().connect(
1544          Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);          sigc::mem_fun(*this, &MainWindow::on_saver_progress));
1545          msg.run();      saver->signal_finished().connect(
1546          return false;          sigc::mem_fun(*this, &MainWindow::on_saver_finished));
1547      }      saver->signal_error().connect(
1548      std::cout << "Saving file done\n" << std::flush;          sigc::mem_fun(*this, &MainWindow::on_saver_error));
1549        saver->launch();
1550    
1551        return true;
1552    }
1553    
1554    void MainWindow::on_saver_progress()
1555    {
1556        progress_dialog->set_fraction(saver->get_progress());
1557    }
1558    
1559    void MainWindow::on_saver_error()
1560    {
1561        file_structure_changed_signal.emit(this->file);
1562        Glib::ustring txt = _("Could not save file: ") + saver->error_message;
1563        Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
1564        msg.run();
1565    }
1566    
1567    void MainWindow::on_saver_finished()
1568    {
1569        this->file = saver->gig;
1570        this->filename = saver->filename;
1571        current_gig_dir = Glib::path_get_dirname(filename);
1572        set_title(Glib::filename_display_basename(filename));
1573        file_has_name = true;
1574        file_is_changed = false;
1575        std::cout << "Saving file done. Importing queued samples now ...\n" << std::flush;
1576      __import_queued_samples();      __import_queued_samples();
1577        std::cout << "Importing queued samples done.\n" << std::flush;
1578    
1579      file_structure_changed_signal.emit(this->file);      file_structure_changed_signal.emit(this->file);
1580      return true;  
1581        __refreshEntireGUI();
1582        progress_dialog->hide();
1583  }  }
1584    
1585  void MainWindow::on_action_file_save_as()  void MainWindow::on_action_file_save_as()
# Line 749  bool MainWindow::file_save_as() Line 1623  bool MainWindow::file_save_as()
1623      // show warning in the dialog      // show warning in the dialog
1624      Gtk::HBox descriptionArea;      Gtk::HBox descriptionArea;
1625      descriptionArea.set_spacing(15);      descriptionArea.set_spacing(15);
1626      Gtk::Image warningIcon(Gtk::Stock::DIALOG_WARNING, Gtk::IconSize(Gtk::ICON_SIZE_DIALOG));      Gtk::Image warningIcon;
1627        warningIcon.set_from_icon_name("dialog-warning",
1628                                       Gtk::IconSize(Gtk::ICON_SIZE_DIALOG));
1629      descriptionArea.pack_start(warningIcon, Gtk::PACK_SHRINK);      descriptionArea.pack_start(warningIcon, Gtk::PACK_SHRINK);
1630  #if GTKMM_MAJOR_VERSION < 3  #if GTKMM_MAJOR_VERSION < 3
1631      view::WrapLabel description;      view::WrapLabel description;
# Line 770  bool MainWindow::file_save_as() Line 1646  bool MainWindow::file_save_as()
1646      descriptionArea.show_all();      descriptionArea.show_all();
1647    
1648      if (dialog.run() == Gtk::RESPONSE_OK) {      if (dialog.run() == Gtk::RESPONSE_OK) {
1649          file_structure_to_be_changed_signal.emit(this->file);          std::string filename = dialog.get_filename();
1650          try {          if (!Glib::str_has_suffix(filename, ".gig")) {
1651              std::string filename = dialog.get_filename();              filename += ".gig";
             if (!Glib::str_has_suffix(filename, ".gig")) {  
                 filename += ".gig";  
             }  
             printf("filename=%s\n", filename.c_str());  
             file->Save(filename);  
             this->filename = filename;  
             current_gig_dir = Glib::path_get_dirname(filename);  
             set_title(Glib::filename_display_basename(filename));  
             file_has_name = true;  
             file_is_changed = false;  
         } catch (RIFF::Exception e) {  
             file_structure_changed_signal.emit(this->file);  
             Glib::ustring txt = _("Could not save file: ") + e.Message;  
             Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);  
             msg.run();  
             return false;  
1652          }          }
1653          __import_queued_samples();          printf("filename=%s\n", filename.c_str());
1654          file_structure_changed_signal.emit(this->file);  
1655            progress_dialog = new ProgressDialog( //FIXME: memory leak!
1656                _("Saving") +  Glib::ustring(" '") +
1657                Glib::filename_display_basename(filename) + "' ...",
1658                *this
1659            );
1660            progress_dialog->show_all();
1661    
1662            saver = new Saver(file, filename); //FIXME: memory leak!
1663            saver->signal_progress().connect(
1664                sigc::mem_fun(*this, &MainWindow::on_saver_progress));
1665            saver->signal_finished().connect(
1666                sigc::mem_fun(*this, &MainWindow::on_saver_finished));
1667            saver->signal_error().connect(
1668                sigc::mem_fun(*this, &MainWindow::on_saver_error));
1669            saver->launch();
1670    
1671          return true;          return true;
1672      }      }
1673      return false;      return false;
# Line 801  bool MainWindow::file_save_as() Line 1677  bool MainWindow::file_save_as()
1677  void MainWindow::__import_queued_samples() {  void MainWindow::__import_queued_samples() {
1678      std::cout << "Starting sample import\n" << std::flush;      std::cout << "Starting sample import\n" << std::flush;
1679      Glib::ustring error_files;      Glib::ustring error_files;
1680      printf("Samples to import: %d\n", m_SampleImportQueue.size());      printf("Samples to import: %d\n", int(m_SampleImportQueue.size()));
1681      for (std::list<SampleImportItem>::iterator iter = m_SampleImportQueue.begin();      for (std::map<gig::Sample*, SampleImportItem>::iterator iter = m_SampleImportQueue.begin();
1682           iter != m_SampleImportQueue.end(); ) {           iter != m_SampleImportQueue.end(); ) {
1683          printf("Importing sample %s\n",(*iter).sample_path.c_str());          printf("Importing sample %s\n",iter->second.sample_path.c_str());
1684          SF_INFO info;          SF_INFO info;
1685          info.format = 0;          info.format = 0;
1686          SNDFILE* hFile = sf_open((*iter).sample_path.c_str(), SFM_READ, &info);          SNDFILE* hFile = sf_open(iter->second.sample_path.c_str(), SFM_READ, &info);
1687          sf_command(hFile, SFC_SET_SCALE_FLOAT_INT_READ, 0, SF_TRUE);          sf_command(hFile, SFC_SET_SCALE_FLOAT_INT_READ, 0, SF_TRUE);
1688          try {          try {
1689              if (!hFile) throw std::string(_("could not open file"));              if (!hFile) throw std::string(_("could not open file"));
# Line 830  void MainWindow::__import_queued_samples Line 1706  void MainWindow::__import_queued_samples
1706                      throw std::string(_("format not supported")); // unsupported subformat (yet?)                      throw std::string(_("format not supported")); // unsupported subformat (yet?)
1707              }              }
1708    
1709                // reset write position for sample
1710                iter->first->SetPos(0);
1711    
1712              const int bufsize = 10000;              const int bufsize = 10000;
1713              switch (bitdepth) {              switch (bitdepth) {
1714                  case 16: {                  case 16: {
# Line 839  void MainWindow::__import_queued_samples Line 1718  void MainWindow::__import_queued_samples
1718                          // libsndfile does the conversion for us (if needed)                          // libsndfile does the conversion for us (if needed)
1719                          int n = sf_readf_short(hFile, buffer, bufsize);                          int n = sf_readf_short(hFile, buffer, bufsize);
1720                          // write from buffer directly (physically) into .gig file                          // write from buffer directly (physically) into .gig file
1721                          iter->gig_sample->Write(buffer, n);                          iter->first->Write(buffer, n);
1722                          cnt -= n;                          cnt -= n;
1723                      }                      }
1724                      delete[] buffer;                      delete[] buffer;
# Line 859  void MainWindow::__import_queued_samples Line 1738  void MainWindow::__import_queued_samples
1738                              dstbuf[j++] = srcbuf[i] >> 24;                              dstbuf[j++] = srcbuf[i] >> 24;
1739                          }                          }
1740                          // write from buffer directly (physically) into .gig file                          // write from buffer directly (physically) into .gig file
1741                          iter->gig_sample->Write(dstbuf, n);                          iter->first->Write(dstbuf, n);
1742                          cnt -= n;                          cnt -= n;
1743                      }                      }
1744                      delete[] srcbuf;                      delete[] srcbuf;
# Line 870  void MainWindow::__import_queued_samples Line 1749  void MainWindow::__import_queued_samples
1749              // cleanup              // cleanup
1750              sf_close(hFile);              sf_close(hFile);
1751              // let the sampler re-cache the sample if needed              // let the sampler re-cache the sample if needed
1752              sample_changed_signal.emit(iter->gig_sample);              sample_changed_signal.emit(iter->first);
1753              // on success we remove the sample from the import queue,              // on success we remove the sample from the import queue,
1754              // otherwise keep it, maybe it works the next time ?              // otherwise keep it, maybe it works the next time ?
1755              std::list<SampleImportItem>::iterator cur = iter;              std::map<gig::Sample*, SampleImportItem>::iterator cur = iter;
1756              ++iter;              ++iter;
1757              m_SampleImportQueue.erase(cur);              m_SampleImportQueue.erase(cur);
1758          } catch (std::string what) {          } catch (std::string what) {
1759              // remember the files that made trouble (and their cause)              // remember the files that made trouble (and their cause)
1760              if (!error_files.empty()) error_files += "\n";              if (!error_files.empty()) error_files += "\n";
1761              error_files += (*iter).sample_path += " (" + what + ")";              error_files += iter->second.sample_path += " (" + what + ")";
1762              ++iter;              ++iter;
1763          }          }
1764      }      }
# Line 897  void MainWindow::on_action_file_properti Line 1776  void MainWindow::on_action_file_properti
1776      propDialog.deiconify();      propDialog.deiconify();
1777  }  }
1778    
1779    void MainWindow::on_action_warn_user_on_extensions() {
1780        Settings::singleton()->warnUserOnExtensions =
1781            !Settings::singleton()->warnUserOnExtensions;
1782    }
1783    
1784    void MainWindow::on_action_sync_sampler_instrument_selection() {
1785        Settings::singleton()->syncSamplerInstrumentSelection =
1786            !Settings::singleton()->syncSamplerInstrumentSelection;
1787    }
1788    
1789    void MainWindow::on_action_move_root_note_with_region_moved() {
1790        Settings::singleton()->moveRootNoteWithRegionMoved =
1791            !Settings::singleton()->moveRootNoteWithRegionMoved;
1792    }
1793    
1794  void MainWindow::on_action_help_about()  void MainWindow::on_action_help_about()
1795  {  {
1796      Gtk::AboutDialog dialog;      Gtk::AboutDialog dialog;
# Line 906  void MainWindow::on_action_help_about() Line 1800  void MainWindow::on_action_help_about()
1800      dialog.set_name("Gigedit");      dialog.set_name("Gigedit");
1801  #endif  #endif
1802      dialog.set_version(VERSION);      dialog.set_version(VERSION);
1803      dialog.set_copyright("Copyright (C) 2006-2013 Andreas Persson");      dialog.set_copyright("Copyright (C) 2006-2017 Andreas Persson");
1804      dialog.set_comments(_(      const std::string sComment =
1805          "Released under the GNU General Public License.\n"          _("Built " __DATE__ "\nUsing ") +
1806          "\n"          ::gig::libraryName() + " " + ::gig::libraryVersion() + "\n\n" +
1807          "Please notice that this is still a very young instrument editor. "          _(
1808          "So better backup your Gigasampler files before editing them with "              "Gigedit is released under the GNU General Public License.\n"
1809          "this application.\n"              "\n"
1810          "\n"              "This program is distributed WITHOUT ANY WARRANTY; So better "
1811          "Please report bugs to: http://bugs.linuxsampler.org")              "backup your Gigasampler/GigaStudio files before editing them with "
1812      );              "this application.\n"
1813                "\n"
1814                "Please report bugs to: http://bugs.linuxsampler.org"
1815            );
1816        dialog.set_comments(sComment.c_str());
1817      dialog.set_website("http://www.linuxsampler.org");      dialog.set_website("http://www.linuxsampler.org");
1818      dialog.set_website_label("http://www.linuxsampler.org");      dialog.set_website_label("http://www.linuxsampler.org");
1819      dialog.run();      dialog.run();
1820  }  }
1821    
1822  PropDialog::PropDialog()  PropDialog::PropDialog()
1823      : eName(_("Name")),      : eFileFormat(_("File Format")),
1824          eName(_("Name")),
1825        eCreationDate(_("Creation date")),        eCreationDate(_("Creation date")),
1826        eComments(_("Comments")),        eComments(_("Comments")),
1827        eProduct(_("Product")),        eProduct(_("Product")),
# Line 939  PropDialog::PropDialog() Line 1838  PropDialog::PropDialog()
1838        eCommissioned(_("Commissioned")),        eCommissioned(_("Commissioned")),
1839        eSubject(_("Subject")),        eSubject(_("Subject")),
1840        quitButton(Gtk::Stock::CLOSE),        quitButton(Gtk::Stock::CLOSE),
1841        table(2, 1)        table(2, 1),
1842          m_file(NULL)
1843  {  {
1844      set_title(_("File Properties"));      set_title(_("File Properties"));
1845      eName.set_width_chars(50);      eName.set_width_chars(50);
# Line 961  PropDialog::PropDialog() Line 1861  PropDialog::PropDialog()
1861      connect(eCommissioned, &DLS::Info::Commissioned);      connect(eCommissioned, &DLS::Info::Commissioned);
1862      connect(eSubject, &DLS::Info::Subject);      connect(eSubject, &DLS::Info::Subject);
1863    
1864        table.add(eFileFormat);
1865      table.add(eName);      table.add(eName);
1866      table.add(eCreationDate);      table.add(eCreationDate);
1867      table.add(eComments);      table.add(eComments);
# Line 991  PropDialog::PropDialog() Line 1892  PropDialog::PropDialog()
1892      quitButton.grab_focus();      quitButton.grab_focus();
1893      quitButton.signal_clicked().connect(      quitButton.signal_clicked().connect(
1894          sigc::mem_fun(*this, &PropDialog::hide));          sigc::mem_fun(*this, &PropDialog::hide));
1895        eFileFormat.signal_value_changed().connect(
1896            sigc::mem_fun(*this, &PropDialog::onFileFormatChanged));
1897    
1898      quitButton.show();      quitButton.show();
1899      vbox.show();      vbox.show();
1900      show_all_children();      show_all_children();
1901  }  }
1902    
1903    void PropDialog::set_file(gig::File* file)
1904    {
1905        m_file = file;
1906    
1907        // update file format version combo box
1908        const std::string sGiga = "Gigasampler/GigaStudio v";
1909        const int major = file->pVersion->major;
1910        std::vector<std::string> txts;
1911        std::vector<int> values;
1912        txts.push_back(sGiga + "2"); values.push_back(2);
1913        txts.push_back(sGiga + "3/v4"); values.push_back(3);
1914        if (major != 2 && major != 3) {
1915            txts.push_back(sGiga + ToString(major)); values.push_back(major);
1916        }
1917        std::vector<const char*> texts;
1918        for (int i = 0; i < txts.size(); ++i) texts.push_back(txts[i].c_str());
1919        texts.push_back(NULL); values.push_back(0);
1920        eFileFormat.set_choices(&texts[0], &values[0]);
1921        eFileFormat.set_value(major);
1922    }
1923    
1924    void PropDialog::onFileFormatChanged() {
1925        const int major = eFileFormat.get_value();
1926        if (m_file) m_file->pVersion->major = major;
1927    }
1928    
1929  void PropDialog::set_info(DLS::Info* info)  void PropDialog::set_info(DLS::Info* info)
1930  {  {
1931      update(info);      update(info);
# Line 1128  void MainWindow::file_changed() Line 2057  void MainWindow::file_changed()
2057      }      }
2058  }  }
2059    
2060    void MainWindow::updateSampleRefCountMap(gig::File* gig) {
2061        sample_ref_count.clear();
2062        
2063        if (!gig) return;
2064    
2065        for (gig::Instrument* instrument = gig->GetFirstInstrument(); instrument;
2066             instrument = gig->GetNextInstrument())
2067        {
2068            for (gig::Region* rgn = instrument->GetFirstRegion(); rgn;
2069                 rgn = instrument->GetNextRegion())
2070            {
2071                for (int i = 0; i < 256; ++i) {
2072                    if (!rgn->pDimensionRegions[i]) continue;
2073                    if (rgn->pDimensionRegions[i]->pSample) {
2074                        sample_ref_count[rgn->pDimensionRegions[i]->pSample]++;
2075                    }
2076                }
2077            }
2078        }
2079    }
2080    
2081  void MainWindow::load_gig(gig::File* gig, const char* filename, bool isSharedInstrument)  void MainWindow::load_gig(gig::File* gig, const char* filename, bool isSharedInstrument)
2082  {  {
2083      file = 0;      file = 0;
2084      set_file_is_shared(isSharedInstrument);      set_file_is_shared(isSharedInstrument);
2085    
2086      this->filename = filename ? filename : _("Unsaved Gig File");      this->filename =
2087            (filename && strlen(filename) > 0) ?
2088                filename : (!gig->GetFileName().empty()) ?
2089                    gig->GetFileName() : _("Unsaved Gig File");
2090      set_title(Glib::filename_display_basename(this->filename));      set_title(Glib::filename_display_basename(this->filename));
2091      file_has_name = filename;      file_has_name = filename;
2092      file_is_changed = false;      file_is_changed = false;
2093    
2094        propDialog.set_file(gig);
2095      propDialog.set_info(gig->pInfo);      propDialog.set_info(gig->pInfo);
2096    
2097      instrument_name_connection.block();      instrument_name_connection.block();
2098        int index = 0;
2099      for (gig::Instrument* instrument = gig->GetFirstInstrument() ; instrument ;      for (gig::Instrument* instrument = gig->GetFirstInstrument() ; instrument ;
2100           instrument = gig->GetNextInstrument()) {           instrument = gig->GetNextInstrument(), ++index) {
2101          Glib::ustring name(gig_to_utf8(instrument->pInfo->Name));          Glib::ustring name(gig_to_utf8(instrument->pInfo->Name));
2102            const int iScriptSlots = instrument->ScriptSlotCount();
2103    
2104          Gtk::TreeModel::iterator iter = m_refTreeModel->append();          Gtk::TreeModel::iterator iter = m_refTreeModel->append();
2105          Gtk::TreeModel::Row row = *iter;          Gtk::TreeModel::Row row = *iter;
2106            row[m_Columns.m_col_nr] = index;
2107          row[m_Columns.m_col_name] = name;          row[m_Columns.m_col_name] = name;
2108          row[m_Columns.m_col_instr] = instrument;          row[m_Columns.m_col_instr] = instrument;
2109            row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";
2110    
2111          add_instrument_to_menu(name);          add_instrument_to_menu(name);
2112      }      }
2113      instrument_name_connection.unblock();      instrument_name_connection.unblock();
2114      uiManager->get_widget("/MenuBar/MenuInstrument")->show();      uiManager->get_widget("/MenuBar/MenuInstrument/AllInstruments")->show();
2115    
2116        updateSampleRefCountMap(gig);
2117    
2118      for (gig::Group* group = gig->GetFirstGroup(); group; group = gig->GetNextGroup()) {      for (gig::Group* group = gig->GetFirstGroup(); group; group = gig->GetNextGroup()) {
2119          if (group->Name != "") {          if (group->Name != "") {
# Line 1171  void MainWindow::load_gig(gig::File* gig Line 2131  void MainWindow::load_gig(gig::File* gig
2131                      gig_to_utf8(sample->pInfo->Name);                      gig_to_utf8(sample->pInfo->Name);
2132                  rowSample[m_SamplesModel.m_col_sample] = sample;                  rowSample[m_SamplesModel.m_col_sample] = sample;
2133                  rowSample[m_SamplesModel.m_col_group]  = NULL;                  rowSample[m_SamplesModel.m_col_group]  = NULL;
2134                    int refcount = sample_ref_count.count(sample) ? sample_ref_count[sample] : 0;
2135                    rowSample[m_SamplesModel.m_col_refcount] = ToString(refcount) + " " + _("Refs.");
2136                    rowSample[m_SamplesModel.m_color] = refcount ? "black" : "red";
2137              }              }
2138          }          }
2139      }      }
2140        
2141        for (int i = 0; gig->GetScriptGroup(i); ++i) {
2142            gig::ScriptGroup* group = gig->GetScriptGroup(i);
2143    
2144            Gtk::TreeModel::iterator iterGroup = m_refScriptsTreeModel->append();
2145            Gtk::TreeModel::Row rowGroup = *iterGroup;
2146            rowGroup[m_ScriptsModel.m_col_name]   = gig_to_utf8(group->Name);
2147            rowGroup[m_ScriptsModel.m_col_group]  = group;
2148            rowGroup[m_ScriptsModel.m_col_script] = NULL;
2149            for (int s = 0; group->GetScript(s); ++s) {
2150                gig::Script* script = group->GetScript(s);
2151    
2152                Gtk::TreeModel::iterator iterScript =
2153                    m_refScriptsTreeModel->append(rowGroup.children());
2154                Gtk::TreeModel::Row rowScript = *iterScript;
2155                rowScript[m_ScriptsModel.m_col_name] = gig_to_utf8(script->Name);
2156                rowScript[m_ScriptsModel.m_col_script] = script;
2157                rowScript[m_ScriptsModel.m_col_group]  = NULL;
2158            }
2159        }
2160        // unfold all sample groups & script groups by default
2161        m_TreeViewSamples.expand_all();
2162        m_TreeViewScripts.expand_all();
2163    
2164      file = gig;      file = gig;
2165    
# Line 1181  void MainWindow::load_gig(gig::File* gig Line 2167  void MainWindow::load_gig(gig::File* gig
2167      m_TreeView.get_selection()->select(Gtk::TreePath("0"));      m_TreeView.get_selection()->select(Gtk::TreePath("0"));
2168    
2169      instr_props_set_instrument();      instr_props_set_instrument();
2170        gig::Instrument* instrument = get_instrument();
2171        if (instrument) {
2172            midiRules.set_instrument(instrument);
2173        }
2174  }  }
2175    
2176  bool MainWindow::instr_props_set_instrument()  bool MainWindow::instr_props_set_instrument()
2177  {  {
2178      instrumentProps.signal_name_changed().clear();      instrumentProps.signal_name_changed().clear();
2179    
2180      Gtk::TreeModel::const_iterator it =      std::vector<Gtk::TreeModel::Path> rows = m_TreeView.get_selection()->get_selected_rows();
2181          m_TreeView.get_selection()->get_selected();      if (rows.empty()) {
2182            instrumentProps.hide();
2183            return false;
2184        }
2185        Gtk::TreeModel::const_iterator it = m_refTreeModel->get_iter(rows[0]);
2186      if (it) {      if (it) {
2187          Gtk::TreeModel::Row row = *it;          Gtk::TreeModel::Row row = *it;
2188          gig::Instrument* instrument = row[m_Columns.m_col_instr];          gig::Instrument* instrument = row[m_Columns.m_col_instr];
# Line 1228  void MainWindow::instr_name_changed_by_i Line 2222  void MainWindow::instr_name_changed_by_i
2222      }      }
2223  }  }
2224    
2225    void MainWindow::show_midi_rules()
2226    {
2227        if (gig::Instrument* instrument = get_instrument())
2228        {
2229            midiRules.set_instrument(instrument);
2230            midiRules.show();
2231            midiRules.deiconify();
2232        }
2233    }
2234    
2235    void MainWindow::show_script_slots() {
2236        if (!file) return;
2237        // get selected instrument
2238        std::vector<Gtk::TreeModel::Path> rows = m_TreeView.get_selection()->get_selected_rows();
2239        if (rows.empty()) return;
2240        Gtk::TreeModel::iterator it = m_refTreeModel->get_iter(rows[0]);
2241        if (!it) return;
2242        Gtk::TreeModel::Row row = *it;
2243        gig::Instrument* instrument = row[m_Columns.m_col_instr];
2244        if (!instrument) return;
2245    
2246        ScriptSlots* window = new ScriptSlots;
2247        window->setInstrument(instrument);
2248        window->signal_script_slots_changed().connect(
2249            sigc::mem_fun(*this, &MainWindow::onScriptSlotsModified)
2250        );
2251        //window->reparent(*this);
2252        window->show();
2253    }
2254    
2255    void MainWindow::onScriptSlotsModified(gig::Instrument* pInstrument) {
2256        if (!pInstrument) return;
2257        const int iScriptSlots = pInstrument->ScriptSlotCount();
2258    
2259        Glib::RefPtr<Gtk::TreeModel> model = m_TreeView.get_model();
2260        for (int i = 0; i < model->children().size(); ++i) {
2261            Gtk::TreeModel::Row row = model->children()[i];
2262            if (row[m_Columns.m_col_instr] != pInstrument) continue;
2263            row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";
2264            break;
2265        }
2266    }
2267    
2268    void MainWindow::on_action_refresh_all() {
2269        __refreshEntireGUI();
2270    }
2271    
2272  void MainWindow::on_action_view_status_bar() {  void MainWindow::on_action_view_status_bar() {
2273      Gtk::CheckMenuItem* item =      Gtk::CheckMenuItem* item =
2274          dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuView/Statusbar"));          dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuView/Statusbar"));
# Line 1239  void MainWindow::on_action_view_status_b Line 2280  void MainWindow::on_action_view_status_b
2280      else                    m_StatusBar.hide();      else                    m_StatusBar.hide();
2281  }  }
2282    
2283    void MainWindow::on_auto_restore_win_dim() {
2284        Gtk::CheckMenuItem* item =
2285            dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuView/AutoRestoreWinDim"));
2286        if (!item) {
2287            std::cerr << "/MenuBar/MenuView/AutoRestoreWinDim == NULL\n";
2288            return;
2289        }
2290        Settings::singleton()->autoRestoreWindowDimension = item->get_active();
2291    }
2292    
2293    void MainWindow::on_save_with_temporary_file() {
2294        Gtk::CheckMenuItem* item =
2295            dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuSettings/SaveWithTemporaryFile"));
2296        if (!item) {
2297            std::cerr << "/MenuBar/MenuSettings/SaveWithTemporaryFile == NULL\n";
2298            return;
2299        }
2300        Settings::singleton()->saveWithTemporaryFile = item->get_active();
2301    }
2302    
2303    bool MainWindow::is_copy_samples_unity_note_enabled() const {
2304        Gtk::CheckMenuItem* item =
2305            dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuEdit/CopySampleUnity"));
2306        if (!item) {
2307            std::cerr << "/MenuBar/MenuEdit/CopySampleUnity == NULL\n";
2308            return true;
2309        }
2310        return item->get_active();
2311    }
2312    
2313    bool MainWindow::is_copy_samples_fine_tune_enabled() const {
2314        Gtk::CheckMenuItem* item =
2315            dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuEdit/CopySampleTune"));
2316        if (!item) {
2317            std::cerr << "/MenuBar/MenuEdit/CopySampleTune == NULL\n";
2318            return true;
2319        }
2320        return item->get_active();
2321    }
2322    
2323    bool MainWindow::is_copy_samples_loop_enabled() const {
2324        Gtk::CheckMenuItem* item =
2325            dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuEdit/CopySampleLoop"));
2326        if (!item) {
2327            std::cerr << "/MenuBar/MenuEdit/CopySampleLoop == NULL\n";
2328            return true;
2329        }
2330        return item->get_active();
2331    }
2332    
2333  void MainWindow::on_button_release(GdkEventButton* button)  void MainWindow::on_button_release(GdkEventButton* button)
2334  {  {
2335      if (button->type == GDK_2BUTTON_PRESS) {      if (button->type == GDK_2BUTTON_PRESS) {
2336          show_instr_props();          show_instr_props();
2337      } else if (button->type == GDK_BUTTON_PRESS && button->button == 3) {      } else if (button->type == GDK_BUTTON_PRESS && button->button == 3) {
2338            // gig v2 files have no midi rules
2339            const bool bEnabled = !(file->pVersion && file->pVersion->major == 2);
2340            static_cast<Gtk::MenuItem*>(
2341                uiManager->get_widget("/MenuBar/MenuInstrument/MidiRules"))->set_sensitive(
2342                    bEnabled
2343                );
2344            static_cast<Gtk::MenuItem*>(
2345                uiManager->get_widget("/PopupMenu/MidiRules"))->set_sensitive(
2346                    bEnabled
2347                );
2348          popup_menu->popup(button->button, button->time);          popup_menu->popup(button->button, button->time);
2349      }      }
2350  }  }
# Line 1263  void MainWindow::on_instrument_selection Line 2364  void MainWindow::on_instrument_selection
2364      }      }
2365  }  }
2366    
2367    void MainWindow::select_instrument(gig::Instrument* instrument) {
2368        if (!instrument) return;
2369    
2370        Glib::RefPtr<Gtk::TreeModel> model = m_TreeView.get_model();
2371        for (int i = 0; i < model->children().size(); ++i) {
2372            Gtk::TreeModel::Row row = model->children()[i];
2373            if (row[m_Columns.m_col_instr] == instrument) {
2374                // select and show the respective instrument in the list view
2375                show_intruments_tab();
2376                m_TreeView.get_selection()->unselect_all();
2377                m_TreeView.get_selection()->select(model->children()[i]);
2378                std::vector<Gtk::TreeModel::Path> rows =
2379                    m_TreeView.get_selection()->get_selected_rows();
2380                if (!rows.empty())
2381                    m_TreeView.scroll_to_row(rows[0]);
2382                on_sel_change(); // the regular instrument selection change callback
2383            }
2384        }
2385    }
2386    
2387    /// Returns true if requested dimension region was successfully selected and scrolled to in the list view, false on error.
2388    bool MainWindow::select_dimension_region(gig::DimensionRegion* dimRgn) {
2389        gig::Region* pRegion = (gig::Region*) dimRgn->GetParent();
2390        gig::Instrument* pInstrument = (gig::Instrument*) pRegion->GetParent();
2391    
2392        Glib::RefPtr<Gtk::TreeModel> model = m_TreeView.get_model();
2393        for (int i = 0; i < model->children().size(); ++i) {
2394            Gtk::TreeModel::Row row = model->children()[i];
2395            if (row[m_Columns.m_col_instr] == pInstrument) {
2396                // select and show the respective instrument in the list view
2397                show_intruments_tab();
2398                m_TreeView.get_selection()->unselect_all();
2399                m_TreeView.get_selection()->select(model->children()[i]);
2400                std::vector<Gtk::TreeModel::Path> rows =
2401                    m_TreeView.get_selection()->get_selected_rows();
2402                if (!rows.empty())
2403                    m_TreeView.scroll_to_row(rows[0]);
2404                on_sel_change(); // the regular instrument selection change callback
2405    
2406                // select respective region in the region selector
2407                m_RegionChooser.set_region(pRegion);
2408    
2409                // select and show the respective dimension region in the editor
2410                //update_dimregs();
2411                if (!m_DimRegionChooser.select_dimregion(dimRgn)) return false;
2412                //dimreg_edit.set_dim_region(dimRgn);
2413    
2414                return true;
2415            }
2416        }
2417    
2418        return false;
2419    }
2420    
2421    void MainWindow::select_sample(gig::Sample* sample) {
2422        Glib::RefPtr<Gtk::TreeModel> model = m_TreeViewSamples.get_model();
2423        for (int g = 0; g < model->children().size(); ++g) {
2424            Gtk::TreeModel::Row rowGroup = model->children()[g];
2425            for (int s = 0; s < rowGroup.children().size(); ++s) {
2426                Gtk::TreeModel::Row rowSample = rowGroup.children()[s];
2427                if (rowSample[m_SamplesModel.m_col_sample] == sample) {
2428                    show_samples_tab();
2429                    m_TreeViewSamples.get_selection()->unselect_all();
2430                    m_TreeViewSamples.get_selection()->select(rowGroup.children()[s]);
2431                    std::vector<Gtk::TreeModel::Path> rows =
2432                        m_TreeViewSamples.get_selection()->get_selected_rows();
2433                    if (rows.empty()) return;
2434                    m_TreeViewSamples.scroll_to_row(rows[0]);
2435                    return;
2436                }
2437            }
2438        }
2439    }
2440    
2441  void MainWindow::on_sample_treeview_button_release(GdkEventButton* button) {  void MainWindow::on_sample_treeview_button_release(GdkEventButton* button) {
2442      if (button->type == GDK_BUTTON_PRESS && button->button == 3) {      if (button->type == GDK_BUTTON_PRESS && button->button == 3) {
2443            // by default if Ctrl keys is pressed down, then a mouse right-click
2444            // does not select the respective row, so we must assure this
2445            // programmatically ...
2446            /*{
2447                Gtk::TreeModel::Path path;
2448                Gtk::TreeViewColumn* pColumn = NULL;
2449                int cellX, cellY;
2450                bool bSuccess = m_TreeViewSamples.get_path_at_pos(
2451                    (int)button->x, (int)button->y,
2452                    path, pColumn, cellX, cellY
2453                );
2454                if (bSuccess) {
2455                    if (m_TreeViewSamples.get_selection()->count_selected_rows() <= 0) {
2456                        printf("not selected !!!\n");
2457                        m_TreeViewSamples.get_selection()->select(path);
2458                    }
2459                }
2460            }*/
2461    
2462          Gtk::Menu* sample_popup =          Gtk::Menu* sample_popup =
2463              dynamic_cast<Gtk::Menu*>(uiManager->get_widget("/SamplePopupMenu"));              dynamic_cast<Gtk::Menu*>(uiManager->get_widget("/SamplePopupMenu"));
2464          // update enabled/disabled state of sample popup items          // update enabled/disabled state of sample popup items
2465          Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();          Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();
2466          Gtk::TreeModel::iterator it = sel->get_selected();          std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
2467          bool group_selected  = false;          const int n = rows.size();
2468          bool sample_selected = false;          int nGroups  = 0;
2469          if (it) {          int nSamples = 0;
2470            for (int r = 0; r < n; ++r) {
2471                Gtk::TreeModel::iterator it = m_refSamplesTreeModel->get_iter(rows[r]);
2472                if (!it) continue;
2473              Gtk::TreeModel::Row row = *it;              Gtk::TreeModel::Row row = *it;
2474              group_selected  = row[m_SamplesModel.m_col_group];              if (row[m_SamplesModel.m_col_group]) nGroups++;
2475              sample_selected = row[m_SamplesModel.m_col_sample];              if (row[m_SamplesModel.m_col_sample]) nSamples++;
2476          }          }
2477    
2478          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/SampleProperties"))->          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/SampleProperties"))->
2479              set_sensitive(group_selected || sample_selected);              set_sensitive(n == 1);
2480          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/AddSample"))->          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/AddSample"))->
2481              set_sensitive(group_selected || sample_selected);              set_sensitive(n);
2482          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/AddGroup"))->          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/AddGroup"))->
2483              set_sensitive(file);              set_sensitive(file);
2484            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/ShowSampleRefs"))->
2485                set_sensitive(nSamples == 1);
2486          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/RemoveSample"))->          dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/SamplePopupMenu/RemoveSample"))->
2487              set_sensitive(group_selected || sample_selected);              set_sensitive(n);
2488          // show sample popup          // show sample popup
2489          sample_popup->popup(button->button, button->time);          sample_popup->popup(button->button, button->time);
2490    
2491            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuSample/SampleProperties"))->
2492                set_sensitive(n == 1);
2493            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuSample/AddSample"))->
2494                set_sensitive(n);
2495            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuSample/AddGroup"))->
2496                set_sensitive(file);
2497            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuSample/ShowSampleRefs"))->
2498                set_sensitive(nSamples == 1);
2499            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuSample/RemoveSample"))->
2500                set_sensitive(n);
2501      }      }
2502  }  }
2503    
2504    void MainWindow::on_script_treeview_button_release(GdkEventButton* button) {
2505        if (button->type == GDK_BUTTON_PRESS && button->button == 3) {
2506            Gtk::Menu* script_popup =
2507                dynamic_cast<Gtk::Menu*>(uiManager->get_widget("/ScriptPopupMenu"));
2508            // update enabled/disabled state of sample popup items
2509            Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewScripts.get_selection();
2510            Gtk::TreeModel::iterator it = sel->get_selected();
2511            bool group_selected  = false;
2512            bool script_selected = false;
2513            if (it) {
2514                Gtk::TreeModel::Row row = *it;
2515                group_selected  = row[m_ScriptsModel.m_col_group];
2516                script_selected = row[m_ScriptsModel.m_col_script];
2517            }
2518            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/ScriptPopupMenu/AddScript"))->
2519                set_sensitive(group_selected || script_selected);
2520            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/ScriptPopupMenu/AddScriptGroup"))->
2521                set_sensitive(file);
2522            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/ScriptPopupMenu/EditScript"))->
2523                set_sensitive(script_selected);    
2524            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/ScriptPopupMenu/RemoveScript"))->
2525                set_sensitive(group_selected || script_selected);
2526            // show sample popup
2527            script_popup->popup(button->button, button->time);
2528    
2529            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuScript/AddScript"))->
2530                set_sensitive(group_selected || script_selected);
2531            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuScript/AddScriptGroup"))->
2532                set_sensitive(file);
2533            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuScript/EditScript"))->
2534                set_sensitive(script_selected);    
2535            dynamic_cast<Gtk::MenuItem*>(uiManager->get_widget("/MenuBar/MenuScript/RemoveScript"))->
2536                set_sensitive(group_selected || script_selected);
2537        }
2538    }
2539    
2540  Gtk::RadioMenuItem* MainWindow::add_instrument_to_menu(  Gtk::RadioMenuItem* MainWindow::add_instrument_to_menu(
2541      const Glib::ustring& name, int position) {      const Glib::ustring& name, int position) {
# Line 1330  void MainWindow::add_instrument(gig::Ins Line 2576  void MainWindow::add_instrument(gig::Ins
2576      instrument_name_connection.block();      instrument_name_connection.block();
2577      Gtk::TreeModel::iterator iterInstr = m_refTreeModel->append();      Gtk::TreeModel::iterator iterInstr = m_refTreeModel->append();
2578      Gtk::TreeModel::Row rowInstr = *iterInstr;      Gtk::TreeModel::Row rowInstr = *iterInstr;
2579        rowInstr[m_Columns.m_col_nr] = m_refTreeModel->children().size() - 1;
2580      rowInstr[m_Columns.m_col_name] = name;      rowInstr[m_Columns.m_col_name] = name;
2581      rowInstr[m_Columns.m_col_instr] = instrument;      rowInstr[m_Columns.m_col_instr] = instrument;
2582        rowInstr[m_Columns.m_col_scripts] = "";
2583      instrument_name_connection.unblock();      instrument_name_connection.unblock();
2584    
2585      add_instrument_to_menu(name);      add_instrument_to_menu(name);
# Line 1358  void MainWindow::on_action_duplicate_ins Line 2606  void MainWindow::on_action_duplicate_ins
2606      // retrieve the currently selected instrument      // retrieve the currently selected instrument
2607      // (being the original instrument to be duplicated)      // (being the original instrument to be duplicated)
2608      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeView.get_selection();      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeView.get_selection();
2609      Gtk::TreeModel::iterator itSelection = sel->get_selected();      std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
2610      if (!itSelection) return;      for (int r = 0; r < rows.size(); ++r) {
2611      Gtk::TreeModel::Row row = *itSelection;          Gtk::TreeModel::iterator it = m_refTreeModel->get_iter(rows[r]);
2612      gig::Instrument* instrOrig = row[m_Columns.m_col_instr];          if (it) {
2613      if (!instrOrig) return;              Gtk::TreeModel::Row row = *it;
2614                gig::Instrument* instrOrig = row[m_Columns.m_col_instr];
2615      // duplicate the orginal instrument              if (instrOrig) {
2616      gig::Instrument* instrNew = file->AddDuplicateInstrument(instrOrig);                  // duplicate the orginal instrument
2617      instrNew->pInfo->Name =                  gig::Instrument* instrNew = file->AddDuplicateInstrument(instrOrig);
2618          instrOrig->pInfo->Name +                  instrNew->pInfo->Name =
2619          gig_from_utf8(Glib::ustring(" (") + _("Copy") + ")");                      instrOrig->pInfo->Name +
2620                        gig_from_utf8(Glib::ustring(" (") + _("Copy") + ")");
2621    
2622      add_instrument(instrNew);                  add_instrument(instrNew);
2623                }
2624            }
2625        }
2626  }  }
2627    
2628  void MainWindow::on_action_remove_instrument() {  void MainWindow::on_action_remove_instrument() {
# Line 1387  void MainWindow::on_action_remove_instru Line 2639  void MainWindow::on_action_remove_instru
2639      }      }
2640    
2641      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeView.get_selection();      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeView.get_selection();
2642      Gtk::TreeModel::iterator it = sel->get_selected();      std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
2643      if (it) {      for (int r = rows.size() - 1; r >= 0; --r) {
2644            Gtk::TreeModel::iterator it = m_refTreeModel->get_iter(rows[r]);
2645            if (!it) continue;
2646          Gtk::TreeModel::Row row = *it;          Gtk::TreeModel::Row row = *it;
2647          gig::Instrument* instr = row[m_Columns.m_col_instr];          gig::Instrument* instr = row[m_Columns.m_col_instr];
2648          try {          try {
# Line 1403  void MainWindow::on_action_remove_instru Line 2657  void MainWindow::on_action_remove_instru
2657    
2658              // remove row from instruments tree view              // remove row from instruments tree view
2659              m_refTreeModel->erase(it);              m_refTreeModel->erase(it);
2660                // update "Nr" column of all instrument rows
2661                {
2662                    int index = 0;
2663                    for (Gtk::TreeModel::iterator it = m_refTreeModel->children().begin();
2664                         it != m_refTreeModel->children().end(); ++it, ++index)
2665                    {
2666                        Gtk::TreeModel::Row row = *it;
2667                        row[m_Columns.m_col_nr] = index;
2668                    }
2669                }
2670    
2671  #if GTKMM_MAJOR_VERSION < 3  #if GTKMM_MAJOR_VERSION < 3
2672              // select another instrument (in gtk3 this is done              // select another instrument (in gtk3 this is done
# Line 1416  void MainWindow::on_action_remove_instru Line 2680  void MainWindow::on_action_remove_instru
2680              }              }
2681  #endif  #endif
2682              instr_props_set_instrument();              instr_props_set_instrument();
2683                instr = get_instrument();
2684                if (instr) {
2685                    midiRules.set_instrument(instr);
2686                } else {
2687                    midiRules.hide();
2688                }
2689          } catch (RIFF::Exception e) {          } catch (RIFF::Exception e) {
2690              Gtk::MessageDialog msg(*this, e.Message.c_str(), false, Gtk::MESSAGE_ERROR);              Gtk::MessageDialog msg(*this, e.Message.c_str(), false, Gtk::MESSAGE_ERROR);
2691              msg.run();              msg.run();
# Line 1431  void MainWindow::on_action_sample_proper Line 2701  void MainWindow::on_action_sample_proper
2701      msg.run();      msg.run();
2702  }  }
2703    
2704    void MainWindow::on_action_add_script_group() {
2705        static int __script_indexer = 0;
2706        if (!file) return;
2707        gig::ScriptGroup* group = file->AddScriptGroup();
2708        group->Name = gig_from_utf8(_("Unnamed Group"));
2709        if (__script_indexer) group->Name += " " + ToString(__script_indexer);
2710        __script_indexer++;
2711        // update sample tree view
2712        Gtk::TreeModel::iterator iterGroup = m_refScriptsTreeModel->append();
2713        Gtk::TreeModel::Row rowGroup = *iterGroup;
2714        rowGroup[m_ScriptsModel.m_col_name] = gig_to_utf8(group->Name);
2715        rowGroup[m_ScriptsModel.m_col_script] = NULL;
2716        rowGroup[m_ScriptsModel.m_col_group] = group;
2717        file_changed();
2718    }
2719    
2720    void MainWindow::on_action_add_script() {
2721        if (!file) return;
2722        // get selected group
2723        Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewScripts.get_selection();
2724        Gtk::TreeModel::iterator it = sel->get_selected();
2725        if (!it) return;
2726        Gtk::TreeModel::Row row = *it;
2727        gig::ScriptGroup* group = row[m_ScriptsModel.m_col_group];
2728        if (!group) { // not a group, but a script is selected (probably)
2729            gig::Script* script = row[m_ScriptsModel.m_col_script];
2730            if (!script) return;
2731            it = row.parent(); // resolve parent (that is the script's group)
2732            if (!it) return;
2733            row = *it;
2734            group = row[m_ScriptsModel.m_col_group];
2735            if (!group) return;
2736        }
2737    
2738        // add a new script to the .gig file
2739        gig::Script* script = group->AddScript();    
2740        Glib::ustring name = _("Unnamed Script");
2741        script->Name = gig_from_utf8(name);
2742    
2743        // add script to the tree view
2744        Gtk::TreeModel::iterator iterScript =
2745            m_refScriptsTreeModel->append(row.children());
2746        Gtk::TreeModel::Row rowScript = *iterScript;
2747        rowScript[m_ScriptsModel.m_col_name] = name;
2748        rowScript[m_ScriptsModel.m_col_script] = script;
2749        rowScript[m_ScriptsModel.m_col_group]  = NULL;
2750    
2751        // unfold group of new script item in treeview
2752        Gtk::TreeModel::Path path(iterScript);
2753        m_TreeViewScripts.expand_to_path(path);
2754    }
2755    
2756    void MainWindow::on_action_edit_script() {
2757        if (!file) return;
2758        // get selected script
2759        Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewScripts.get_selection();
2760        Gtk::TreeModel::iterator it = sel->get_selected();
2761        if (!it) return;
2762        Gtk::TreeModel::Row row = *it;
2763        gig::Script* script = row[m_ScriptsModel.m_col_script];
2764        if (!script) return;
2765    
2766        ScriptEditor* editor = new ScriptEditor;
2767        editor->signal_script_to_be_changed.connect(
2768            signal_script_to_be_changed.make_slot()
2769        );
2770        editor->signal_script_changed.connect(
2771            signal_script_changed.make_slot()
2772        );
2773        editor->setScript(script);
2774        //editor->reparent(*this);
2775        editor->show();
2776    }
2777    
2778    void MainWindow::on_action_remove_script() {
2779        if (!file) return;
2780        Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewScripts.get_selection();
2781        Gtk::TreeModel::iterator it = sel->get_selected();
2782        if (it) {
2783            Gtk::TreeModel::Row row = *it;
2784            gig::ScriptGroup* group = row[m_ScriptsModel.m_col_group];
2785            gig::Script* script     = row[m_ScriptsModel.m_col_script];
2786            Glib::ustring name      = row[m_ScriptsModel.m_col_name];
2787            try {
2788                // remove script group or script from the gig file
2789                if (group) {
2790                    // notify everybody that we're going to remove these samples
2791    //TODO:         scripts_to_be_removed_signal.emit(members);
2792                    // delete the group in the .gig file including the
2793                    // samples that belong to the group
2794                    file->DeleteScriptGroup(group);
2795                    // notify that we're done with removal
2796    //TODO:         scripts_removed_signal.emit();
2797                    file_changed();
2798                } else if (script) {
2799                    // notify everybody that we're going to remove this sample
2800    //TODO:         std::list<gig::Script*> lscripts;
2801    //TODO:         lscripts.push_back(script);
2802    //TODO:         scripts_to_be_removed_signal.emit(lscripts);
2803                    // remove sample from the .gig file
2804                    script->GetGroup()->DeleteScript(script);
2805                    // notify that we're done with removal
2806    //TODO:         scripts_removed_signal.emit();
2807                    dimreg_changed();
2808                    file_changed();
2809                }
2810                // remove respective row(s) from samples tree view
2811                m_refScriptsTreeModel->erase(it);
2812            } catch (RIFF::Exception e) {
2813                // pretend we're done with removal (i.e. to avoid dead locks)
2814    //TODO:     scripts_removed_signal.emit();
2815                // show error message
2816                Gtk::MessageDialog msg(*this, e.Message.c_str(), false, Gtk::MESSAGE_ERROR);
2817                msg.run();
2818            }
2819        }
2820    }
2821    
2822  void MainWindow::on_action_add_group() {  void MainWindow::on_action_add_group() {
2823      static int __sample_indexer = 0;      static int __sample_indexer = 0;
2824      if (!file) return;      if (!file) return;
# Line 1447  void MainWindow::on_action_add_group() { Line 2835  void MainWindow::on_action_add_group() {
2835      file_changed();      file_changed();
2836  }  }
2837    
2838    void MainWindow::on_action_replace_sample() {
2839        add_or_replace_sample(true);
2840    }
2841    
2842  void MainWindow::on_action_add_sample() {  void MainWindow::on_action_add_sample() {
2843        add_or_replace_sample(false);
2844    }
2845    
2846    void MainWindow::add_or_replace_sample(bool replace) {
2847      if (!file) return;      if (!file) return;
2848      // get selected group  
2849        // get selected group (and probably selected sample)
2850      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();
2851      Gtk::TreeModel::iterator it = sel->get_selected();      std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
2852        if (rows.empty()) return;
2853        Gtk::TreeModel::iterator it = m_refSamplesTreeModel->get_iter(rows[0]);
2854      if (!it) return;      if (!it) return;
2855      Gtk::TreeModel::Row row = *it;      Gtk::TreeModel::Row row = *it;
2856        gig::Sample* sample = NULL;
2857      gig::Group* group = row[m_SamplesModel.m_col_group];      gig::Group* group = row[m_SamplesModel.m_col_group];
2858      if (!group) { // not a group, but a sample is selected (probably)      if (!group) { // not a group, but a sample is selected (probably)
2859          gig::Sample* sample = row[m_SamplesModel.m_col_sample];          if (replace) sample = row[m_SamplesModel.m_col_sample];
2860          if (!sample) return;          if (!row[m_SamplesModel.m_col_sample]) return;
2861          it = row.parent(); // resolve parent (that is the sample's group)          it = row.parent(); // resolve parent (that is the sample's group)
2862          if (!it) return;          if (!it) return;
2863          row = *it;          if (!replace) row = *it;
2864          group = row[m_SamplesModel.m_col_group];          group = (*it)[m_SamplesModel.m_col_group];
2865          if (!group) return;          if (!group) return;
2866      }      }
2867        if (replace && !sample) return;
2868    
2869      // show 'browse for file' dialog      // show 'browse for file' dialog
2870      Gtk::FileChooserDialog dialog(*this, _("Add Sample(s)"));      Gtk::FileChooserDialog dialog(*this, replace ? _("Replace Sample with") : _("Add Sample(s)"));
2871      dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);      dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
2872      dialog.add_button(Gtk::Stock::OPEN, Gtk::RESPONSE_OK);      dialog.add_button(Gtk::Stock::OPEN, Gtk::RESPONSE_OK);
2873      dialog.set_select_multiple(true);      dialog.set_select_multiple(!replace); // allow multi audio file selection only when adding new samples, does not make sense when replacing a specific sample
2874    
2875      // matches all file types supported by libsndfile      // matches all file types supported by libsndfile
2876  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
# Line 1539  void MainWindow::on_action_add_sample() Line 2941  void MainWindow::on_action_add_sample()
2941                          sf_close(hFile); // close sound file                          sf_close(hFile); // close sound file
2942                          throw std::string(_("format not supported")); // unsupported subformat (yet?)                          throw std::string(_("format not supported")); // unsupported subformat (yet?)
2943                  }                  }
2944                  // add a new sample to the .gig file                  // add a new sample to the .gig file (if adding is requested actually)
2945                  gig::Sample* sample = file->AddSample();                  if (!replace) sample = file->AddSample();
2946                  // file name without path                  // file name without path
2947                  Glib::ustring filename = Glib::filename_display_basename(*iter);                  Glib::ustring filename = Glib::filename_display_basename(*iter);
2948                  // remove file extension if there is one                  // remove file extension if there is one
# Line 1564  void MainWindow::on_action_add_sample() Line 2966  void MainWindow::on_action_add_sample()
2966                                 &instrument, sizeof(instrument)) != SF_FALSE)                                 &instrument, sizeof(instrument)) != SF_FALSE)
2967                  {                  {
2968                      sample->MIDIUnityNote = instrument.basenote;                      sample->MIDIUnityNote = instrument.basenote;
2969                        sample->FineTune      = instrument.detune;
2970    
2971                      if (instrument.loop_count && instrument.loops[0].mode != SF_LOOP_NONE) {                      if (instrument.loop_count && instrument.loops[0].mode != SF_LOOP_NONE) {
2972                          sample->Loops = 1;                          sample->Loops = 1;
# Line 1590  void MainWindow::on_action_add_sample() Line 2993  void MainWindow::on_action_add_sample()
2993                  // physically when File::Save() is called)                  // physically when File::Save() is called)
2994                  sample->Resize(info.frames);                  sample->Resize(info.frames);
2995                  // make sure sample is part of the selected group                  // make sure sample is part of the selected group
2996                  group->AddSample(sample);                  if (!replace) group->AddSample(sample);
2997                  // schedule that physical resize and sample import                  // schedule that physical resize and sample import
2998                  // (data copying), performed when "Save" is requested                  // (data copying), performed when "Save" is requested
2999                  SampleImportItem sched_item;                  SampleImportItem sched_item;
3000                  sched_item.gig_sample  = sample;                  sched_item.gig_sample  = sample;
3001                  sched_item.sample_path = *iter;                  sched_item.sample_path = *iter;
3002                  m_SampleImportQueue.push_back(sched_item);                  m_SampleImportQueue[sample] = sched_item;
3003                  // add sample to the tree view                  // add sample to the tree view
3004                  Gtk::TreeModel::iterator iterSample =                  if (replace) {
3005                      m_refSamplesTreeModel->append(row.children());                      row[m_SamplesModel.m_col_name] = gig_to_utf8(sample->pInfo->Name);
3006                  Gtk::TreeModel::Row rowSample = *iterSample;                  } else {
3007                  rowSample[m_SamplesModel.m_col_name] =                      Gtk::TreeModel::iterator iterSample =
3008                      gig_to_utf8(sample->pInfo->Name);                          m_refSamplesTreeModel->append(row.children());
3009                  rowSample[m_SamplesModel.m_col_sample] = sample;                      Gtk::TreeModel::Row rowSample = *iterSample;
3010                  rowSample[m_SamplesModel.m_col_group]  = NULL;                      rowSample[m_SamplesModel.m_col_name] =
3011                            gig_to_utf8(sample->pInfo->Name);
3012                        rowSample[m_SamplesModel.m_col_sample] = sample;
3013                        rowSample[m_SamplesModel.m_col_group]  = NULL;
3014                    }
3015                  // close sound file                  // close sound file
3016                  sf_close(hFile);                  sf_close(hFile);
3017                  file_changed();                  file_changed();
# Line 1615  void MainWindow::on_action_add_sample() Line 3022  void MainWindow::on_action_add_sample()
3022          }          }
3023          // show error message box when some file(s) could not be opened / added          // show error message box when some file(s) could not be opened / added
3024          if (!error_files.empty()) {          if (!error_files.empty()) {
3025              Glib::ustring txt = _("Could not add the following sample(s):\n") + error_files;              Glib::ustring txt =
3026                    (replace
3027                        ? _("Failed to replace sample with:\n")
3028                        : _("Could not add the following sample(s):\n"))
3029                    + error_files;
3030              Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);              Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
3031              msg.run();              msg.run();
3032          }          }
# Line 1682  void MainWindow::on_action_replace_all_s Line 3093  void MainWindow::on_action_replace_all_s
3093              try              try
3094              {              {
3095                  if (!hFile) throw std::string(_("could not open file"));                  if (!hFile) throw std::string(_("could not open file"));
                 int bitdepth;  
3096                  switch (info.format & 0xff) {                  switch (info.format & 0xff) {
3097                      case SF_FORMAT_PCM_S8:                      case SF_FORMAT_PCM_S8:
3098                      case SF_FORMAT_PCM_16:                      case SF_FORMAT_PCM_16:
3099                      case SF_FORMAT_PCM_U8:                      case SF_FORMAT_PCM_U8:
                         bitdepth = 16;  
                         break;  
3100                      case SF_FORMAT_PCM_24:                      case SF_FORMAT_PCM_24:
3101                      case SF_FORMAT_PCM_32:                      case SF_FORMAT_PCM_32:
3102                      case SF_FORMAT_FLOAT:                      case SF_FORMAT_FLOAT:
3103                      case SF_FORMAT_DOUBLE:                      case SF_FORMAT_DOUBLE:
                         bitdepth = 24;  
3104                          break;                          break;
3105                      default:                      default:
3106                          sf_close(hFile);                          sf_close(hFile);
# Line 1702  void MainWindow::on_action_replace_all_s Line 3109  void MainWindow::on_action_replace_all_s
3109                  SampleImportItem sched_item;                  SampleImportItem sched_item;
3110                  sched_item.gig_sample  = sample;                  sched_item.gig_sample  = sample;
3111                  sched_item.sample_path = filename;                  sched_item.sample_path = filename;
3112                  m_SampleImportQueue.push_back(sched_item);                  m_SampleImportQueue[sample] = sched_item;
3113                  sf_close(hFile);                  sf_close(hFile);
3114                  file_changed();                  file_changed();
3115              }              }
# Line 1726  void MainWindow::on_action_replace_all_s Line 3133  void MainWindow::on_action_replace_all_s
3133  void MainWindow::on_action_remove_sample() {  void MainWindow::on_action_remove_sample() {
3134      if (!file) return;      if (!file) return;
3135      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();
3136      Gtk::TreeModel::iterator it = sel->get_selected();      std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
3137      if (it) {      for (int r = rows.size() - 1; r >= 0; --r) {
3138            Gtk::TreeModel::iterator it = m_refSamplesTreeModel->get_iter(rows[r]);
3139            if (!it) continue;
3140          Gtk::TreeModel::Row row = *it;          Gtk::TreeModel::Row row = *it;
3141          gig::Group* group   = row[m_SamplesModel.m_col_group];          gig::Group* group   = row[m_SamplesModel.m_col_group];
3142          gig::Sample* sample = row[m_SamplesModel.m_col_sample];          gig::Sample* sample = row[m_SamplesModel.m_col_sample];
# Line 1752  void MainWindow::on_action_remove_sample Line 3161  void MainWindow::on_action_remove_sample
3161                  // if sample(s) were just previously added, remove                  // if sample(s) were just previously added, remove
3162                  // them from the import queue                  // them from the import queue
3163                  for (std::list<gig::Sample*>::iterator member = members.begin();                  for (std::list<gig::Sample*>::iterator member = members.begin();
3164                       member != members.end(); ++member) {                       member != members.end(); ++member)
3165                      for (std::list<SampleImportItem>::iterator iter = m_SampleImportQueue.begin();                  {
3166                           iter != m_SampleImportQueue.end(); ++iter) {                      if (m_SampleImportQueue.count(*member)) {
3167                          if ((*iter).gig_sample == *member) {                          printf("Removing previously added sample '%s' from group '%s'\n",
3168                              printf("Removing previously added sample '%s' from group '%s'\n",                                 m_SampleImportQueue[sample].sample_path.c_str(), name.c_str());
3169                                     (*iter).sample_path.c_str(), name.c_str());                          m_SampleImportQueue.erase(*member);
                             m_SampleImportQueue.erase(iter);  
                             break;  
                         }  
3170                      }                      }
3171                  }                  }
3172                  file_changed();                  file_changed();
# Line 1775  void MainWindow::on_action_remove_sample Line 3181  void MainWindow::on_action_remove_sample
3181                  samples_removed_signal.emit();                  samples_removed_signal.emit();
3182                  // if sample was just previously added, remove it from                  // if sample was just previously added, remove it from
3183                  // the import queue                  // the import queue
3184                  for (std::list<SampleImportItem>::iterator iter = m_SampleImportQueue.begin();                  if (m_SampleImportQueue.count(sample)) {
3185                       iter != m_SampleImportQueue.end(); ++iter) {                      printf("Removing previously added sample '%s'\n",
3186                      if ((*iter).gig_sample == sample) {                             m_SampleImportQueue[sample].sample_path.c_str());
3187                          printf("Removing previously added sample '%s'\n",                      m_SampleImportQueue.erase(sample);
                                (*iter).sample_path.c_str());  
                         m_SampleImportQueue.erase(iter);  
                         break;  
                     }  
3188                  }                  }
3189                  dimreg_changed();                  dimreg_changed();
3190                  file_changed();                  file_changed();
# Line 1799  void MainWindow::on_action_remove_sample Line 3201  void MainWindow::on_action_remove_sample
3201      }      }
3202  }  }
3203    
3204    void MainWindow::on_action_remove_unused_samples() {
3205        if (!file) return;
3206    
3207        // collect all samples that are not referenced by any instrument
3208        std::list<gig::Sample*> lsamples;
3209        for (int iSample = 0; file->GetSample(iSample); ++iSample) {
3210            gig::Sample* sample = file->GetSample(iSample);
3211            bool isUsed = false;
3212            for (gig::Instrument* instrument = file->GetFirstInstrument(); instrument;
3213                                  instrument = file->GetNextInstrument())
3214            {
3215                for (gig::Region* rgn = instrument->GetFirstRegion(); rgn;
3216                                  rgn = instrument->GetNextRegion())
3217                {
3218                    for (int i = 0; i < 256; ++i) {
3219                        if (!rgn->pDimensionRegions[i]) continue;
3220                        if (rgn->pDimensionRegions[i]->pSample != sample) continue;
3221                        isUsed = true;
3222                        goto endOfRefSearch;
3223                    }
3224                }
3225            }
3226            endOfRefSearch:
3227            if (!isUsed) lsamples.push_back(sample);
3228        }
3229    
3230        if (lsamples.empty()) return;
3231    
3232        // notify everybody that we're going to remove these samples
3233        samples_to_be_removed_signal.emit(lsamples);
3234    
3235        // remove collected samples
3236        try {
3237            for (std::list<gig::Sample*>::iterator itSample = lsamples.begin();
3238                 itSample != lsamples.end(); ++itSample)
3239            {
3240                gig::Sample* sample = *itSample;
3241                // remove sample from the .gig file
3242                file->DeleteSample(sample);
3243                // if sample was just previously added, remove it from the import queue
3244                if (m_SampleImportQueue.count(sample)) {
3245                    printf("Removing previously added sample '%s'\n",
3246                           m_SampleImportQueue[sample].sample_path.c_str());
3247                    m_SampleImportQueue.erase(sample);
3248                }
3249            }
3250        } catch (RIFF::Exception e) {
3251            // show error message
3252            Gtk::MessageDialog msg(*this, e.Message.c_str(), false, Gtk::MESSAGE_ERROR);
3253            msg.run();
3254        }
3255    
3256        // notify everybody that we're done with removal
3257        samples_removed_signal.emit();
3258    
3259        dimreg_changed();
3260        file_changed();
3261        __refreshEntireGUI();
3262    }
3263    
3264    // see comment on on_sample_treeview_drag_begin()
3265    void MainWindow::on_scripts_treeview_drag_begin(const Glib::RefPtr<Gdk::DragContext>& context)
3266    {
3267        first_call_to_drag_data_get = true;
3268    }
3269    
3270    void MainWindow::on_scripts_treeview_drag_data_get(const Glib::RefPtr<Gdk::DragContext>&,
3271                                                       Gtk::SelectionData& selection_data, guint, guint)
3272    {
3273        if (!first_call_to_drag_data_get) return;
3274        first_call_to_drag_data_get = false;
3275    
3276        // get selected script
3277        gig::Script* script = NULL;
3278        Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewScripts.get_selection();
3279        Gtk::TreeModel::iterator it = sel->get_selected();
3280        if (it) {
3281            Gtk::TreeModel::Row row = *it;
3282            script = row[m_ScriptsModel.m_col_script];
3283        }
3284        // pass the gig::Script as pointer
3285        selection_data.set(selection_data.get_target(), 0/*unused*/,
3286                           (const guchar*)&script,
3287                           sizeof(script)/*length of data in bytes*/);
3288    }
3289    
3290    // see comment on on_sample_treeview_drag_begin()
3291    void MainWindow::on_instruments_treeview_drag_begin(const Glib::RefPtr<Gdk::DragContext>& context)
3292    {
3293        first_call_to_drag_data_get = true;
3294    }
3295    
3296    void MainWindow::on_instruments_treeview_drag_data_get(const Glib::RefPtr<Gdk::DragContext>&,
3297                                                           Gtk::SelectionData& selection_data, guint, guint)
3298    {
3299        if (!first_call_to_drag_data_get) return;
3300        first_call_to_drag_data_get = false;
3301    
3302        // get selected source instrument
3303        gig::Instrument* src = NULL;
3304        {
3305            Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeView.get_selection();
3306            std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
3307            if (!rows.empty()) {
3308                Gtk::TreeModel::iterator it = m_refTreeModel->get_iter(rows[0]);
3309                if (it) {
3310                    Gtk::TreeModel::Row row = *it;
3311                    src = row[m_Columns.m_col_instr];
3312                }
3313            }
3314        }
3315        if (!src) return;
3316    
3317        // pass the source gig::Instrument as pointer
3318        selection_data.set(selection_data.get_target(), 0/*unused*/, (const guchar*)&src,
3319                           sizeof(src)/*length of data in bytes*/);
3320    }
3321    
3322    void MainWindow::on_instruments_treeview_drop_drag_data_received(
3323        const Glib::RefPtr<Gdk::DragContext>& context, int x, int y,
3324        const Gtk::SelectionData& selection_data, guint, guint time)
3325    {
3326        gig::Instrument* src = *((gig::Instrument**) selection_data.get_data());
3327        if (!src || selection_data.get_length() != sizeof(gig::Instrument*))
3328            return;
3329    
3330        gig::Instrument* dst = NULL;
3331        {
3332            Gtk::TreeModel::Path path;
3333            const bool found = m_TreeView.get_path_at_pos(x, y, path);
3334            if (!found) return;
3335    
3336            Gtk::TreeModel::iterator iter = m_refTreeModel->get_iter(path);
3337            if (!iter) return;
3338            Gtk::TreeModel::Row row = *iter;
3339            dst = row[m_Columns.m_col_instr];
3340        }
3341        if (!dst) return;
3342    
3343        //printf("dragdrop received src=%s dst=%s\n", src->pInfo->Name.c_str(), dst->pInfo->Name.c_str());
3344        src->MoveTo(dst);
3345        __refreshEntireGUI();
3346        select_instrument(src);
3347    }
3348    
3349  // For some reason drag_data_get gets called two times for each  // For some reason drag_data_get gets called two times for each
3350  // drag'n'drop (at least when target is an Entry). This work-around  // drag'n'drop (at least when target is an Entry). This work-around
3351  // makes sure the code in drag_data_get and drop_drag_data_received is  // makes sure the code in drag_data_get and drop_drag_data_received is
# Line 1817  void MainWindow::on_sample_treeview_drag Line 3364  void MainWindow::on_sample_treeview_drag
3364      // get selected sample      // get selected sample
3365      gig::Sample* sample = NULL;      gig::Sample* sample = NULL;
3366      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();      Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();
3367      Gtk::TreeModel::iterator it = sel->get_selected();      std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
3368      if (it) {      if (!rows.empty()) {
3369          Gtk::TreeModel::Row row = *it;          Gtk::TreeModel::iterator it = m_refSamplesTreeModel->get_iter(rows[0]);
3370          sample = row[m_SamplesModel.m_col_sample];          if (it) {
3371                Gtk::TreeModel::Row row = *it;
3372                sample = row[m_SamplesModel.m_col_sample];
3373            }
3374      }      }
3375      // pass the gig::Sample as pointer      // pass the gig::Sample as pointer
3376      selection_data.set(selection_data.get_target(), 0/*unused*/, (const guchar*)&sample,      selection_data.set(selection_data.get_target(), 0/*unused*/, (const guchar*)&sample,
# Line 1857  void MainWindow::on_sample_label_drop_dr Line 3407  void MainWindow::on_sample_label_drop_dr
3407          bool channels_changed = false;          bool channels_changed = false;
3408          if (sample->Channels == 1 && stereo_dimension) {          if (sample->Channels == 1 && stereo_dimension) {
3409              // remove the samplechannel dimension              // remove the samplechannel dimension
3410    /* commented out, because it makes it impossible building up an instrument from scratch using two separate L/R samples
3411              region->DeleteDimension(stereo_dimension);              region->DeleteDimension(stereo_dimension);
3412              channels_changed = true;              channels_changed = true;
3413              region_changed();              region_changed();
3414    */
3415          }          }
3416          dimreg_edit.set_sample(sample);          dimreg_edit.set_sample(
3417                sample,
3418                is_copy_samples_unity_note_enabled(),
3419                is_copy_samples_fine_tune_enabled(),
3420                is_copy_samples_loop_enabled()
3421            );
3422    
3423          if (sample->Channels == 2 && !stereo_dimension) {          if (sample->Channels == 2 && !stereo_dimension) {
3424              // add samplechannel dimension              // add samplechannel dimension
# Line 1920  void MainWindow::sample_name_changed(con Line 3477  void MainWindow::sample_name_changed(con
3477      }      }
3478  }  }
3479    
3480    void MainWindow::script_name_changed(const Gtk::TreeModel::Path& path,
3481                                         const Gtk::TreeModel::iterator& iter) {
3482        if (!iter) return;
3483        Gtk::TreeModel::Row row = *iter;
3484        Glib::ustring name      = row[m_ScriptsModel.m_col_name];
3485        gig::ScriptGroup* group = row[m_ScriptsModel.m_col_group];
3486        gig::Script* script     = row[m_ScriptsModel.m_col_script];
3487        gig::String gigname(gig_from_utf8(name));
3488        if (group) {
3489            if (group->Name != gigname) {
3490                group->Name = gigname;
3491                printf("script group name changed\n");
3492                file_changed();
3493            }
3494        } else if (script) {
3495            if (script->Name != gigname) {
3496                script->Name = gigname;
3497                printf("script name changed\n");
3498                file_changed();
3499            }
3500        }
3501    }
3502    
3503    void MainWindow::script_double_clicked(const Gtk::TreeModel::Path& path,
3504                                           Gtk::TreeViewColumn* column)
3505    {
3506        Gtk::TreeModel::iterator iter = m_refScriptsTreeModel->get_iter(path);
3507        if (!iter) return;
3508        Gtk::TreeModel::Row row = *iter;
3509        gig::Script* script = row[m_ScriptsModel.m_col_script];
3510        if (!script) return;
3511    
3512        ScriptEditor* editor = new ScriptEditor;
3513        editor->signal_script_to_be_changed.connect(
3514            signal_script_to_be_changed.make_slot()
3515        );
3516        editor->signal_script_changed.connect(
3517            signal_script_changed.make_slot()
3518        );
3519        editor->setScript(script);
3520        //editor->reparent(*this);
3521        editor->show();
3522    }
3523    
3524  void MainWindow::instrument_name_changed(const Gtk::TreeModel::Path& path,  void MainWindow::instrument_name_changed(const Gtk::TreeModel::Path& path,
3525                                           const Gtk::TreeModel::iterator& iter) {                                           const Gtk::TreeModel::iterator& iter) {
3526      if (!iter) return;      if (!iter) return;
# Line 1954  void MainWindow::instrument_name_changed Line 3555  void MainWindow::instrument_name_changed
3555      }      }
3556  }  }
3557    
3558    void MainWindow::on_action_combine_instruments() {
3559        CombineInstrumentsDialog* d = new CombineInstrumentsDialog(*this, file);
3560        d->show_all();
3561        d->resize(500, 400);
3562        d->run();
3563        if (d->fileWasChanged()) {
3564            // update GUI with new instrument just created
3565            add_instrument(d->newCombinedInstrument());
3566        }
3567        delete d;
3568    }
3569    
3570    void MainWindow::on_action_view_references() {
3571        Glib::RefPtr<Gtk::TreeSelection> sel = m_TreeViewSamples.get_selection();
3572        std::vector<Gtk::TreeModel::Path> rows = sel->get_selected_rows();
3573        if (rows.empty()) return;
3574        Gtk::TreeModel::iterator it = m_refSamplesTreeModel->get_iter(rows[0]);
3575        if (!it) return;
3576        Gtk::TreeModel::Row row = *it;
3577        gig::Sample* sample = row[m_SamplesModel.m_col_sample];
3578        if (!sample) return;
3579    
3580        ReferencesView* d = new ReferencesView(*this);
3581        d->setSample(sample);
3582        d->dimension_region_selected.connect(
3583            sigc::mem_fun(*this, &MainWindow::select_dimension_region)
3584        );
3585        d->show_all();
3586        d->resize(500, 400);
3587        d->run();
3588        delete d;
3589    }
3590    
3591    void MainWindow::mergeFiles(const std::vector<std::string>& filenames) {
3592        struct _Source {
3593            std::vector<RIFF::File*> riffs;
3594            std::vector<gig::File*> gigs;
3595            
3596            ~_Source() {
3597                for (int k = 0; k < gigs.size(); ++k) delete gigs[k];
3598                for (int k = 0; k < riffs.size(); ++k) delete riffs[k];
3599                riffs.clear();
3600                gigs.clear();
3601            }
3602        } sources;
3603    
3604        if (filenames.empty())
3605            throw RIFF::Exception(_("No files selected, so nothing done."));
3606    
3607        // first open all input files (to avoid output file corruption)
3608        int i;
3609        try {
3610            for (i = 0; i < filenames.size(); ++i) {
3611                const std::string& filename = filenames[i];
3612                printf("opening file=%s\n", filename.c_str());
3613    
3614                RIFF::File* riff = new RIFF::File(filename);
3615                sources.riffs.push_back(riff);
3616    
3617                gig::File* gig = new gig::File(riff);
3618                sources.gigs.push_back(gig);
3619            }
3620        } catch (RIFF::Exception e) {
3621            throw RIFF::Exception(
3622                _("Error occurred while opening '") +
3623                filenames[i] +
3624                "': " +
3625                e.Message
3626            );
3627        } catch (...) {
3628            throw RIFF::Exception(
3629                _("Unknown exception occurred while opening '") +
3630                filenames[i] + "'"
3631            );
3632        }
3633    
3634        // now merge the opened .gig files to the main .gig file currently being
3635        // open in gigedit
3636        try {
3637            for (i = 0; i < filenames.size(); ++i) {
3638                const std::string& filename = filenames[i];
3639                printf("merging file=%s\n", filename.c_str());
3640                assert(i < sources.gigs.size());
3641    
3642                this->file->AddContentOf(sources.gigs[i]);
3643            }
3644        } catch (RIFF::Exception e) {
3645            throw RIFF::Exception(
3646                _("Error occurred while merging '") +
3647                filenames[i] +
3648                "': " +
3649                e.Message
3650            );
3651        } catch (...) {
3652            throw RIFF::Exception(
3653                _("Unknown exception occurred while merging '") +
3654                filenames[i] + "'"
3655            );
3656        }
3657    
3658        // Finally save gig file persistently to disk ...
3659        //NOTE: requires that this gig file already has a filename !
3660        {
3661            std::cout << "Saving file\n" << std::flush;
3662            file_structure_to_be_changed_signal.emit(this->file);
3663    
3664            progress_dialog = new ProgressDialog( //FIXME: memory leak!
3665                _("Saving") +  Glib::ustring(" '") +
3666                Glib::filename_display_basename(this->filename) + "' ...",
3667                *this
3668            );
3669            progress_dialog->show_all();
3670            saver = new Saver(this->file); //FIXME: memory leak!
3671            saver->signal_progress().connect(
3672                sigc::mem_fun(*this, &MainWindow::on_saver_progress));
3673            saver->signal_finished().connect(
3674                sigc::mem_fun(*this, &MainWindow::on_saver_finished));
3675            saver->signal_error().connect(
3676                sigc::mem_fun(*this, &MainWindow::on_saver_error));
3677            saver->launch();
3678        }
3679    }
3680    
3681    void MainWindow::on_action_merge_files() {
3682        if (this->file->GetFileName().empty()) {
3683            Glib::ustring txt = _(
3684                "You seem to have a new .gig file open that has not been saved "
3685                "yet. You must save it somewhere before starting to merge it with "
3686                "other .gig files though, because during the merge operation the "
3687                "other files' sample data must be written on file level to the "
3688                "target .gig file."
3689            );
3690            Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
3691            msg.run();
3692            return;
3693        }
3694    
3695        Gtk::FileChooserDialog dialog(*this, _("Merge .gig files"));
3696        dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
3697        dialog.add_button(_("Merge"), Gtk::RESPONSE_OK);
3698        dialog.set_default_response(Gtk::RESPONSE_CANCEL);
3699    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
3700        Gtk::FileFilter filter;
3701        filter.add_pattern("*.gig");
3702    #else
3703        Glib::RefPtr<Gtk::FileFilter> filter = Gtk::FileFilter::create();
3704        filter->add_pattern("*.gig");
3705    #endif
3706        dialog.set_filter(filter);
3707        if (current_gig_dir != "") {
3708            dialog.set_current_folder(current_gig_dir);
3709        }
3710        dialog.set_select_multiple(true);
3711    
3712        // show warning in the file picker dialog
3713        Gtk::HBox descriptionArea;
3714        descriptionArea.set_spacing(15);
3715        Gtk::Image warningIcon;
3716        warningIcon.set_from_icon_name("dialog-warning",
3717                                       Gtk::IconSize(Gtk::ICON_SIZE_DIALOG));
3718        descriptionArea.pack_start(warningIcon, Gtk::PACK_SHRINK);
3719    #if GTKMM_MAJOR_VERSION < 3
3720        view::WrapLabel description;
3721    #else
3722        Gtk::Label description;
3723        description.set_line_wrap();
3724    #endif
3725        description.set_markup(_(
3726            "\nSelect at least one .gig file that shall be merged to the .gig file "
3727            "currently being open in gigedit.\n\n"
3728            "<b>Please Note:</b> Merging with other files will modify your "
3729            "currently open .gig file on file level! And be aware that the current "
3730            "merge algorithm does not detect duplicate samples yet. So if you are "
3731            "merging files which are using equivalent sample data, those "
3732            "equivalent samples will currently be treated as separate samples and "
3733            "will accordingly be stored separately in the target .gig file!"
3734        ));
3735        descriptionArea.pack_start(description);
3736        dialog.get_vbox()->pack_start(descriptionArea, Gtk::PACK_SHRINK);
3737        descriptionArea.show_all();
3738    
3739        if (dialog.run() == Gtk::RESPONSE_OK) {
3740            printf("on_action_merge_files self=%p\n",
3741                   static_cast<void*>(Glib::Threads::Thread::self()));
3742            std::vector<std::string> filenames = dialog.get_filenames();
3743    
3744            // merge the selected files to the currently open .gig file
3745            try {
3746                mergeFiles(filenames);
3747            } catch (RIFF::Exception e) {
3748                Gtk::MessageDialog msg(*this, e.Message, false, Gtk::MESSAGE_ERROR);
3749                msg.run();
3750            }
3751    
3752            // update GUI
3753            __refreshEntireGUI();
3754        }
3755    }
3756    
3757  void MainWindow::set_file_is_shared(bool b) {  void MainWindow::set_file_is_shared(bool b) {
3758      this->file_is_shared = b;      this->file_is_shared = b;
3759    
# Line 1968  void MainWindow::set_file_is_shared(bool Line 3768  void MainWindow::set_file_is_shared(bool
3768              Gdk::Pixbuf::create_from_xpm_data(status_detached_xpm)              Gdk::Pixbuf::create_from_xpm_data(status_detached_xpm)
3769          );          );
3770      }      }
3771    
3772        {
3773            Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
3774                uiManager->get_widget("/MenuBar/MenuSettings/SyncSamplerInstrumentSelection"));
3775            if (item) item->set_sensitive(b);
3776        }
3777    }
3778    
3779    void MainWindow::on_sample_ref_count_incremented(gig::Sample* sample, int offset) {
3780        if (!sample) return;
3781        sample_ref_count[sample] += offset;
3782        const int refcount = sample_ref_count[sample];
3783    
3784        Glib::RefPtr<Gtk::TreeModel> model = m_TreeViewSamples.get_model();
3785        for (int g = 0; g < model->children().size(); ++g) {
3786            Gtk::TreeModel::Row rowGroup = model->children()[g];
3787            for (int s = 0; s < rowGroup.children().size(); ++s) {
3788                Gtk::TreeModel::Row rowSample = rowGroup.children()[s];
3789                if (rowSample[m_SamplesModel.m_col_sample] != sample) continue;
3790                rowSample[m_SamplesModel.m_col_refcount] = ToString(refcount) + " " + _("Refs.");
3791                rowSample[m_SamplesModel.m_color] = refcount ? "black" : "red";
3792            }
3793        }
3794    }
3795    
3796    void MainWindow::on_sample_ref_changed(gig::Sample* oldSample, gig::Sample* newSample) {
3797        on_sample_ref_count_incremented(oldSample, -1);
3798        on_sample_ref_count_incremented(newSample, +1);
3799    }
3800    
3801    void MainWindow::on_samples_to_be_removed(std::list<gig::Sample*> samples) {
3802        // just in case a new sample is added later with exactly the same memory
3803        // address, which would lead to incorrect refcount if not deleted here
3804        for (std::list<gig::Sample*>::const_iterator it = samples.begin();
3805             it != samples.end(); ++it)
3806        {
3807            sample_ref_count.erase(*it);
3808        }
3809    }
3810    
3811    void MainWindow::show_samples_tab() {
3812        m_TreeViewNotebook.set_current_page(0);
3813    }
3814    
3815    void MainWindow::show_intruments_tab() {
3816        m_TreeViewNotebook.set_current_page(1);
3817    }
3818    
3819    void MainWindow::show_scripts_tab() {
3820        m_TreeViewNotebook.set_current_page(2);
3821    }
3822    
3823    void MainWindow::select_prev_region() {
3824        m_RegionChooser.select_prev_region();
3825    }
3826    
3827    void MainWindow::select_next_region() {
3828        m_RegionChooser.select_next_region();
3829    }
3830    
3831    void MainWindow::select_next_dim_rgn_zone() {
3832        if (m_DimRegionChooser.has_focus()) return; // avoid conflict with key stroke handler of DimenionRegionChooser
3833        m_DimRegionChooser.select_next_dimzone();
3834    }
3835    
3836    void MainWindow::select_prev_dim_rgn_zone() {
3837        if (m_DimRegionChooser.has_focus()) return; // avoid conflict with key stroke handler of DimenionRegionChooser
3838        m_DimRegionChooser.select_prev_dimzone();
3839    }
3840    
3841    void MainWindow::select_add_next_dim_rgn_zone() {
3842        m_DimRegionChooser.select_next_dimzone(true);
3843    }
3844    
3845    void MainWindow::select_add_prev_dim_rgn_zone() {
3846        m_DimRegionChooser.select_prev_dimzone(true);
3847    }
3848    
3849    void MainWindow::select_prev_dimension() {
3850        if (m_DimRegionChooser.has_focus()) return; // avoid conflict with key stroke handler of DimenionRegionChooser
3851        m_DimRegionChooser.select_prev_dimension();
3852    }
3853    
3854    void MainWindow::select_next_dimension() {
3855        if (m_DimRegionChooser.has_focus()) return; // avoid conflict with key stroke handler of DimenionRegionChooser
3856        m_DimRegionChooser.select_next_dimension();
3857    }
3858    
3859    #define CLIPBOARD_DIMENSIONREGION_TARGET \
3860        ("libgig.DimensionRegion." + m_serializationArchive.rawDataFormat())
3861    
3862    void MainWindow::copy_selected_dimrgn() {
3863        gig::DimensionRegion* pDimRgn = m_DimRegionChooser.get_main_dimregion();
3864        if (!pDimRgn) {
3865            updateClipboardPasteAvailable();
3866            updateClipboardCopyAvailable();
3867            return;
3868        }
3869    
3870        std::vector<Gtk::TargetEntry> targets;
3871        targets.push_back( Gtk::TargetEntry(CLIPBOARD_DIMENSIONREGION_TARGET) );
3872    
3873        Glib::RefPtr<Gtk::Clipboard> clipboard = Gtk::Clipboard::get();
3874        clipboard->set(
3875            targets,
3876            sigc::mem_fun(*this, &MainWindow::on_clipboard_get),
3877            sigc::mem_fun(*this, &MainWindow::on_clipboard_clear)
3878        );
3879    
3880        m_serializationArchive.serialize(pDimRgn);
3881    
3882        updateClipboardPasteAvailable();
3883    }
3884    
3885    void MainWindow::paste_copied_dimrgn() {
3886        Glib::RefPtr<Gtk::Clipboard> clipboard = Gtk::Clipboard::get();
3887        clipboard->request_contents(
3888            CLIPBOARD_DIMENSIONREGION_TARGET,
3889            sigc::mem_fun(*this, &MainWindow::on_clipboard_received)
3890        );
3891        updateClipboardPasteAvailable();
3892    }
3893    
3894    void MainWindow::adjust_clipboard_content() {
3895        MacroEditor* editor = new MacroEditor();
3896        editor->setMacro(&m_serializationArchive, true);
3897        editor->show();
3898    }
3899    
3900    void MainWindow::updateClipboardPasteAvailable() {
3901        Glib::RefPtr<Gtk::Clipboard> clipboard = Gtk::Clipboard::get();
3902        clipboard->request_targets(
3903            sigc::mem_fun(*this, &MainWindow::on_clipboard_received_targets)
3904        );
3905    }
3906    
3907    void MainWindow::updateClipboardCopyAvailable() {
3908        bool bDimensionRegionCopyIsPossible = m_DimRegionChooser.get_main_dimregion();
3909        static_cast<Gtk::MenuItem*>(
3910            uiManager->get_widget("/MenuBar/MenuEdit/CopyDimRgn")
3911        )->set_sensitive(bDimensionRegionCopyIsPossible);
3912    }
3913    
3914    void MainWindow::on_clipboard_owner_change(GdkEventOwnerChange* event) {
3915        updateClipboardPasteAvailable();
3916    }
3917    
3918    void MainWindow::on_clipboard_get(Gtk::SelectionData& selection_data, guint /*info*/) {
3919        const std::string target = selection_data.get_target();
3920        if (target == CLIPBOARD_DIMENSIONREGION_TARGET) {
3921            selection_data.set(
3922                CLIPBOARD_DIMENSIONREGION_TARGET, 8 /* "format": probably unused*/,
3923                &m_serializationArchive.rawData()[0],
3924                m_serializationArchive.rawData().size()
3925            );
3926        } else {
3927            std::cerr << "Clipboard: content for unknown target '" << target << "' requested\n";
3928        }
3929    }
3930    
3931    void MainWindow::on_clipboard_clear() {
3932        m_serializationArchive.clear();
3933        updateClipboardPasteAvailable();
3934        updateClipboardCopyAvailable();
3935    }
3936    
3937    //NOTE: Might throw exception !!!
3938    void MainWindow::applyMacro(Serialization::Archive& macro) {
3939        gig::DimensionRegion* pDimRgn = m_DimRegionChooser.get_main_dimregion();
3940        if (!pDimRgn) return;
3941    
3942        for (std::set<gig::DimensionRegion*>::iterator itDimReg = dimreg_edit.dimregs.begin();
3943             itDimReg != dimreg_edit.dimregs.end(); ++itDimReg)
3944        {
3945            gig::DimensionRegion* pDimRgn = *itDimReg;
3946            DimRegionChangeGuard(this, pDimRgn);
3947            macro.deserialize(pDimRgn);
3948        }
3949        //region_changed()
3950        file_changed();
3951        dimreg_changed();
3952    }
3953    
3954    void MainWindow::on_clipboard_received(const Gtk::SelectionData& selection_data) {
3955        const std::string target = selection_data.get_target();
3956        if (target == CLIPBOARD_DIMENSIONREGION_TARGET) {
3957            Glib::ustring errorText;
3958            try {
3959                m_serializationArchive.decode(
3960                    selection_data.get_data(), selection_data.get_length()
3961                );
3962                applyMacro(m_serializationArchive);
3963            } catch (Serialization::Exception e) {
3964                errorText = e.Message;
3965            } catch (...) {
3966                errorText = _("Unknown exception while pasting DimensionRegion");
3967            }
3968            if (!errorText.empty()) {
3969                Glib::ustring txt = _("Pasting DimensionRegion failed:\n") + errorText;
3970                Gtk::MessageDialog msg(*this, txt, false, Gtk::MESSAGE_ERROR);
3971                msg.run();
3972            }
3973        }
3974    }
3975    
3976    void MainWindow::on_clipboard_received_targets(const std::vector<Glib::ustring>& targets) {
3977        const bool bDimensionRegionPasteIsPossible =
3978            std::find(targets.begin(), targets.end(),
3979                      CLIPBOARD_DIMENSIONREGION_TARGET) != targets.end();
3980    
3981        static_cast<Gtk::MenuItem*>(
3982            uiManager->get_widget("/MenuBar/MenuEdit/PasteDimRgn")
3983        )->set_sensitive(bDimensionRegionPasteIsPossible);
3984    
3985        static_cast<Gtk::MenuItem*>(
3986            uiManager->get_widget("/MenuBar/MenuEdit/AdjustClipboard")
3987        )->set_sensitive(bDimensionRegionPasteIsPossible);
3988  }  }
3989    
3990  sigc::signal<void, gig::File*>& MainWindow::signal_file_structure_to_be_changed() {  sigc::signal<void, gig::File*>& MainWindow::signal_file_structure_to_be_changed() {
# Line 2025  sigc::signal<void, int/*key*/, int/*velo Line 4042  sigc::signal<void, int/*key*/, int/*velo
4042  sigc::signal<void, int/*key*/, int/*velocity*/>& MainWindow::signal_keyboard_key_released() {  sigc::signal<void, int/*key*/, int/*velocity*/>& MainWindow::signal_keyboard_key_released() {
4043      return m_RegionChooser.signal_keyboard_key_released();      return m_RegionChooser.signal_keyboard_key_released();
4044  }  }
4045    
4046    sigc::signal<void, gig::Instrument*>& MainWindow::signal_switch_sampler_instrument() {
4047        return switch_sampler_instrument_signal;
4048    }

Legend:
Removed from v.2446  
changed lines
  Added in v.3197

  ViewVC Help
Powered by ViewVC