/[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 2604 by schoenebeck, Sat Jun 7 22:34:31 2014 UTC revision 3197 by schoenebeck, Sat May 20 17:15:36 2017 UTC
# Line 1  Line 1 
1  /*  /*
2   * Copyright (C) 2006-2014 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 46  Line 55 
55  #include "Settings.h"  #include "Settings.h"
56  #include "CombineInstrumentsDialog.h"  #include "CombineInstrumentsDialog.h"
57  #include "scripteditor.h"  #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 93  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."));      dimreg_label.set_tooltip_text(_("To automatically apply your changes above globally to the entire instrument, check all 3 check boxes on the right."));
# Line 134  MainWindow::MainWindow() : Line 184  MainWindow::MainWindow() :
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",      actionGroup->add(Gtk::Action::create("MidiRules",
187                                           _("_Midi Rules")),                                           _("_Midi Rules...")),
188                       sigc::mem_fun(                       sigc::mem_fun(
189                           *this, &MainWindow::show_midi_rules));                           *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")));      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    
275      Glib::RefPtr<Gtk::ToggleAction> toggle_action =      Glib::RefPtr<Gtk::ToggleAction> toggle_action =
276          Gtk::ToggleAction::create("CopySampleUnity", _("Copy Sample's _Unity Note"));          Gtk::ToggleAction::create("CopySampleUnity", _("Copy Sample's _Unity Note"));
277      toggle_action->set_active(true);      toggle_action->set_active(true);
# Line 161  MainWindow::MainWindow() : Line 288  MainWindow::MainWindow() :
288      actionGroup->add(toggle_action);      actionGroup->add(toggle_action);
289    
290    
291      actionGroup->add(Gtk::Action::create("MenuView", _("_View")));      actionGroup->add(Gtk::Action::create("MenuMacro", _("_Macro")));
292    
293    
294        actionGroup->add(Gtk::Action::create("MenuView", _("Vie_w")));
295      toggle_action =      toggle_action =
296          Gtk::ToggleAction::create("Statusbar", _("_Statusbar"));          Gtk::ToggleAction::create("Statusbar", _("_Statusbar"));
297      toggle_action->set_active(true);      toggle_action->set_active(true);
# Line 169  MainWindow::MainWindow() : Line 299  MainWindow::MainWindow() :
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 199  MainWindow::MainWindow() : Line 348  MainWindow::MainWindow() :
348          sigc::mem_fun(*this, &MainWindow::on_action_warn_user_on_extensions)          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")));      actionGroup->add(Gtk::Action::create("MenuTools", _("_Tools")));
369    
# Line 231  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)
# Line 273  MainWindow::MainWindow() : Line 451  MainWindow::MainWindow() :
451          "      <menuitem action='Quit'/>"          "      <menuitem action='Quit'/>"
452          "    </menu>"          "    </menu>"
453          "    <menu action='MenuEdit'>"          "    <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'/>"          "      <menuitem action='CopySampleUnity'/>"
469          "      <menuitem action='CopySampleTune'/>"          "      <menuitem action='CopySampleTune'/>"
470          "      <menuitem action='CopySampleLoop'/>"          "      <menuitem action='CopySampleLoop'/>"
471          "    </menu>"          "    </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>"          "    </menu>"
510          "    <menu action='MenuTools'>"          "    <menu action='MenuTools'>"
511          "      <menuitem action='CombineInstruments'/>"          "      <menuitem action='CombineInstruments'/>"
# Line 288  MainWindow::MainWindow() : Line 513  MainWindow::MainWindow() :
513          "    </menu>"          "    </menu>"
514          "    <menu action='MenuSettings'>"          "    <menu action='MenuSettings'>"
515          "      <menuitem action='WarnUserOnExtensions'/>"          "      <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 296  MainWindow::MainWindow() : Line 524  MainWindow::MainWindow() :
524          "  <popup name='PopupMenu'>"          "  <popup name='PopupMenu'>"
525          "    <menuitem action='InstrProperties'/>"          "    <menuitem action='InstrProperties'/>"
526          "    <menuitem action='MidiRules'/>"          "    <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 305  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>"          "  </popup>"
544          "  <popup name='ScriptPopupMenu'>"          "  <popup name='ScriptPopupMenu'>"
545          "    <menuitem action='AddScriptGroup'/>"          "    <menuitem action='AddScriptGroup'/>"
# Line 347  MainWindow::MainWindow() : Line 579  MainWindow::MainWindow() :
579      }      }
580      {      {
581          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(          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"));              uiManager->get_widget("/MenuBar/MenuTools/CombineInstruments"));
612          item->set_tooltip_text(_("Create combi sounds out of individual sounds of this .gig file."));          item->set_tooltip_text(_("Create combi sounds out of individual sounds of this .gig file."));
613      }      }
# Line 358  MainWindow::MainWindow() : Line 619  MainWindow::MainWindow() :
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 384  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.set_tooltip_text(_("Right click here for actions on instruments & MIDI Rules."));      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."));      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 411  MainWindow::MainWindow() : Line 711  MainWindow::MainWindow() :
711      m_refScriptsTreeModel = ScriptsTreeStore::create(m_ScriptsModel);      m_refScriptsTreeModel = ScriptsTreeStore::create(m_ScriptsModel);
712      m_TreeViewScripts.set_model(m_refScriptsTreeModel);      m_TreeViewScripts.set_model(m_refScriptsTreeModel);
713      m_TreeViewScripts.set_tooltip_text(_(      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 "          "Note: instrument scripts are a LinuxSampler extension of the gig "
717          "format. This feature will not work with the GigaStudio software!"          "format. This feature will not work with the GigaStudio software!"
718      ));      ));
# Line 420  MainWindow::MainWindow() : Line 722  MainWindow::MainWindow() :
722      m_TreeViewScripts.signal_button_press_event().connect_notify(      m_TreeViewScripts.signal_button_press_event().connect_notify(
723          sigc::mem_fun(*this, &MainWindow::on_script_treeview_button_release)          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(      m_refScriptsTreeModel->signal_row_changed().connect(
730          sigc::mem_fun(*this, &MainWindow::script_name_changed)          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 442  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(
# Line 457  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 501  MainWindow::MainWindow() : Line 830  MainWindow::MainWindow() :
830    
831      // select 'Instruments' tab by default      // select 'Instruments' tab by default
832      // (gtk allows this only if the tab childs are visible, thats why it's here)      // (gtk allows this only if the tab childs are visible, thats why it's here)
833      m_TreeViewNotebook.set_current_page(1);      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 527  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 571  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 582  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 615  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 642  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 665  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 680  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 694  void MainWindow::__refreshEntireGUI() { Line 1310  void MainWindow::__refreshEntireGUI() {
1310      // clear the samples and instruments tree views      // clear the samples and instruments tree views
1311      m_refTreeModel->clear();      m_refTreeModel->clear();
1312      m_refSamplesTreeModel->clear();      m_refSamplesTreeModel->clear();
1313        m_refScriptsTreeModel->clear();
1314      // remove all entries from "Instrument" menu      // remove all entries from "Instrument" menu
1315      while (!instrument_menu->get_children().empty()) {      while (!instrument_menu->get_children().empty()) {
1316          remove_instrument_from_menu(0);          remove_instrument_from_menu(0);
# Line 736  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 779  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 788  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 810  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 862  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 922  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 943  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 974  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 1003  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 1012  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 1032  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 1043  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 1075  void MainWindow::on_action_warn_user_on_ Line 1781  void MainWindow::on_action_warn_user_on_
1781          !Settings::singleton()->warnUserOnExtensions;          !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 1084  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-2014 Andreas Persson");      dialog.set_copyright("Copyright (C) 2006-2017 Andreas Persson");
1804      const std::string sComment =      const std::string sComment =
1805          _("Built " __DATE__ "\nUsing ") +          _("Built " __DATE__ "\nUsing ") +
1806          ::gig::libraryName() + " " + ::gig::libraryVersion() + "\n\n" +          ::gig::libraryName() + " " + ::gig::libraryVersion() + "\n\n" +
1807          _(          _(
1808              "Gigedit is released under the GNU General Public License.\n"              "Gigedit is released under the GNU General Public License.\n"
1809              "\n"              "\n"
1810              "Please notice that this is still a very young instrument editor. "              "This program is distributed WITHOUT ANY WARRANTY; So better "
1811              "So better backup your Gigasampler files before editing them with "              "backup your Gigasampler/GigaStudio files before editing them with "
1812              "this application.\n"              "this application.\n"
1813              "\n"              "\n"
1814              "Please report bugs to: http://bugs.linuxsampler.org"              "Please report bugs to: http://bugs.linuxsampler.org"
# Line 1341  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;
# Line 1355  void MainWindow::load_gig(gig::File* gig Line 2095  void MainWindow::load_gig(gig::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 1385  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      }      }
# Line 1408  void MainWindow::load_gig(gig::File* gig Line 2157  void MainWindow::load_gig(gig::File* gig
2157              rowScript[m_ScriptsModel.m_col_group]  = NULL;              rowScript[m_ScriptsModel.m_col_group]  = NULL;
2158          }          }
2159      }      }
2160      // unfold all script groups by default      // unfold all sample groups & script groups by default
2161        m_TreeViewSamples.expand_all();
2162      m_TreeViewScripts.expand_all();      m_TreeViewScripts.expand_all();
2163    
2164      file = gig;      file = gig;
# Line 1427  bool MainWindow::instr_props_set_instrum Line 2177  bool MainWindow::instr_props_set_instrum
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 1478  void MainWindow::show_midi_rules() Line 2232  void MainWindow::show_midi_rules()
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 1489  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 {  bool MainWindow::is_copy_samples_unity_note_enabled() const {
2304      Gtk::CheckMenuItem* item =      Gtk::CheckMenuItem* item =
2305          dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuEdit/CopySampleUnity"));          dynamic_cast<Gtk::CheckMenuItem*>(uiManager->get_widget("/MenuBar/MenuEdit/CopySampleUnity"));
# Line 1525  void MainWindow::on_button_release(GdkEv Line 2336  void MainWindow::on_button_release(GdkEv
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          // 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*>(          static_cast<Gtk::MenuItem*>(
2345              uiManager->get_widget("/PopupMenu/MidiRules"))->set_sensitive(              uiManager->get_widget("/PopupMenu/MidiRules"))->set_sensitive(
2346                  !(file->pVersion && file->pVersion->major == 2));                  bEnabled
2347                );
2348          popup_menu->popup(button->button, button->time);          popup_menu->popup(button->button, button->time);
2349      }      }
2350  }  }
# Line 1547  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    
# Line 1598  void MainWindow::on_script_treeview_butt Line 2525  void MainWindow::on_script_treeview_butt
2525              set_sensitive(group_selected || script_selected);              set_sensitive(group_selected || script_selected);
2526          // show sample popup          // show sample popup
2527          script_popup->popup(button->button, button->time);          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    
# Line 1640  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 1668  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 1697  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 1713  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 1810  void MainWindow::on_action_edit_script() Line 2764  void MainWindow::on_action_edit_script()
2764      if (!script) return;      if (!script) return;
2765    
2766      ScriptEditor* editor = new ScriptEditor;      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);      editor->setScript(script);
2774      //editor->reparent(*this);      //editor->reparent(*this);
2775      editor->show();      editor->show();
# Line 1875  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 1967  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 2019  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 2044  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 2111  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 2131  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 2155  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 2181  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 2204  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 2228  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 2246  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 2379  void MainWindow::script_name_changed(con Line 3500  void MainWindow::script_name_changed(con
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 2425  void MainWindow::on_action_combine_instr Line 3567  void MainWindow::on_action_combine_instr
3567      delete d;      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) {  void MainWindow::mergeFiles(const std::vector<std::string>& filenames) {
3592      struct _Source {      struct _Source {
3593          std::vector<RIFF::File*> riffs;          std::vector<RIFF::File*> riffs;
# Line 2492  void MainWindow::mergeFiles(const std::v Line 3655  void MainWindow::mergeFiles(const std::v
3655          );          );
3656      }      }
3657    
3658      // Note: requires that this file already has a filename !      // Finally save gig file persistently to disk ...
3659      this->file->Save();      //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() {  void MainWindow::on_action_merge_files() {
# Line 2530  void MainWindow::on_action_merge_files() Line 3712  void MainWindow::on_action_merge_files()
3712      // show warning in the file picker dialog      // show warning in the file picker dialog
3713      Gtk::HBox descriptionArea;      Gtk::HBox descriptionArea;
3714      descriptionArea.set_spacing(15);      descriptionArea.set_spacing(15);
3715      Gtk::Image warningIcon(Gtk::Stock::DIALOG_WARNING, Gtk::IconSize(Gtk::ICON_SIZE_DIALOG));      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);      descriptionArea.pack_start(warningIcon, Gtk::PACK_SHRINK);
3719  #if GTKMM_MAJOR_VERSION < 3  #if GTKMM_MAJOR_VERSION < 3
3720      view::WrapLabel description;      view::WrapLabel description;
# Line 2553  void MainWindow::on_action_merge_files() Line 3737  void MainWindow::on_action_merge_files()
3737      descriptionArea.show_all();      descriptionArea.show_all();
3738    
3739      if (dialog.run() == Gtk::RESPONSE_OK) {      if (dialog.run() == Gtk::RESPONSE_OK) {
3740          printf("on_action_merge_files self=%x\n", Glib::Threads::Thread::self());          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();          std::vector<std::string> filenames = dialog.get_filenames();
3743    
3744          // merge the selected files to the currently open .gig file          // merge the selected files to the currently open .gig file
# Line 2565  void MainWindow::on_action_merge_files() Line 3750  void MainWindow::on_action_merge_files()
3750          }          }
3751    
3752          // update GUI          // update GUI
3753          __refreshEntireGUI();                  __refreshEntireGUI();
3754      }      }
3755  }  }
3756    
# Line 2583  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 2640  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.2604  
changed lines
  Added in v.3197

  ViewVC Help
Powered by ViewVC