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

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

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

revision 1420 by persson, Sun Oct 14 15:39:32 2007 UTC revision 2626 by schoenebeck, Thu Jun 12 15:12:00 2014 UTC
# Line 1  Line 1 
1  /*  /*
2   * Copyright (C) 2006, 2007 Andreas Persson   * Copyright (C) 2006-2014 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 18  Line 18 
18   */   */
19    
20  #include "dimregionchooser.h"  #include "dimregionchooser.h"
21    #include <cairomm/context.h>
22  #include <gdkmm/cursor.h>  #include <gdkmm/cursor.h>
23    #include <gdkmm/general.h>
24    #include <glibmm/stringutils.h>
25    #include <gtkmm/stock.h>
26    #include <glibmm/ustring.h>
27    #include <gtkmm/messagedialog.h>
28    
29    #include "global.h"
30    
31    // taken from gdk/gdkkeysyms.h
32    // (define on demand, to avoid unnecessary dev lib package build dependency)
33    #ifndef GDK_KEY_Control_L
34    # define GDK_KEY_Control_L 0xffe3
35    #endif
36    #ifndef GDK_KEY_Control_R
37    # define GDK_KEY_Control_R 0xffe4
38    #endif
39    
40    static std::map<gig::dimension_t,int> caseOfDimRegion(gig::DimensionRegion* dr, bool* isValidZone) {
41        std::map<gig::dimension_t,int> dimCase;
42        if (!dr) {
43            *isValidZone = false;
44            return dimCase;
45        }
46    
47        gig::Region* rgn = (gig::Region*) dr->GetParent();
48    
49        // find the dimension region index of the passed dimension region
50        int drIndex;
51        for (drIndex = 0; drIndex < 256; ++drIndex)
52            if (rgn->pDimensionRegions[drIndex] == dr)
53                break;
54    
55        // not found in region, something's horribly wrong
56        if (drIndex == 256) {
57            fprintf(stderr, "DimRegionChooser: ERROR: index of dim region not found!\n");
58            *isValidZone = false;
59            return std::map<gig::dimension_t,int>();
60        }
61    
62        for (int d = 0, baseBits = 0; d < rgn->Dimensions; ++d) {
63            const int bits = rgn->pDimensionDefinitions[d].bits;
64            dimCase[rgn->pDimensionDefinitions[d].dimension] =
65                (drIndex >> baseBits) & ((1 << bits) - 1);
66            baseBits += bits;
67            // there are also DimensionRegion objects of unused zones, skip them
68            if (dimCase[rgn->pDimensionDefinitions[d].dimension] >= rgn->pDimensionDefinitions[d].zones) {
69                *isValidZone = false;
70                return std::map<gig::dimension_t,int>();
71            }
72        }
73    
74        *isValidZone = true;
75        return dimCase;
76    }
77    
78  DimRegionChooser::DimRegionChooser()  DimRegionChooser::DimRegionChooser(Gtk::Window& window) :
79        red("#8070ff"),
80        black("black"),
81        white("white")
82  {  {
     // get_window() would return 0 because the Gdk::Window has not yet been realized  
     // So we can only allocate the colors here - the rest will happen in on_realize().  
     Glib::RefPtr<Gdk::Colormap> colormap = get_default_colormap();  
   
     black = Gdk::Color("black");  
     white = Gdk::Color("white");  
     red = Gdk::Color("#8070ff");  
     blue = Gdk::Color("blue");  
     green = Gdk::Color("green");  
   
     colormap->alloc_color(black);  
     colormap->alloc_color(white);  
     colormap->alloc_color(red);  
     colormap->alloc_color(blue);  
     colormap->alloc_color(green);  
83      instrument = 0;      instrument = 0;
84      region = 0;      region = 0;
85      dimregno = -1;      maindimregno = -1;
86      focus_line = 0;      focus_line = 0;
87      resize.active = false;      resize.active = false;
88      cursor_is_resize = false;      cursor_is_resize = false;
89      h = 20;      h = 20;
90      w = 800;      multiSelectKeyDown = false;
91      set_flags(Gtk::CAN_FOCUS);      set_can_focus();
92    
93        actionGroup = Gtk::ActionGroup::create();
94        actionGroup->add(
95            Gtk::Action::create("SplitDimZone", _("Split Dimensions Zone")),
96            sigc::mem_fun(*this, &DimRegionChooser::split_dimension_zone)
97        );
98        actionGroup->add(
99            Gtk::Action::create("DeleteDimZone", _("Delete Dimension Zone")),
100            sigc::mem_fun(*this, &DimRegionChooser::delete_dimension_zone)
101        );
102    
103        uiManager = Gtk::UIManager::create();
104        uiManager->insert_action_group(actionGroup);
105        Glib::ustring ui_info =
106            "<ui>"
107            "  <popup name='PopupMenuInsideDimRegion'>"
108            "    <menuitem action='SplitDimZone'/>"
109            "    <menuitem action='DeleteDimZone'/>"
110            "  </popup>"
111    //         "  <popup name='PopupMenuOutsideDimRegion'>"
112    //         "    <menuitem action='Add'/>"
113    //         "  </popup>"
114            "</ui>";
115        uiManager->add_ui_from_string(ui_info);
116    
117        popup_menu_inside_dimregion = dynamic_cast<Gtk::Menu*>(
118            uiManager->get_widget("/PopupMenuInsideDimRegion"));
119    //     popup_menu_outside_dimregion = dynamic_cast<Gtk::Menu*>(
120    //         uiManager->get_widget("/PopupMenuOutsideDimRegion"));
121    
122      add_events(Gdk::BUTTON_PRESS_MASK | Gdk::POINTER_MOTION_MASK |      add_events(Gdk::BUTTON_PRESS_MASK | Gdk::POINTER_MOTION_MASK |
123                 Gdk::POINTER_MOTION_HINT_MASK);                 Gdk::POINTER_MOTION_HINT_MASK);
124    
125      for (int i = 0 ; i < 256 ; i++) dimvalue[i] = 0;      labels_changed = true;
126    
127        set_tooltip_text(_(
128            "Right click here for options on altering dimension zones. Press and "
129            "hold CTRL key for selecting multiple dimension zones simultaniously."
130        ));
131        
132        window.signal_key_press_event().connect(
133            sigc::mem_fun(*this, &DimRegionChooser::onKeyPressed)
134        );
135        window.signal_key_release_event().connect(
136            sigc::mem_fun(*this, &DimRegionChooser::onKeyReleased)
137        );
138  }  }
139    
140  DimRegionChooser::~DimRegionChooser()  DimRegionChooser::~DimRegionChooser()
141  {  {
142  }  }
143    
144  void DimRegionChooser::on_realize()  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
145    bool DimRegionChooser::on_expose_event(GdkEventExpose* e)
146  {  {
147      // We need to call the base on_realize()      double clipx1 = e->area.x;
148      Gtk::DrawingArea::on_realize();      double clipx2 = e->area.x + e->area.width;
149        double clipy1 = e->area.y;
150      // Now we can allocate any additional resources we need      double clipy2 = e->area.y + e->area.height;
151      Glib::RefPtr<Gdk::Window> window = get_window();  
152      gc = Gdk::GC::create(window);      const Cairo::RefPtr<Cairo::Context>& cr =
153  }          get_window()->create_cairo_context();
154    #else
155    bool DimRegionChooser::on_draw(const Cairo::RefPtr<Cairo::Context>& cr)
156    {
157        double clipx1, clipx2, clipy1, clipy2;
158        cr->get_clip_extents(clipx1, clipy1, clipx2, clipy2);
159    #endif
160    
 bool DimRegionChooser::on_expose_event(GdkEventExpose* event)  
 {  
161      if (!region) return true;      if (!region) return true;
162    
163      // This is where we draw on the window      // This is where we draw on the window
164      Glib::RefPtr<Gdk::Window> window = get_window();      int w = get_width();
165      Glib::RefPtr<Pango::Context> context = get_pango_context();      Glib::RefPtr<Pango::Context> context = get_pango_context();
166    
167      Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);      Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
168        cr->set_line_width(1);
169    
     window->clear();  
   
     // draw labels on the left (reflecting the dimension type)  
170      int y = 0;      int y = 0;
171      double maxwidth = 0;      if (labels_changed || label_width - 10 > clipx1) {
172      for (int i = 0 ; i < region->Dimensions ; i++) {          // draw labels on the left (reflecting the dimension type)
173          int nbZones = region->pDimensionDefinitions[i].zones;          double maxwidth = 0;
174          if (nbZones) {          for (int i = 0 ; i < region->Dimensions ; i++) {
175              const char* dstr;              int nbZones = region->pDimensionDefinitions[i].zones;
176              char dstrbuf[10];              if (nbZones) {
177              switch (region->pDimensionDefinitions[i].dimension) {                  const char* dstr;
178              case gig::dimension_none: dstr="none"; break;                  char dstrbuf[10];
179              case gig::dimension_samplechannel: dstr="samplechannel"; break;                  switch (region->pDimensionDefinitions[i].dimension) {
180              case gig::dimension_layer: dstr="layer"; break;                  case gig::dimension_none: dstr=_("none"); break;
181              case gig::dimension_velocity: dstr="velocity"; break;                  case gig::dimension_samplechannel: dstr=_("samplechannel");
182              case gig::dimension_channelaftertouch: dstr="channelaftertouch"; break;                      break;
183              case gig::dimension_releasetrigger: dstr="releasetrigger"; break;                  case gig::dimension_layer: dstr=_("layer"); break;
184              case gig::dimension_keyboard: dstr="keyboard"; break;                  case gig::dimension_velocity: dstr=_("velocity"); break;
185              case gig::dimension_roundrobin: dstr="roundrobin"; break;                  case gig::dimension_channelaftertouch:
186              case gig::dimension_random: dstr="random"; break;                      dstr=_("channelaftertouch"); break;
187              case gig::dimension_smartmidi: dstr="smartmidi"; break;                  case gig::dimension_releasetrigger:
188              case gig::dimension_roundrobinkeyboard: dstr="roundrobinkeyboard"; break;                      dstr=_("releasetrigger"); break;
189              case gig::dimension_modwheel: dstr="modwheel"; break;                  case gig::dimension_keyboard: dstr=_("keyswitching"); break;
190              case gig::dimension_breath: dstr="breath"; break;                  case gig::dimension_roundrobin: dstr=_("roundrobin"); break;
191              case gig::dimension_foot: dstr="foot"; break;                  case gig::dimension_random: dstr=_("random"); break;
192              case gig::dimension_portamentotime: dstr="portamentotime"; break;                  case gig::dimension_smartmidi: dstr=_("smartmidi"); break;
193              case gig::dimension_effect1: dstr="effect1"; break;                  case gig::dimension_roundrobinkeyboard:
194              case gig::dimension_effect2: dstr="effect2"; break;                      dstr=_("roundrobinkeyboard"); break;
195              case gig::dimension_genpurpose1: dstr="genpurpose1"; break;                  case gig::dimension_modwheel: dstr=_("modwheel"); break;
196              case gig::dimension_genpurpose2: dstr="genpurpose2"; break;                  case gig::dimension_breath: dstr=_("breath"); break;
197              case gig::dimension_genpurpose3: dstr="genpurpose3"; break;                  case gig::dimension_foot: dstr=_("foot"); break;
198              case gig::dimension_genpurpose4: dstr="genpurpose4"; break;                  case gig::dimension_portamentotime:
199              case gig::dimension_sustainpedal: dstr="sustainpedal"; break;                      dstr=_("portamentotime"); break;
200              case gig::dimension_portamento: dstr="portamento"; break;                  case gig::dimension_effect1: dstr=_("effect1"); break;
201              case gig::dimension_sostenutopedal: dstr="sostenutopedal"; break;                  case gig::dimension_effect2: dstr=_("effect2"); break;
202              case gig::dimension_softpedal: dstr="softpedal"; break;                  case gig::dimension_genpurpose1: dstr=_("genpurpose1"); break;
203              case gig::dimension_genpurpose5: dstr="genpurpose5"; break;                  case gig::dimension_genpurpose2: dstr=_("genpurpose2"); break;
204              case gig::dimension_genpurpose6: dstr="genpurpose6"; break;                  case gig::dimension_genpurpose3: dstr=_("genpurpose3"); break;
205              case gig::dimension_genpurpose7: dstr="genpurpose7"; break;                  case gig::dimension_genpurpose4: dstr=_("genpurpose4"); break;
206              case gig::dimension_genpurpose8: dstr="genpurpose8"; break;                  case gig::dimension_sustainpedal:
207              case gig::dimension_effect1depth: dstr="effect1depth"; break;                      dstr=_("sustainpedal"); break;
208              case gig::dimension_effect2depth: dstr="effect2depth"; break;                  case gig::dimension_portamento: dstr=_("portamento"); break;
209              case gig::dimension_effect3depth: dstr="effect3depth"; break;                  case gig::dimension_sostenutopedal:
210              case gig::dimension_effect4depth: dstr="effect4depth"; break;                      dstr=_("sostenutopedal"); break;
211              case gig::dimension_effect5depth: dstr="effect5depth"; break;                  case gig::dimension_softpedal: dstr=_("softpedal"); break;
212              default:                  case gig::dimension_genpurpose5: dstr=_("genpurpose5"); break;
213                  sprintf(dstrbuf, "%d",                  case gig::dimension_genpurpose6: dstr=_("genpurpose6"); break;
214                          region->pDimensionDefinitions[i].dimension);                  case gig::dimension_genpurpose7: dstr=_("genpurpose7"); break;
215                  dstr = dstrbuf;                  case gig::dimension_genpurpose8: dstr=_("genpurpose8"); break;
216                  break;                  case gig::dimension_effect1depth:
217              }                      dstr=_("effect1depth"); break;
218              layout->set_text(dstr);                  case gig::dimension_effect2depth:
219                        dstr=_("effect2depth"); break;
220              Pango::Rectangle rectangle = layout->get_logical_extents();                  case gig::dimension_effect3depth:
221              double text_w = double(rectangle.get_width()) / Pango::SCALE;                      dstr=_("effect3depth"); break;
222              if (text_w > maxwidth) maxwidth = text_w;                  case gig::dimension_effect4depth:
223              double text_h = double(rectangle.get_height()) / Pango::SCALE;                      dstr=_("effect4depth"); break;
224              Glib::RefPtr<const Gdk::GC> fg = get_style()->get_fg_gc(get_state());                  case gig::dimension_effect5depth:
225              window->draw_layout(fg, 4, int(y + (h - text_h) / 2 + 0.5), layout);                      dstr=_("effect5depth"); break;
226                    default:
227                        sprintf(dstrbuf, "%d",
228                                region->pDimensionDefinitions[i].dimension);
229                        dstr = dstrbuf;
230                        break;
231                    }
232                    layout->set_text(dstr);
233    
234                    Pango::Rectangle rectangle = layout->get_logical_extents();
235                    double text_w = double(rectangle.get_width()) / Pango::SCALE;
236                    if (text_w > maxwidth) maxwidth = text_w;
237    
238                    if (y + h > clipy1 && y < clipy2 && text_w >= clipx1) {
239                        double text_h = double(rectangle.get_height()) /
240                            Pango::SCALE;
241    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
242                        const Gdk::Color fg = get_style()->get_fg(get_state());
243    #else
244                        const Gdk::RGBA fg =
245                            get_style_context()->get_color(get_state_flags());
246    #endif
247                        Gdk::Cairo::set_source_rgba(cr, fg);
248                        cr->move_to(4, int(y + (h - text_h) / 2 + 0.5));
249    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
250                        pango_cairo_show_layout(cr->cobj(), layout->gobj());
251    #else
252                        layout->show_in_cairo_context(cr);
253    #endif
254                    }
255                }
256                y += h;
257          }          }
258          y += h;          label_width = int(maxwidth + 10);
259            labels_changed = false;
260      }      }
261        if (label_width >= clipx2) return true;
262    
263      // draw dimensions' zones areas      // draw dimensions' zones areas
264      y = 0;      y = 0;
265      int bitpos = 0;      int bitpos = 0;
     label_width = int(maxwidth + 10);  
266      for (int i = 0 ; i < region->Dimensions ; i++) {      for (int i = 0 ; i < region->Dimensions ; i++) {
267          int nbZones = region->pDimensionDefinitions[i].zones;          int nbZones = region->pDimensionDefinitions[i].zones;
268          if (nbZones) {          if (nbZones) {
269              // draw focus rectangle around dimension's label and zones              const gig::dimension_t dimension = region->pDimensionDefinitions[i].dimension;
             if (has_focus() && focus_line == i) {  
                 Gdk::Rectangle farea(0, y, 150, 20);  
                 get_style()->paint_focus(window, get_state(), farea, *this, "",  
                                          0, y, label_width, 20);  
             }  
   
             Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();  
             // draw top and bottom lines of dimension's zones  
             window->draw_line(black, label_width, y, w - 1, y);  
             window->draw_line(black, w - 1, y + h - 1, label_width, y + h - 1);  
             // erase whole dimension's zones area  
             window->draw_rectangle(get_style()->get_white_gc(), true,  
                                    label_width + 1, y + 1, (w - label_width - 2), h - 2);  
   
             int c = 0;  
             if (dimregno >= 0) {  
                 int mask = ~(((1 << region->pDimensionDefinitions[i].bits) - 1) << bitpos);  
                 c = dimregno & mask; // mask away this dimension  
             }  
             bool customsplits =  
                 ((region->pDimensionDefinitions[i].split_type == gig::split_type_normal &&  
                  region->pDimensionRegions[c]->DimensionUpperLimits[i]) ||  
                 (region->pDimensionDefinitions[i].dimension == gig::dimension_velocity &&  
                  region->pDimensionRegions[c]->VelocityUpperLimit));  
270    
271              // draw dimension's zone borders              if (y >= clipy2) break;
272              if (customsplits) {              if (y + h > clipy1) {
273                  window->draw_line(black, label_width, y + 1, label_width, y + h - 2);                  // draw focus rectangle around dimension's label and zones
274                  for (int j = 0 ; j < nbZones ; j++) {                  if (has_focus() && focus_line == i) {
275                      gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
276                      int upperLimit = d->DimensionUpperLimits[i];                      Gdk::Rectangle farea(0, y, 150, 20);
277                      if (!upperLimit) upperLimit = d->VelocityUpperLimit;                      get_style()->paint_focus(get_window(), get_state(), farea,
278                      int v = upperLimit + 1;                                               *this, "",
279                      int x = int((w - label_width - 1) * v / 128.0 + 0.5);                                               0, y, label_width, 20);
280                      window->draw_line(black, label_width + x, y + 1, label_width + x, y + h - 2);  #else
281                  }                      get_style_context()->render_focus(cr,
282              } else {                                                        0, y, label_width, 20);
283                  for (int j = 0 ; j <= nbZones ; j++) {  #endif
284                      int x = int((w - label_width - 1) * j / double(nbZones) + 0.5);                  }
285                      window->draw_line(black, label_width + x, y + 1, label_width + x, y + h - 2);  
286                  }                  // draw top and bottom lines of dimension's zones
287              }                  Gdk::Cairo::set_source_rgba(cr, black);
288                    cr->move_to(label_width, y + 0.5);
289                    cr->line_to(w, y + 0.5);
290                    cr->move_to(w, y + h - 0.5);
291                    cr->line_to(label_width, y + h - 0.5);
292                    cr->stroke();
293    
294                    // erase whole dimension's zones area
295                    Gdk::Cairo::set_source_rgba(cr, white);
296                    cr->rectangle(label_width + 1, y + 1,
297                                  (w - label_width - 2), h - 2);
298                    cr->fill();
299    
300                    int c = 0;
301                    if (maindimregno >= 0) {
302                        int mask =
303                            ~(((1 << region->pDimensionDefinitions[i].bits) - 1) <<
304                              bitpos);
305                        c = maindimregno & mask; // mask away this dimension
306                    }
307                    bool customsplits =
308                        ((region->pDimensionDefinitions[i].split_type ==
309                          gig::split_type_normal &&
310                          region->pDimensionRegions[c]->DimensionUpperLimits[i]) ||
311                         (region->pDimensionDefinitions[i].dimension ==
312                          gig::dimension_velocity &&
313                          region->pDimensionRegions[c]->VelocityUpperLimit));
314    
315              // draw fill for currently selected zone                  // draw dimension zones
316              if (dimregno >= 0) {                  Gdk::Cairo::set_source_rgba(cr, black);
                 gc->set_foreground(red);  
                 int dr = (dimregno >> bitpos) & ((1 << region->pDimensionDefinitions[i].bits) - 1);  
317                  if (customsplits) {                  if (customsplits) {
318                      int x1 = 0;                      cr->move_to(label_width + 0.5, y + 1);
319                        cr->line_to(label_width + 0.5, y + h - 1);
320                        int prevX = label_width;
321                        int prevUpperLimit = 0;
322    
323                      for (int j = 0 ; j < nbZones ; j++) {                      for (int j = 0 ; j < nbZones ; j++) {
324                          gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];                          // draw dimension zone's borders for custom splits
325                            gig::DimensionRegion* d =
326                                region->pDimensionRegions[c + (j << bitpos)];
327                          int upperLimit = d->DimensionUpperLimits[i];                          int upperLimit = d->DimensionUpperLimits[i];
328                          if (!upperLimit) upperLimit = d->VelocityUpperLimit;                          if (!upperLimit) upperLimit = d->VelocityUpperLimit;
329                          int v = upperLimit + 1;                          int v = upperLimit + 1;
330                          int x2 = int((w - label_width - 1) * v / 128.0 + 0.5);                          int x = int((w - label_width - 1) * v / 128.0 + 0.5) +
331                          if (j == dr && x1 < x2) {                              label_width;
332                              window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1,                          if (x >= clipx2) break;
333                                                     (x2 - x1) - 1, h - 2);                          if (x < clipx1) continue;
334                              break;                          Gdk::Cairo::set_source_rgba(cr, black);
335                            cr->move_to(x + 0.5, y + 1);
336                            cr->line_to(x + 0.5, y + h - 1);
337                            cr->stroke();
338    
339                            // draw fill for zone
340                            bool isSelectedZone = this->dimzones[dimension].count(j);
341                            Gdk::Cairo::set_source_rgba(cr, isSelectedZone ? red : white);
342                            cr->rectangle(prevX + 1, y + 1, x - prevX - 1, h - 1);
343                            cr->fill();
344    
345                            // draw text showing the beginning of the dimension zone
346                            // as numeric value to the user
347                            {
348                                Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
349                                layout->set_text(Glib::Ascii::dtostr(prevUpperLimit));
350                                Gdk::Cairo::set_source_rgba(cr, black);
351                                Pango::Rectangle rect = layout->get_logical_extents();
352                                // get the text dimensions
353                                int text_width, text_height;
354                                layout->get_pixel_size(text_width, text_height);
355                                // move text to the left end of the dimension zone
356                                cr->move_to(prevX + 3, y + 1);
357    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
358                                pango_cairo_show_layout(cr->cobj(), layout->gobj());
359    #else
360                                layout->show_in_cairo_context(cr);
361    #endif
362                            }
363                            // draw text showing the end of the dimension zone
364                            // as numeric value to the user
365                            {
366                                Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
367                                layout->set_text(Glib::Ascii::dtostr(upperLimit));
368                                Gdk::Cairo::set_source_rgba(cr, black);
369                                Pango::Rectangle rect = layout->get_logical_extents();
370                                // get the text dimensions
371                                int text_width, text_height;
372                                layout->get_pixel_size(text_width, text_height);
373                                // move text to the left end of the dimension zone
374                                cr->move_to(x - 3 - text_width, y + 1);
375    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
376                                pango_cairo_show_layout(cr->cobj(), layout->gobj());
377    #else
378                                layout->show_in_cairo_context(cr);
379    #endif
380                          }                          }
381                          x1 = x2;  
382                            prevX = x;
383                            prevUpperLimit = upperLimit;
384                      }                      }
385                  } else {                  } else {
386                      if (dr < nbZones) {                      int prevX = 0;
387                          int x1 = int((w - label_width - 1) * dr / double(nbZones) + 0.5);                      for (int j = 0 ; j <= nbZones ; j++) {
388                          int x2 = int((w - label_width - 1) * (dr + 1) / double(nbZones) + 0.5);                          // draw dimension zone's borders for normal splits
389                          window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1,                          int x = int((w - label_width - 1) * j /
390                                                 (x2 - x1) - 1, h - 2);                                      double(nbZones) + 0.5) + label_width;
391                      }                          if (x >= clipx2) break;
392                            if (x < clipx1) continue;
393                            Gdk::Cairo::set_source_rgba(cr, black);
394                            cr->move_to(x + 0.5, y + 1);
395                            cr->line_to(x + 0.5, y + h - 1);
396                            cr->stroke();
397    
398                            if (j != 0) {
399                                // draw fill for zone
400                                bool isSelectedZone = this->dimzones[dimension].count(j-1);
401                                Gdk::Cairo::set_source_rgba(cr, isSelectedZone ? red : white);
402                                cr->rectangle(prevX + 1, y + 1, x - prevX - 1, h - 1);
403                                cr->fill();
404    
405                                // draw text showing the beginning of the dimension zone
406                                // as numeric value to the user
407                                {
408                                    Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
409                                    layout->set_text(Glib::Ascii::dtostr((j-1) * 128/nbZones));
410                                    Gdk::Cairo::set_source_rgba(cr, black);
411                                    Pango::Rectangle rect = layout->get_logical_extents();
412                                    // get the text dimensions
413                                    int text_width, text_height;
414                                    layout->get_pixel_size(text_width, text_height);
415                                    // move text to the left end of the dimension zone
416                                    cr->move_to(prevX + 3, y + 1);
417    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
418                                    pango_cairo_show_layout(cr->cobj(), layout->gobj());
419    #else
420                                    layout->show_in_cairo_context(cr);
421    #endif
422                                }
423                                // draw text showing the end of the dimension zone
424                                // as numeric value to the user
425                                {
426                                    Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
427                                    layout->set_text(Glib::Ascii::dtostr(j * 128/nbZones - 1));
428                                    Gdk::Cairo::set_source_rgba(cr, black);
429                                    Pango::Rectangle rect = layout->get_logical_extents();
430                                    // get the text dimensions
431                                    int text_width, text_height;
432                                    layout->get_pixel_size(text_width, text_height);
433                                    // move text to the left end of the dimension zone
434                                    cr->move_to(x - 3 - text_width, y + 1);
435    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
436                                    pango_cairo_show_layout(cr->cobj(), layout->gobj());
437    #else
438                                    layout->show_in_cairo_context(cr);
439    #endif
440                                }
441                            }
442                            prevX = x;
443                        }      
444                  }                  }
445              }              }
   
446              y += h;              y += h;
447          }          }
448          bitpos += region->pDimensionDefinitions[i].bits;          bitpos += region->pDimensionDefinitions[i].bits;
# Line 228  bool DimRegionChooser::on_expose_event(G Line 451  bool DimRegionChooser::on_expose_event(G
451      return true;      return true;
452  }  }
453    
 void DimRegionChooser::on_size_request(GtkRequisition* requisition)  
 {  
     *requisition = GtkRequisition();  
     requisition->height = region ? nbDimensions * 20 : 0;  
     requisition->width = 800;  
 }  
   
