/[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 1261 by persson, Thu Jul 5 17:12:20 2007 UTC revision 2841 by persson, Sun Aug 30 10:00:49 2015 UTC
# Line 1  Line 1 
1  /*  /*
2   * Copyright (C) 2006, 2007 Andreas Persson   * Copyright (C) 2006-2015 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  DimRegionChooser::DimRegionChooser()      // 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(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 = 24;
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++) {      labels_changed = true;
126          dimvalue_from[i] = 0;  
127          dimvalue_to[i] = 1;      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_expose_event(GdkEventExpose* event)  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    
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, h);
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, h);
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, h);
283                  for (int j = 0 ; j <= nbZones ; j++) {  #endif
                     int x = int((w - label_width - 1) * j / double(nbZones) + 0.5);  
                     window->draw_line(black, label_width + x, y + 1, label_width + x, y + h - 2);  
284                  }                  }
             }  
285    
286              // draw fill for currently selected zone                  // draw top and bottom lines of dimension's zones
287              if (dimregno >= 0) {                  Gdk::Cairo::set_source_rgba(cr, black);
288                  gc->set_foreground(red);                  cr->move_to(label_width, y + 0.5);
289                  int dr = (dimregno >> bitpos) & ((1 << region->pDimensionDefinitions[i].bits) - 1);                  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 dimension zones
316                    Gdk::Cairo::set_source_rgba(cr, black);
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 = -1;
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+1));
350                                Gdk::Cairo::set_source_rgba(cr, black);
351                                // get the text dimensions
352                                int text_width, text_height;
353                                layout->get_pixel_size(text_width, text_height);
354                                // move text to the left end of the dimension zone
355                                cr->move_to(prevX + 3, y + (h - text_height) / 2);
356    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
357                                pango_cairo_show_layout(cr->cobj(), layout->gobj());
358    #else
359                                layout->show_in_cairo_context(cr);
360    #endif
361                            }
362                            // draw text showing the end of the dimension zone
363                            // as numeric value to the user
364                            {
365                                Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
366                                layout->set_text(Glib::Ascii::dtostr(upperLimit));
367                                Gdk::Cairo::set_source_rgba(cr, black);
368                                // get the text dimensions
369                                int text_width, text_height;
370                                layout->get_pixel_size(text_width, text_height);
371                                // move text to the left end of the dimension zone
372                                cr->move_to(x - 3 - text_width, y + (h - text_height) / 2);
373    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
374                                pango_cairo_show_layout(cr->cobj(), layout->gobj());
375    #else
376                                layout->show_in_cairo_context(cr);
377    #endif
378                          }                          }
379                          x1 = x2;  
380                            prevX = x;
381                            prevUpperLimit = upperLimit;
382                      }                      }
383                  } else {                  } else {
384                      if (dr < nbZones) {                      int prevX = 0;
385                          int x1 = int((w - label_width - 1) * dr / double(nbZones) + 0.5);                      for (int j = 0 ; j <= nbZones ; j++) {
386                          int x2 = int((w - label_width - 1) * (dr + 1) / double(nbZones) + 0.5);                          // draw dimension zone's borders for normal splits
387                          window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1,                          int x = int((w - label_width - 1) * j /
388                                                 (x2 - x1) - 1, h - 2);                                      double(nbZones) + 0.5) + label_width;
389                      }                          if (x >= clipx2) break;
390                            if (x < clipx1) continue;
391                            Gdk::Cairo::set_source_rgba(cr, black);
392                            cr->move_to(x + 0.5, y + 1);
393                            cr->line_to(x + 0.5, y + h - 1);
394                            cr->stroke();
395    
396                            if (j != 0) {
397                                // draw fill for zone
398                                bool isSelectedZone = this->dimzones[dimension].count(j-1);
399                                Gdk::Cairo::set_source_rgba(cr, isSelectedZone ? red : white);
400                                cr->rectangle(prevX + 1, y + 1, x - prevX - 1, h - 1);
401                                cr->fill();
402    
403                                // draw text showing the beginning of the dimension zone
404                                // as numeric value to the user
405                                {
406                                    Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
407                                    layout->set_text(Glib::Ascii::dtostr((j-1) * 128/nbZones));
408                                    Gdk::Cairo::set_source_rgba(cr, black);
409                                    // get the text dimensions
410                                    int text_width, text_height;
411                                    layout->get_pixel_size(text_width, text_height);
412                                    // move text to the left end of the dimension zone
413                                    cr->move_to(prevX + 3, y + (h - text_height) / 2);
414    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
415                                    pango_cairo_show_layout(cr->cobj(), layout->gobj());
416    #else
417                                    layout->show_in_cairo_context(cr);
418    #endif
419                                }
420                                // draw text showing the end of the dimension zone
421                                // as numeric value to the user
422                                {
423                                    Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
424                                    layout->set_text(Glib::Ascii::dtostr(j * 128/nbZones - 1));
425                                    Gdk::Cairo::set_source_rgba(cr, black);
426                                    // get the text dimensions
427                                    int text_width, text_height;
428                                    layout->get_pixel_size(text_width, text_height);
429                                    // move text to the left end of the dimension zone
430                                    cr->move_to(x - 3 - text_width, y + (h - text_height) / 2);
431    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
432                                    pango_cairo_show_layout(cr->cobj(), layout->gobj());
433    #else
434                                    layout->show_in_cairo_context(cr);
435    #endif
436                                }
437                            }
438                            prevX = x;
439                        }      
440                  }                  }
441              }              }
   
442              y += h;              y += h;
443          }          }
444          bitpos += region->pDimensionDefinitions[i].bits;          bitpos += region->pDimensionDefinitions[i].bits;
# Line 231  bool DimRegionChooser::on_expose_event(G Line 447  bool DimRegionChooser::on_expose_event(G
447      return true;      return true;
448  }  }
449    
 void DimRegionChooser::on_size_request(GtkRequisition* requisition)  
 {  
     printf("DimRegionChooser::on_size_request\n");  
     *requisition = GtkRequisition();  
     requisition->height = region ? nbDimensions * 20 : 0;  
     requisition->width = 800;  
 }  
   
450  void DimRegionChooser::set_region(gig::Region* region)  void DimRegionChooser::set_region(gig::Region* region)
451  {  {
452      this->region = region;      this->region = region;
453      dimregno = 0;      maindimregno = 0;
454      nbDimensions = 0;      nbDimensions = 0;
455      if (region) {      if (region) {
456          int bitcount = 0;          int bitcount = 0;
# Line 250  void DimRegionChooser::set_region(gig::R Line 458  void DimRegionChooser::set_region(gig::R
458              if (region->pDimensionDefinitions[dim].bits == 0) continue;              if (region->pDimensionDefinitions[dim].bits == 0) continue;
459              nbDimensions++;              nbDimensions++;
460    
461              int from = dimvalue_from[region->pDimensionDefinitions[dim].dimension];              int z = std::min(maindimcase[region->pDimensionDefinitions[dim].dimension],
462              int to = dimvalue_to[region->pDimensionDefinitions[dim].dimension];                               region->pDimensionDefinitions[dim].zones - 1);
463              int z;              maindimregno |= (z << bitcount);
             switch (region->pDimensionDefinitions[dim].split_type) {  
             case gig::split_type_normal:  
                 z = int((to + from) / 2.0 / region->pDimensionDefinitions[dim].zone_size);  
                 break;  
             case gig::split_type_bit:  
                 z = std::min(from, region->pDimensionDefinitions[dim].zones - 1);  
                 break;  
             }  
             int mask =  
                 ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) <<  
                   bitcount);  
             dimregno &= mask;  
             dimregno |= (z << bitcount);  
464              bitcount += region->pDimensionDefinitions[dim].bits;              bitcount += region->pDimensionDefinitions[dim].bits;
465          }          }
         dimreg = region->pDimensionRegions[dimregno];  
     } else {  
         dimreg = 0;  
466      }      }
467      dimregion_selected();      dimregion_selected();
468        set_size_request(800, region ? nbDimensions * h : 0);
469    
470        labels_changed = true;
471      queue_resize();      queue_resize();
472        queue_draw();
473  }  }
474    
475  bool DimRegionChooser::on_button_release_event(GdkEventButton* event)  void DimRegionChooser::refresh_all() {
476        set_region(region);
477    }
478    
479    void DimRegionChooser::get_dimregions(const gig::Region* region, bool stereo,
480                                          std::set<gig::DimensionRegion*>& dimregs) const
481  {  {
482      if (resize.active) {      for (int iDimRgn = 0; iDimRgn < 256; ++iDimRgn) {
483          get_window()->pointer_ungrab(event->time);          gig::DimensionRegion* dimRgn = region->pDimensionRegions[iDimRgn];
484          resize.active = false;          if (!dimRgn) continue;
485            bool isValidZone;
486            std::map<gig::dimension_t,int> dimCase = caseOfDimRegion(dimRgn, &isValidZone);
487            if (!isValidZone) continue;
488            for (std::map<gig::dimension_t,int>::const_iterator it = dimCase.begin();
489                 it != dimCase.end(); ++it)
490            {
491                if (stereo && it->first == gig::dimension_samplechannel) continue; // is selected
492    
493          if (region->pDimensionDefinitions[resize.dimension].dimension == gig::dimension_velocity) {              std::map<gig::dimension_t, std::set<int> >::const_iterator itSelectedDimension =
494                    this->dimzones.find(it->first);
495                if (itSelectedDimension != this->dimzones.end() &&
496                    itSelectedDimension->second.count(it->second)) continue; // is selected
497    
498              int bitpos = 0;              goto notSelected;
499              for (int j = 0 ; j < resize.dimension ; j++) {          }
500                  bitpos += region->pDimensionDefinitions[j].bits;  
501            dimregs.insert(dimRgn);
502    
503            notSelected:
504            ;
505        }
506    }
507    
508    void DimRegionChooser::update_after_resize()
509    {
510        if (region->pDimensionDefinitions[resize.dimension].dimension == gig::dimension_velocity) {
511    
512            int bitpos = 0;
513            for (int j = 0 ; j < resize.dimension ; j++) {
514                bitpos += region->pDimensionDefinitions[j].bits;
515            }
516            int mask =
517                ~(((1 << region->pDimensionDefinitions[resize.dimension].bits) - 1) << bitpos);
518            int c = maindimregno & mask; // mask away this dimension
519    
520            if (region->pDimensionRegions[c]->DimensionUpperLimits[resize.dimension] == 0) {
521                // the velocity dimension didn't previously have
522                // custom v3 splits, so we initialize all splits with
523                // default values
524                int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
525                for (int j = 0 ; j < nbZones ; j++) {
526                    gig::DimensionRegion* d = region->pDimensionRegions[c + (j << bitpos)];
527                    d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);
528                }
529            }
530            if (region->pDimensionRegions[c]->VelocityUpperLimit == 0) {
531                // the velocity dimension didn't previously have
532                // custom v2 splits, so we initialize all splits with
533                // default values
534                int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
535                for (int j = 0 ; j < nbZones ; j++) {
536                    gig::DimensionRegion* d = region->pDimensionRegions[c + (j << bitpos)];
537                    d->VelocityUpperLimit = int(128.0 * (j + 1) / nbZones - 1);
538              }              }
539              int mask =          }
540                  ~(((1 << region->pDimensionDefinitions[resize.dimension].bits) - 1) << bitpos);  
541              int c = dimregno & mask; // mask away this dimension          gig::DimensionRegion* d = region->pDimensionRegions[c + resize.offset];
542            // update both v2 and v3 values
543              if (region->pDimensionRegions[c]->DimensionUpperLimits[resize.dimension] == 0) {          d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
544                  // the velocity dimension didn't previously have          d->VelocityUpperLimit = resize.pos - 1;
545                  // custom v3 splits, so we initialize all splits with  
546                  // default values      } else {
547                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;          for (int i = 0 ; i < region->DimensionRegions ; ) {
548                  for (int j = 0 ; j < nbZones ; j++) {  
549                      gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];              if (region->pDimensionRegions[i]->DimensionUpperLimits[resize.dimension] == 0) {
550                      d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);                  // the dimension didn't previously have custom
551                    // limits, so we have to set default limits for
552                    // all the dimension regions
553                    int bitpos = 0;
554                    for (int j = 0 ; j < resize.dimension ; j++) {
555                        bitpos += region->pDimensionDefinitions[j].bits;
556                  }                  }
             }  
             if (region->pDimensionRegions[c]->VelocityUpperLimit == 0) {  
                 // the velocity dimension didn't previously have  
                 // custom v2 splits, so we initialize all splits with  
                 // default values  
557                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;                  int nbZones = region->pDimensionDefinitions[resize.dimension].zones;
558    
559                  for (int j = 0 ; j < nbZones ; j++) {                  for (int j = 0 ; j < nbZones ; j++) {
560                      gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];                      gig::DimensionRegion* d = region->pDimensionRegions[i + (j << bitpos)];
561                      d->VelocityUpperLimit = int(128.0 * (j + 1) / nbZones - 1);                      d->DimensionUpperLimits[resize.dimension] = int(128.0 * (j + 1) / nbZones - 1);
562                  }                  }
563              }              }
564                gig::DimensionRegion* d = region->pDimensionRegions[i + resize.offset];
             gig::DimensionRegion *d = region->pDimensionRegions[c + resize.offset];  
             // update both v2 and v3 values  
565              d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;              d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
             d->VelocityUpperLimit = resize.pos - 1;  
566    
567          } else {              int bitpos = 0;
568              for (int i = 0 ; i < region->DimensionRegions ; ) {              int j;
569                for (j = 0 ; j < region->Dimensions ; j++) {
570                  if (region->pDimensionRegions[i]->DimensionUpperLimits[resize.dimension] == 0) {                  if (j != resize.dimension) {
571                      // the dimension didn't previously have custom                      int maxzones = 1 << region->pDimensionDefinitions[j].bits;
572                      // limits, so we have to set default limits for                      int dimj = (i >> bitpos) & (maxzones - 1);
573                      // all the dimension regions                      if (dimj + 1 < region->pDimensionDefinitions[j].zones) break;
                     int bitpos = 0;  
                     for (int j = 0 ; j < resize.dimension ; j++) {  
                         bitpos += region->pDimensionDefinitions[j].bits;  
                     }  
                     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;  
574                  }                  }
575                  if (j == region->Dimensions) break;                  bitpos += region->pDimensionDefinitions[j].bits;
                 i = (i & ~((1 << bitpos) - 1)) + (1 << bitpos);  
576              }              }
577                if (j == region->Dimensions) break;
578                i = (i & ~((1 << bitpos) - 1)) + (1 << bitpos);
579          }          }
580        }
581    }
582    
583    bool DimRegionChooser::on_button_release_event(GdkEventButton* event)
584    {
585        if (resize.active) {
586    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
587            get_window()->pointer_ungrab(event->time);
588    #else
589            Glib::wrap(event->device, true)->ungrab(event->time);
590    #endif
591            resize.active = false;
592    
593          region_changed();          region_changed();
594    
595          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 365  bool DimRegionChooser::on_button_release Line 602  bool DimRegionChooser::on_button_release
602    
603  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)
604  {  {
605        int w = get_width();
606      if (region && event->y < nbDimensions * h &&      if (region && event->y < nbDimensions * h &&
607          event->x >= label_width && event->x < w) {          event->x >= label_width && event->x < w) {
608    
609          if (is_in_resize_zone(event->x, event->y)) {          if (is_in_resize_zone(event->x, event->y)) {
610              Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
611              get_window()->pointer_grab(false,              get_window()->pointer_grab(false,
612                                         Gdk::BUTTON_RELEASE_MASK |                                         Gdk::BUTTON_RELEASE_MASK |
613                                         Gdk::POINTER_MOTION_MASK |                                         Gdk::POINTER_MOTION_MASK |
614                                         Gdk::POINTER_MOTION_HINT_MASK,                                         Gdk::POINTER_MOTION_HINT_MASK,
615                                         double_arrow, event->time);                                         Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW),
616                                           event->time);
617    #else
618                Glib::wrap(event->device, true)->grab(get_window(),
619                                                      Gdk::OWNERSHIP_NONE,
620                                                      false,
621                                                      Gdk::BUTTON_RELEASE_MASK |
622                                                      Gdk::POINTER_MOTION_MASK |
623                                                      Gdk::POINTER_MOTION_HINT_MASK,
624                                                      Gdk::Cursor::create(Gdk::SB_H_DOUBLE_ARROW),
625                                                      event->time);
626    #endif
627              resize.active = true;              resize.active = true;
628          } else {          } else {
629              int ydim = int(event->y / h);              int ydim = int(event->y / h);
# Line 393  bool DimRegionChooser::on_button_press_e Line 642  bool DimRegionChooser::on_button_press_e
642              }              }
643    
644              int i = dim;              int i = dim;
645              if (dimregno < 0) dimregno = 0;              if (maindimregno < 0) maindimregno = 0;
646              int mask = ~(((1 << region->pDimensionDefinitions[i].bits) - 1) << bitpos);              int mask = ~(((1 << region->pDimensionDefinitions[i].bits) - 1) << bitpos);
647              int c = dimregno & mask; // mask away this dimension              int c = this->maindimregno & mask; // mask away this dimension
648    
649              bool customsplits =              bool customsplits =
650                  ((region->pDimensionDefinitions[i].split_type == gig::split_type_normal &&                  ((region->pDimensionDefinitions[i].split_type == gig::split_type_normal &&
# Line 407  bool DimRegionChooser::on_button_press_e Line 656  bool DimRegionChooser::on_button_press_e
656    
657                  if (region->pDimensionRegions[c]->DimensionUpperLimits[i]) {                  if (region->pDimensionRegions[c]->DimensionUpperLimits[i]) {
658                      for (z = 0 ; z < nbZones ; z++) {                      for (z = 0 ; z < nbZones ; z++) {
659                          gig::DimensionRegion *d = region->pDimensionRegions[c + (z << bitpos)];                          gig::DimensionRegion* d = region->pDimensionRegions[c + (z << bitpos)];
660                          if (val <= d->DimensionUpperLimits[i]) break;                          if (val <= d->DimensionUpperLimits[i]) break;
661                      }                      }
662                  } else {                  } else {
663                      for (z = 0 ; z < nbZones ; z++) {                      for (z = 0 ; z < nbZones ; z++) {
664                          gig::DimensionRegion *d = region->pDimensionRegions[c + (z << bitpos)];                          gig::DimensionRegion* d = region->pDimensionRegions[c + (z << bitpos)];
665                          if (val <= d->VelocityUpperLimit) break;                          if (val <= d->VelocityUpperLimit) break;
666                      }                      }
667                  }                  }
# Line 425  bool DimRegionChooser::on_button_press_e Line 674  bool DimRegionChooser::on_button_press_e
674                     region->pDimensionDefinitions[dim].split_type,                     region->pDimensionDefinitions[dim].split_type,
675                     region->pDimensionDefinitions[dim].zones,                     region->pDimensionDefinitions[dim].zones,
676                     region->pDimensionDefinitions[dim].zone_size);                     region->pDimensionDefinitions[dim].zone_size);
677  #if 0              this->maindimcase[region->pDimensionDefinitions[dim].dimension] = z;
678              switch (region->pDimensionDefinitions[dim].split_type) {              this->maindimregno = c | (z << bitpos);
679              case gig::split_type_normal:              this->maindimtype = region->pDimensionDefinitions[dim].dimension;
680                  dimvalue_from[region->pDimensionDefinitions[dim].dimension] =  
681                      int(z * region->pDimensionDefinitions[dim].zone_size);              if (multiSelectKeyDown) {
682                  dimvalue_to[region->pDimensionDefinitions[dim].dimension] =                  if (dimzones[this->maindimtype].count(z)) {
683                      int((z + 1) * region->pDimensionDefinitions[dim].zone_size) - 1;                      if (dimzones[this->maindimtype].size() > 1) {
684                  break;                          dimzones[this->maindimtype].erase(z);
685              case gig::split_type_bit:                      }
686                  dimvalue_from[region->pDimensionDefinitions[dim].dimension] = z;                  } else {
687                  dimvalue_to[region->pDimensionDefinitions[dim].dimension] = z + 1;                      dimzones[this->maindimtype].insert(z);
688                  break;                  }
689                } else {
690                    this->dimzones.clear();
691                    for (std::map<gig::dimension_t,int>::const_iterator it = this->maindimcase.begin();
692                         it != this->maindimcase.end(); ++it)
693                    {
694                        this->dimzones[it->first].insert(it->second);
695                    }
696              }              }
 #endif  
   
             dimregno = c | (z << bitpos);  
697    
698              focus_line = dim;              focus_line = dim;
699              if (has_focus()) queue_draw();              if (has_focus()) queue_draw();
700              else grab_focus();              else grab_focus();
             dimreg = region->pDimensionRegions[dimregno];  
701              dimregion_selected();              dimregion_selected();
702    
703                if (event->button == 3) {
704                    printf("dimregion right click\n");
705                    popup_menu_inside_dimregion->popup(event->button, event->time);
706                }
707    
708                queue_draw();
709          }          }
710      }      }
711      return true;      return true;
# Line 460  bool DimRegionChooser::on_motion_notify_ Line 719  bool DimRegionChooser::on_motion_notify_
719      window->get_pointer(x, y, state);      window->get_pointer(x, y, state);
720    
721      if (resize.active) {      if (resize.active) {
722            int w = get_width();
723          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);
724    
725          if (k < resize.min) k = resize.min;          if (k < resize.min) k = resize.min;
# Line 468  bool DimRegionChooser::on_motion_notify_ Line 728  bool DimRegionChooser::on_motion_notify_
728          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
729    
730          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();  
   
731              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;
732              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;
733              int y = resize.dimension * h;              int y = resize.dimension * h;
734                int x1, x2;
735              if (resize.selected == resize.none) {              if (k > resize.pos) {
736                  if (resize.pos != resize.min && resize.pos != resize.max) {                  x1 = prevx;
737                      window->draw_line(white, prevx, y + 1, prevx, y + h - 2);                  x2 = x;
                 }  
738              } else {              } else {
739                  gc->set_foreground(red);                  x1 = x;
740                    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);  
                 }  
741              }              }
742              window->draw_line(black, x, y + 1, x, y + h - 2);              Gdk::Rectangle rect(x1, y + 1, x2 - x1 + 1, h - 2);
743    
744              resize.pos = k;              resize.pos = k;
745                update_after_resize();
746                get_window()->invalidate_rect(rect, false); // not sufficient ...
747                queue_draw(); // ... so do a complete redraw instead.
748          }          }
749      } else {      } else {
750          if (is_in_resize_zone(x, y)) {          if (is_in_resize_zone(x, y)) {
751              if (!cursor_is_resize) {              if (!cursor_is_resize) {
752                  Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
753                  window->set_cursor(double_arrow);                  window->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));
754    #else
755                    window->set_cursor(Gdk::Cursor::create(Gdk::SB_H_DOUBLE_ARROW));
756    #endif
757                  cursor_is_resize = true;                  cursor_is_resize = true;
758              }              }
759          } else if (cursor_is_resize) {          } else if (cursor_is_resize) {
# Line 522  bool DimRegionChooser::on_motion_notify_ Line 766  bool DimRegionChooser::on_motion_notify_
766    
767  bool DimRegionChooser::is_in_resize_zone(double x, double y)  bool DimRegionChooser::is_in_resize_zone(double x, double y)
768  {  {
769        int w = get_width();
770      if (region && y < nbDimensions * h && x >= label_width && x < w) {      if (region && y < nbDimensions * h && x >= label_width && x < w) {
771          int ydim = int(y / h);          int ydim = int(y / h);
772          int dim;          int dim;
# Line 535  bool DimRegionChooser::is_in_resize_zone Line 780  bool DimRegionChooser::is_in_resize_zone
780          int nbZones = region->pDimensionDefinitions[dim].zones;          int nbZones = region->pDimensionDefinitions[dim].zones;
781    
782          int c = 0;          int c = 0;
783          if (dimregno >= 0) {          if (maindimregno >= 0) {
784              int mask = ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) << bitpos);              int mask = ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) << bitpos);
785              c = dimregno & mask; // mask away this dimension              c = maindimregno & mask; // mask away this dimension
786          }          }
787          const bool customsplits =          const bool customsplits =
788              ((region->pDimensionDefinitions[dim].split_type == gig::split_type_normal &&              ((region->pDimensionDefinitions[dim].split_type == gig::split_type_normal &&
# Line 549  bool DimRegionChooser::is_in_resize_zone Line 794  bool DimRegionChooser::is_in_resize_zone
794          if (region->pDimensionDefinitions[dim].split_type != gig::split_type_bit) {          if (region->pDimensionDefinitions[dim].split_type != gig::split_type_bit) {
795              int prev_limit = 0;              int prev_limit = 0;
796              for (int iZone = 0 ; iZone < nbZones - 1 ; iZone++) {              for (int iZone = 0 ; iZone < nbZones - 1 ; iZone++) {
797                  gig::DimensionRegion *d = region->pDimensionRegions[c + (iZone << bitpos)];                  gig::DimensionRegion* d = region->pDimensionRegions[c + (iZone << bitpos)];
798                  const int upperLimit =                  const int upperLimit =
799                      (customsplits) ?                      (customsplits) ?
800                          (d->DimensionUpperLimits[dim]) ?                          (d->DimensionUpperLimits[dim]) ?
# Line 564  bool DimRegionChooser::is_in_resize_zone Line 809  bool DimRegionChooser::is_in_resize_zone
809                      resize.pos = limit;                      resize.pos = limit;
810                      resize.min = prev_limit;                      resize.min = prev_limit;
811    
812                      int dr = (dimregno >> bitpos) &                      int dr = (maindimregno >> bitpos) &
813                          ((1 << region->pDimensionDefinitions[dim].bits) - 1);                          ((1 << region->pDimensionDefinitions[dim].bits) - 1);
814                      resize.selected = dr == iZone ? resize.left :                      resize.selected = dr == iZone ? resize.left :
815                          dr == iZone + 1 ? resize.right : resize.none;                          dr == iZone + 1 ? resize.right : resize.none;
816    
817                      iZone++;                      iZone++;
818                      gig::DimensionRegion *d = region->pDimensionRegions[c + (iZone << bitpos)];                      gig::DimensionRegion* d = region->pDimensionRegions[c + (iZone << bitpos)];
819    
820                      const int upperLimit =                      const int upperLimit =
821                          (customsplits) ?                          (customsplits) ?
# Line 589  bool DimRegionChooser::is_in_resize_zone Line 834  bool DimRegionChooser::is_in_resize_zone
834      return false;      return false;
835  }  }
836    
837  sigc::signal<void> DimRegionChooser::signal_dimregion_selected()  sigc::signal<void>& DimRegionChooser::signal_dimregion_selected()
838  {  {
839      return dimregion_selected;      return dimregion_selected;
840  }  }
841    
842  sigc::signal<void> DimRegionChooser::signal_region_changed()  sigc::signal<void>& DimRegionChooser::signal_region_changed()
843  {  {
844      return region_changed;      return region_changed;
845  }  }
846    
847  bool DimRegionChooser::on_focus(Gtk::DirectionType direction)  bool DimRegionChooser::on_focus(Gtk::DirectionType direction)
848  {  {
849      // TODO: kolla att region finns osv, dvs att det går att sätta      // TODO: check that region exists etc, that is, that it's possible
850      // fokus.      // to set focus
851      if (direction == Gtk::DIR_TAB_FORWARD ||      if (direction == Gtk::DIR_TAB_FORWARD ||
852          direction == Gtk::DIR_DOWN) {          direction == Gtk::DIR_DOWN) {
853          if (!has_focus()) {          if (!has_focus()) {
# Line 634  bool DimRegionChooser::on_focus(Gtk::Dir Line 879  bool DimRegionChooser::on_focus(Gtk::Dir
879              }              }
880          }          }
881      } else if (!has_focus()) {      } else if (!has_focus()) {
882          // TODO: kolla att focus_line finns!          // TODO: check that focus_line exists
883          grab_focus();          grab_focus();
884          return true;          return true;
885      } else {      } else {
886          // TODO: öka eller minska värde!          // TODO: increase or decrease value
887        }
888        return false;
889    }
890    
891    void DimRegionChooser::split_dimension_zone() {    
892        printf("split_dimension_zone() type=%d, zone=%d\n", maindimtype, maindimcase[maindimtype]);
893        try {
894            region->SplitDimensionZone(maindimtype, maindimcase[maindimtype]);
895        } catch (RIFF::Exception e) {
896            Gtk::MessageDialog msg(e.Message, false, Gtk::MESSAGE_ERROR);
897            msg.run();
898        } catch (...) {
899            Glib::ustring txt = _("An unknown exception occurred!");
900            Gtk::MessageDialog msg(txt, false, Gtk::MESSAGE_ERROR);
901            msg.run();
902        }
903        refresh_all();
904    }
905    
906    void DimRegionChooser::delete_dimension_zone() {
907        printf("delete_dimension_zone() type=%d, zone=%d\n", maindimtype, maindimcase[maindimtype]);
908        try {
909            region->DeleteDimensionZone(maindimtype, maindimcase[maindimtype]);
910        } catch (RIFF::Exception e) {
911            Gtk::MessageDialog msg(e.Message, false, Gtk::MESSAGE_ERROR);
912            msg.run();
913        } catch (...) {
914            Glib::ustring txt = _("An unknown exception occurred!");
915            Gtk::MessageDialog msg(txt, false, Gtk::MESSAGE_ERROR);
916            msg.run();
917        }
918        refresh_all();
919    }
920    
921    bool DimRegionChooser::onKeyPressed(GdkEventKey* key) {
922        //printf("key down\n");
923        if (key->keyval == GDK_KEY_Control_L || key->keyval == GDK_KEY_Control_R)
924            multiSelectKeyDown = true;
925        return false;
926    }
927    
928    bool DimRegionChooser::onKeyReleased(GdkEventKey* key) {
929        //printf("key up\n");
930        if (key->keyval == GDK_KEY_Control_L || key->keyval == GDK_KEY_Control_R)
931            multiSelectKeyDown = false;
932        return false;
933    }
934    
935    void DimRegionChooser::resetSelectedZones() {
936        this->dimzones.clear();
937        if (!region) {
938            queue_draw(); // redraw required parts
939            return;
940      }      }
941        if (maindimregno < 0 || maindimregno >= region->DimensionRegions) {
942            queue_draw(); // redraw required parts
943            return;
944        }
945        if (!region->pDimensionRegions[maindimregno]) {
946            queue_draw(); // redraw required parts
947            return;
948        }
949        gig::DimensionRegion* dimrgn = region->pDimensionRegions[maindimregno];
950    
951        bool isValidZone;
952        this->maindimcase = caseOfDimRegion(dimrgn, &isValidZone);
953        if (!isValidZone) {
954            queue_draw(); // redraw required parts
955            return;
956        }
957    
958        for (std::map<gig::dimension_t,int>::const_iterator it = this->maindimcase.begin();
959             it != this->maindimcase.end(); ++it)
960        {
961            this->dimzones[it->first].insert(it->second);
962        }
963    
964        // redraw required parts
965        queue_draw();
966    }
967    
968    bool DimRegionChooser::select_dimregion(gig::DimensionRegion* dimrgn) {
969        if (!region) return false; //.selection failed
970    
971        for (int dr = 0; dr < region->DimensionRegions && region->pDimensionRegions[dr]; ++dr) {
972            if (region->pDimensionRegions[dr] == dimrgn) {
973                // reset dim region zone selection to the requested specific dim region case
974                maindimregno = dr;
975                resetSelectedZones();
976    
977                // emit signal that dimregion selection has changed, for external entities
978                dimregion_selected();
979    
980                return true; // selection success
981            }
982        }
983    
984        return false; //.selection failed
985    }
986    
987    gig::DimensionRegion* DimRegionChooser::get_main_dimregion() const {
988        if (!region) return NULL;
989        return region->pDimensionRegions[maindimregno];
990  }  }

Legend:
Removed from v.1261  
changed lines
  Added in v.2841

  ViewVC Help
Powered by ViewVC