454  void DimRegionChooser::set_region(gig::Region* region)  void DimRegionChooser::set_region(gig::Region* region)
455  {  {
456      this->region = region;      this->region = region;
457      dimregno = 0;      maindimregno = 0;
458      nbDimensions = 0;      nbDimensions = 0;
459      if (region) {      if (region) {
460          int bitcount = 0;          int bitcount = 0;
# Line 246  void DimRegionChooser::set_region(gig::R Line 462  void DimRegionChooser::set_region(gig::R
462              if (region->pDimensionDefinitions[dim].bits == 0) continue;              if (region->pDimensionDefinitions[dim].bits == 0) continue;
463              nbDimensions++;              nbDimensions++;
464    
465              int z = std::min(dimvalue[region->pDimensionDefinitions[dim].dimension],              int z = std::min(maindimcase[region->pDimensionDefinitions[dim].dimension],
466                               region->pDimensionDefinitions[dim].zones - 1);                               region->pDimensionDefinitions[dim].zones - 1);
467              int mask =              maindimregno |= (z << bitcount);
                 ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) <<  
                   bitcount);  
             dimregno &= mask;  
             dimregno |= (z << bitcount);  
468              bitcount += region->pDimensionDefinitions[dim].bits;              bitcount += region->pDimensionDefinitions[dim].bits;
469          }          }
         dimreg = region->pDimensionRegions[dimregno];  
     } else {  
         dimreg = 0;  
470      }      }
471      dimregion_selected();      dimregion_selected();
472        set_size_request(800, region ? nbDimensions * 20 : 0);
473    
474        labels_changed = true;
475      queue_resize();      queue_resize();
476        queue_draw();
477  }  }
478    
479  bool DimRegionChooser::on_button_release_event(GdkEventButton* event)  void DimRegionChooser::refresh_all() {
480        set_region(region);
481    }
482    
483    void DimRegionChooser::get_dimregions(const gig::Region* region, bool stereo,
484                                          std::set<gig::DimensionRegion*>& dimregs) const
485  {  {
486      if (resize.active) {      for (int iDimRgn = 0; iDimRgn < 256; ++iDimRgn) {
487          get_window()->pointer_ungrab(event->time);          gig::DimensionRegion* dimRgn = region->pDimensionRegions[iDimRgn];
488          resize.active = false;          if (!dimRgn) continue;
489            bool isValidZone;
490            std::map<gig::dimension_t,int> dimCase = caseOfDimRegion(dimRgn, &isValidZone);
491            if (!isValidZone) continue;
492            for (std::map<gig::dimension_t,int>::const_iterator it = dimCase.begin();
493                 it != dimCase.end(); ++it)
494            {
495                if (stereo && it->first == gig::dimension_samplechannel) continue; // is selected
496    
497          if (region->pDimensionDefinitions[resize.dimension].dimension == gig::dimension_velocity) {              std::map<gig::dimension_t, std::set<int> >::const_iterator itSelectedDimension =
498                    this->dimzones.find(it->first);
499                if (itSelectedDimension != this->dimzones.end() &&
500                    itSelectedDimension->second.count(it->second)) continue; // is selected
501    
502              int bitpos = 0;              goto notSelected;
503              for (int j = 0 ; j < resize.dimension ; j++) {          }
504                  bitpos += region->pDimensionDefinitions[j].bits;  
505            dimregs.insert(dimRgn);
506    
507            notSelected:
508            ;
509        }
510    }
511    
512    void DimRegionChooser::update_after_resize()
513    {
514        if (region->pDimensionDefinitions[resize.dimension].dimension == gig::dimension_velocity) {
515    
516            int bitpos = 0;
517            for (int j = 0 ; j < resize.dimension ; j++) {
518                bitpos += region->pDimensionDefinitions[j].bits;
519            }
520            int mask =
521                ~(((1 << region->pDimensionDefinitions[resize.dimension].bits) - 1) << bitpos);
522            int c = maindimregno & mask; // mask away this dimension
523    
524            if (region->pDimensionRegions[c]->DimensionUpperLimits[resize.dimension] == 0) {
525                // the velocity dimension didn't previously have
526                // custom v3 splits, so we initialize all splits with
527                // default values
528                int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
529                for (int j = 0 ; j < nbZones ; j++) {
530                    gig::DimensionRegion* d = region->pDimensionRegions[c + (j << bitpos)];
531                    d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);
532                }
533            }
534            if (region->pDimensionRegions[c]->VelocityUpperLimit == 0) {
535                // the velocity dimension didn't previously have
536                // custom v2 splits, so we initialize all splits with
537                // default values
538                int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
539                for (int j = 0 ; j < nbZones ; j++) {
540                    gig::DimensionRegion* d = region->pDimensionRegions[c + (j << bitpos)];
541                    d->VelocityUpperLimit = int(128.0 * (j + 1) / nbZones - 1);
542              }              }
543              int mask =          }
544                  ~(((1 << region->pDimensionDefinitions[resize.dimension].bits) - 1) << bitpos);  
545              int c = dimregno & mask; // mask away this dimension          gig::DimensionRegion* d = region->pDimensionRegions[c + resize.offset];
546            // update both v2 and v3 values
547              if (region->pDimensionRegions[c]->DimensionUpperLimits[resize.dimension] == 0) {          d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
548                  // the velocity dimension didn't previously have          d->VelocityUpperLimit = resize.pos - 1;
549                  // custom v3 splits, so we initialize all splits with  
550                  // default values      } else {
551                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;          for (int i = 0 ; i < region->DimensionRegions ; ) {
552                  for (int j = 0 ; j < nbZones ; j++) {  
553                      gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];              if (region->pDimensionRegions[i]->DimensionUpperLimits[resize.dimension] == 0) {
554                      d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);                  // the dimension didn't previously have custom
555                    // limits, so we have to set default limits for
556                    // all the dimension regions
557                    int bitpos = 0;
558                    for (int j = 0 ; j < resize.dimension ; j++) {
559                        bitpos += region->pDimensionDefinitions[j].bits;
560                  }                  }
             }  
             if (region->pDimensionRegions[c]->VelocityUpperLimit == 0) {  
                 // the velocity dimension didn't previously have  
                 // custom v2 splits, so we initialize all splits with  
                 // default values  
561                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
562    
563                  for (int j = 0 ; j < nbZones ; j++) {                  for (int j = 0 ; j < nbZones ; j++) {
564                      gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];                      gig::DimensionRegion* d = region->pDimensionRegions[i + (j << bitpos)];
565                      d->VelocityUpperLimit = int(128.0 * (j + 1) / nbZones - 1);                      d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);
566                  }                  }
567              }              }
568                gig::DimensionRegion* d = region->pDimensionRegions[i + resize.offset];
             gig::DimensionRegion *d = region->pDimensionRegions[c + resize.offset];  
             // update both v2 and v3 values  
569              d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;              d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
             d->VelocityUpperLimit = resize.pos - 1;  
   
         } else {  
             for (int i = 0 ; i < region->DimensionRegions ; ) {  
570    
571                  if (region->pDimensionRegions[i]->DimensionUpperLimits[resize.dimension] == 0) {              int bitpos = 0;
572                      // the dimension didn't previously have custom              int j;
573                      // limits, so we have to set default limits for              for (j = 0 ; j < region->Dimensions ; j++) {
574                      // all the dimension regions                  if (j != resize.dimension) {
575                      int bitpos = 0;                      int maxzones = 1 << region->pDimensionDefinitions[j].bits;
576                      for (int j = 0 ; j < resize.dimension ; j++) {                      int dimj = (i >> bitpos) & (maxzones - 1);
577                          bitpos += region->pDimensionDefinitions[j].bits;                      if (dimj + 1 < region->pDimensionDefinitions[j].zones) break;
                     }  
                     int nbZones = region->pDimensionDefinitions[resize.dimension].zones;  
   
                     for (int j = 0 ; j < nbZones ; j++) {  
                         gig::DimensionRegion *d = region->pDimensionRegions[i + (j << bitpos)];  
                         d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);  
                     }  
                 }  
                 gig::DimensionRegion *d = region->pDimensionRegions[i + resize.offset];  
                 d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;  
   
                 int bitpos = 0;  
                 int j;  
                 for (j = 0 ; j < region->Dimensions ; j++) {  
                     if (j != resize.dimension) {  
                         int maxzones = 1 << region->pDimensionDefinitions[j].bits;  
                         int dimj = (i >> bitpos) & (maxzones - 1);  
                         if (dimj + 1 < region->pDimensionDefinitions[j].zones) break;  
                     }  
                     bitpos += region->pDimensionDefinitions[j].bits;  
578                  }                  }
579                  if (j == region->Dimensions) break;                  bitpos += region->pDimensionDefinitions[j].bits;
                 i = (i & ~((1 << bitpos) - 1)) + (1 << bitpos);  
580              }              }
581                if (j == region->Dimensions) break;
582                i = (i & ~((1 << bitpos) - 1)) + (1 << bitpos);
583          }          }
584        }
585    }
586    
587    bool DimRegionChooser::on_button_release_event(GdkEventButton* event)
588    {
589        if (resize.active) {
590    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
591            get_window()->pointer_ungrab(event->time);
592    #else
593            Glib::wrap(event->device, true)->ungrab(event->time);
594    #endif
595            resize.active = false;
596    
597          region_changed();          region_changed();
598    
599          if (!is_in_resize_zone(event->x, event->y) && cursor_is_resize) {          if (!is_in_resize_zone(event->x, event->y) && cursor_is_resize) {
# Line 352  bool DimRegionChooser::on_button_release Line 606  bool DimRegionChooser::on_button_release
606    
607  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)
608  {  {
609        int w = get_width();
610      if (region && event->y < nbDimensions * h &&      if (region && event->y < nbDimensions * h &&
611          event->x >= label_width && event->x < w) {          event->x >= label_width && event->x < w) {
612    
613          if (is_in_resize_zone(event->x, event->y)) {          if (is_in_resize_zone(event->x, event->y)) {
614              Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
615              get_window()->pointer_grab(false,              get_window()->pointer_grab(false,
616                                         Gdk::BUTTON_RELEASE_MASK |                                         Gdk::BUTTON_RELEASE_MASK |
617                                         Gdk::POINTER_MOTION_MASK |                                         Gdk::POINTER_MOTION_MASK |
618                                         Gdk::POINTER_MOTION_HINT_MASK,                                         Gdk::POINTER_MOTION_HINT_MASK,
619                                         double_arrow, event->time);                                         Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW),
620                                           event->time);
621    #else
622                Glib::wrap(event->device, true)->grab(get_window(),
623                                                      Gdk::OWNERSHIP_NONE,
624                                                      false,
625                                                      Gdk::BUTTON_RELEASE_MASK |
626                                                      Gdk::POINTER_MOTION_MASK |
627                                                      Gdk::POINTER_MOTION_HINT_MASK,
628                                                      Gdk::Cursor::create(Gdk::SB_H_DOUBLE_ARROW),
629                                                      event->time);
630    #endif
631              resize.active = true;              resize.active = true;
632          } else {          } else {
633              int ydim = int(event->y / h);              int ydim = int(event->y / h);
# Line 380  bool DimRegionChooser::on_button_press_e Line 646  bool DimRegionChooser::on_button_press_e
646              }              }
647    
648              int i = dim;              int i = dim;
649              if (dimregno < 0) dimregno = 0;              if (maindimregno < 0) maindimregno = 0;
650              int mask = ~(((1 << region->pDimensionDefinitions[i].bits) - 1) << bitpos);              int mask = ~(((1 << region->pDimensionDefinitions[i].bits) - 1) << bitpos);
651              int c = dimregno & mask; // mask away this dimension              int c = this->maindimregno & mask; // mask away this dimension
652    
653              bool customsplits =              bool customsplits =
654                  ((region->pDimensionDefinitions[i].split_type == gig::split_type_normal &&                  ((region->pDimensionDefinitions[i].split_type == gig::split_type_normal &&
# Line 394  bool DimRegionChooser::on_button_press_e Line 660  bool DimRegionChooser::on_button_press_e
660    
661                  if (region->pDimensionRegions[c]->DimensionUpperLimits[i]) {                  if (region->pDimensionRegions[c]->DimensionUpperLimits[i]) {
662                      for (z = 0 ; z < nbZones ; z++) {                      for (z = 0 ; z < nbZones ; z++) {
663                          gig::DimensionRegion *d = region->pDimensionRegions[c + (z << bitpos)];                          gig::DimensionRegion* d = region->pDimensionRegions[c + (z << bitpos)];
664                          if (val <= d->DimensionUpperLimits[i]) break;                          if (val <= d->DimensionUpperLimits[i]) break;
665                      }                      }
666                  } else {                  } else {
667                      for (z = 0 ; z < nbZones ; z++) {                      for (z = 0 ; z < nbZones ; z++) {
668                          gig::DimensionRegion *d = region->pDimensionRegions[c + (z << bitpos)];                          gig::DimensionRegion* d = region->pDimensionRegions[c + (z << bitpos)];
669                          if (val <= d->VelocityUpperLimit) break;                          if (val <= d->VelocityUpperLimit) break;
670                      }                      }
671                  }                  }
# Line 412  bool DimRegionChooser::on_button_press_e Line 678  bool DimRegionChooser::on_button_press_e
678                     region->pDimensionDefinitions[dim].split_type,                     region->pDimensionDefinitions[dim].split_type,
679                     region->pDimensionDefinitions[dim].zones,                     region->pDimensionDefinitions[dim].zones,
680                     region->pDimensionDefinitions[dim].zone_size);                     region->pDimensionDefinitions[dim].zone_size);
681              dimvalue[region->pDimensionDefinitions[dim].dimension] = z;              this->maindimcase[region->pDimensionDefinitions[dim].dimension] = z;
682                this->maindimregno = c | (z << bitpos);
683              dimregno = c | (z << bitpos);              this->maindimtype = region->pDimensionDefinitions[dim].dimension;
684    
685                if (multiSelectKeyDown) {
686                    if (dimzones[this->maindimtype].count(z)) {
687                        if (dimzones[this->maindimtype].size() > 1) {
688                            dimzones[this->maindimtype].erase(z);
689                        }
690                    } else {
691                        dimzones[this->maindimtype].insert(z);
692                    }
693                } else {
694                    this->dimzones.clear();
695                    for (std::map<gig::dimension_t,int>::const_iterator it = this->maindimcase.begin();
696                         it != this->maindimcase.end(); ++it)
697                    {
698                        this->dimzones[it->first].insert(it->second);
699                    }
700                }
701    
702              focus_line = dim;              focus_line = dim;
703              if (has_focus()) queue_draw();              if (has_focus()) queue_draw();
704              else grab_focus();              else grab_focus();
             dimreg = region->pDimensionRegions[dimregno];  
705              dimregion_selected();              dimregion_selected();
706    
707                if (event->button == 3) {
708                    printf("dimregion right click\n");
709                    popup_menu_inside_dimregion->popup(event->button, event->time);
710                }
711    
712                queue_draw();
713          }          }
714      }      }
715      return true;      return true;
# Line 434  bool DimRegionChooser::on_motion_notify_ Line 723  bool DimRegionChooser::on_motion_notify_
723      window->get_pointer(x, y, state);      window->get_pointer(x, y, state);
724    
725      if (resize.active) {      if (resize.active) {
726            int w = get_width();
727          int k = int((x - label_width) * 128.0 / (w - label_width - 1) + 0.5);          int k = int((x - label_width) * 128.0 / (w - label_width - 1) + 0.5);
728    
729          if (k < resize.min) k = resize.min;          if (k < resize.min) k = resize.min;
# Line 442  bool DimRegionChooser::on_motion_notify_ Line 732  bool DimRegionChooser::on_motion_notify_
732          if (k < 2) k = 2; // k is upper limit + 1, upper limit 0 is forbidden          if (k < 2) k = 2; // k is upper limit + 1, upper limit 0 is forbidden
733    
734          if (k != resize.pos) {          if (k != resize.pos) {
             Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();  
             Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();  
   
735              int prevx = int((w - label_width - 1) * resize.pos / 128.0 + 0.5) + label_width;              int prevx = int((w - label_width - 1) * resize.pos / 128.0 + 0.5) + label_width;
736              int x = int((w - label_width - 1) * k / 128.0 + 0.5) + label_width;              int x = int((w - label_width - 1) * k / 128.0 + 0.5) + label_width;
737              int y = resize.dimension * h;              int y = resize.dimension * h;
738                int x1, x2;
739              if (resize.selected == resize.none) {              if (k > resize.pos) {
740                  if (resize.pos != resize.min && resize.pos != resize.max) {                  x1 = prevx;
741                      window->draw_line(white, prevx, y + 1, prevx, y + h - 2);                  x2 = x;
                 }  
742              } else {              } else {
743                  gc->set_foreground(red);                  x1 = x;
744                    x2 = prevx;
                 Glib::RefPtr<const Gdk::GC> left;  
                 Glib::RefPtr<const Gdk::GC> right;  
                 if (resize.selected == resize.left) {  
                     left = gc;  
                     right = white;  
                 } else {  
                     left = white;  
                     right = gc;  
                 }  
   
                 if (k > resize.pos) {  
                     int xx = resize.pos == resize.min ? 1 : 0;  
                     window->draw_rectangle(left, true,  
                                            prevx + xx, y + 1, x - prevx - xx, h - 2);  
                 } else {  
                     int xx = resize.pos == resize.max ? 0 : 1;  
                     window->draw_rectangle(right, true,  
                                            x, y + 1, prevx - x + xx, h - 2);  
                 }  
745              }              }
746              window->draw_line(black, x, y + 1, x, y + h - 2);              Gdk::Rectangle rect(x1, y + 1, x2 - x1 + 1, h - 2);
747    
748              resize.pos = k;              resize.pos = k;
749                update_after_resize();
750                get_window()->invalidate_rect(rect, false); // not sufficient ...
751                queue_draw(); // ... so do a complete redraw instead.
752          }          }
753      } else {      } else {
754          if (is_in_resize_zone(x, y)) {          if (is_in_resize_zone(x, y)) {
755              if (!cursor_is_resize) {              if (!cursor_is_resize) {
756                  Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
757                  window->set_cursor(double_arrow);                  window->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));
758    #else
759                    window->set_cursor(Gdk::Cursor::create(Gdk::SB_H_DOUBLE_ARROW));
760    #endif
761                  cursor_is_resize = true;                  cursor_is_resize = true;
762              }              }
763          } else if (cursor_is_resize) {          } else if (cursor_is_resize) {
# Line 496  bool DimRegionChooser::on_motion_notify_ Line 770  bool DimRegionChooser::on_motion_notify_
770    
771  bool DimRegionChooser::is_in_resize_zone(double x, double y)  bool DimRegionChooser::is_in_resize_zone(double x, double y)
772  {  {
773        int w = get_width();
774      if (region && y < nbDimensions * h && x >= label_width && x < w) {      if (region && y < nbDimensions * h && x >= label_width && x < w) {
775          int ydim = int(y / h);          int ydim = int(y / h);
776          int dim;          int dim;
# Line 509  bool DimRegionChooser::is_in_resize_zone Line 784  bool DimRegionChooser::is_in_resize_zone
784          int nbZones = region->pDimensionDefinitions[dim].zones;          int nbZones = region->pDimensionDefinitions[dim].zones;
785    
786          int c = 0;          int c = 0;
787          if (dimregno >= 0) {          if (maindimregno >= 0) {
788              int mask = ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) << bitpos);              int mask = ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) << bitpos);
789              c = dimregno & mask; // mask away this dimension              c = maindimregno & mask; // mask away this dimension
790          }          }
791          const bool customsplits =          const bool customsplits =
792              ((region->pDimensionDefinitions[dim].split_type == gig::split_type_normal &&              ((region->pDimensionDefinitions[dim].split_type == gig::split_type_normal &&
# Line 523  bool DimRegionChooser::is_in_resize_zone Line 798  bool DimRegionChooser::is_in_resize_zone
798          if (region->pDimensionDefinitions[dim].split_type != gig::split_type_bit) {          if (region->pDimensionDefinitions[dim].split_type != gig::split_type_bit) {
799              int prev_limit = 0;              int prev_limit = 0;
800              for (int iZone = 0 ; iZone < nbZones - 1 ; iZone++) {              for (int iZone = 0 ; iZone < nbZones - 1 ; iZone++) {
801                  gig::DimensionRegion *d = region->pDimensionRegions[c + (iZone << bitpos)];                  gig::DimensionRegion* d = region->pDimensionRegions[c + (iZone << bitpos)];
802                  const int upperLimit =                  const int upperLimit =
803                      (customsplits) ?                      (customsplits) ?
804                          (d->DimensionUpperLimits[dim]) ?                          (d->DimensionUpperLimits[dim]) ?
# Line 538  bool DimRegionChooser::is_in_resize_zone Line 813  bool DimRegionChooser::is_in_resize_zone
813                      resize.pos = limit;                      resize.pos = limit;
814                      resize.min = prev_limit;                      resize.min = prev_limit;
815    
816                      int dr = (dimregno >> bitpos) &                      int dr = (maindimregno >> bitpos) &
817                          ((1 << region->pDimensionDefinitions[dim].bits) - 1);                          ((1 << region->pDimensionDefinitions[dim].bits) - 1);
818                      resize.selected = dr == iZone ? resize.left :                      resize.selected = dr == iZone ? resize.left :
819                          dr == iZone + 1 ? resize.right : resize.none;                          dr == iZone + 1 ? resize.right : resize.none;
820    
821                      iZone++;                      iZone++;
822                      gig::DimensionRegion *d = region->pDimensionRegions[c + (iZone << bitpos)];                      gig::DimensionRegion* d = region->pDimensionRegions[c + (iZone << bitpos)];
823    
824                      const int upperLimit =                      const int upperLimit =
825                          (customsplits) ?                          (customsplits) ?
# Line 615  bool DimRegionChooser::on_focus(Gtk::Dir Line 890  bool DimRegionChooser::on_focus(Gtk::Dir
890          // TODO: öka eller minska värde!          // TODO: öka eller minska värde!
891      }      }
892  }  }
893    
894    void DimRegionChooser::split_dimension_zone() {    
895        printf("split_dimension_zone() type=%d, zone=%d\n", maindimtype, maindimcase[maindimtype]);
896        try {
897            region->SplitDimensionZone(maindimtype, maindimcase[maindimtype]);
898        } catch (RIFF::Exception e) {
899            Gtk::MessageDialog msg(e.Message, false, Gtk::MESSAGE_ERROR);
900            msg.run();
901        } catch (...) {
902            Glib::ustring txt = _("An unknown exception occurred!");
903            Gtk::MessageDialog msg(txt, false, Gtk::MESSAGE_ERROR);
904            msg.run();
905        }
906        refresh_all();
907    }
908    
909    void DimRegionChooser::delete_dimension_zone() {
910        printf("delete_dimension_zone() type=%d, zone=%d\n", maindimtype, maindimcase[maindimtype]);
911        try {
912            region->DeleteDimensionZone(maindimtype, maindimcase[maindimtype]);
913        } catch (RIFF::Exception e) {
914            Gtk::MessageDialog msg(e.Message, false, Gtk::MESSAGE_ERROR);
915            msg.run();
916        } catch (...) {
917            Glib::ustring txt = _("An unknown exception occurred!");
918            Gtk::MessageDialog msg(txt, false, Gtk::MESSAGE_ERROR);
919            msg.run();
920        }
921        refresh_all();
922    }
923    
924    bool DimRegionChooser::onKeyPressed(GdkEventKey* key) {
925        //printf("key down\n");
926        if (key->keyval == GDK_KEY_Control_L || key->keyval == GDK_KEY_Control_R)
927            multiSelectKeyDown = true;
928    }
929    
930    bool DimRegionChooser::onKeyReleased(GdkEventKey* key) {
931        //printf("key up\n");
932        if (key->keyval == GDK_KEY_Control_L || key->keyval == GDK_KEY_Control_R)
933            multiSelectKeyDown = false;
934    }
935    
936    gig::DimensionRegion* DimRegionChooser::get_main_dimregion() const {
937        if (!region) return NULL;
938        return region->pDimensionRegions[maindimregno];
939    }

Legend:
Removed from v.1420  
changed lines
  Added in v.2626

  ViewVC Help
Powered by ViewVC