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

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

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

revision 2169 by persson, Sun Mar 6 07:51:04 2011 UTC revision 3226 by schoenebeck, Fri May 26 22:15:06 2017 UTC
# Line 1  Line 1 
1  /*  /*
2   * Copyright (C) 2006-2011 Andreas Persson   * Copyright (C) 2006-2017 Andreas Persson
3   *   *
4   * This program is free software; you can redistribute it and/or   * This program is free software; you can redistribute it and/or
5   * modify it under the terms of the GNU General Public License as   * modify it under the terms of the GNU General Public License as
# Line 17  Line 17 
17   * 02110-1301 USA.   * 02110-1301 USA.
18   */   */
19    
20    #include "global.h"
21  #include "regionchooser.h"  #include "regionchooser.h"
22    
23  #include <algorithm>  #include <algorithm>
24  #include <sstream>  #include <assert.h>
25    
26  #include <cairomm/context.h>  #include <cairomm/context.h>
27  #include <gdkmm/general.h>  #include <gdkmm/general.h>
28  #include <gdkmm/cursor.h>  #include <gdkmm/cursor.h>
29  #include <gtkmm/stock.h>  #include <gtkmm/stock.h>
30    #include <gdkmm/pixbuf.h>
31  #include <gtkmm/spinbutton.h>  #include <gtkmm/spinbutton.h>
32  #include <gtkmm/dialog.h>  #include <gtkmm/dialog.h>
33    
34  #include "global.h"  #include "Settings.h"
35    #include "gfx/builtinpix.h"
36    
37  #define REGION_BLOCK_HEIGHT             20  #define REGION_BLOCK_HEIGHT             30
38  #define KEYBOARD_HEIGHT                 40  #define KEYBOARD_HEIGHT                 40
39    
40    struct RegionFeatures {
41        int sampleRefs;
42        int loops;
43    
44        RegionFeatures() {
45            sampleRefs = loops = 0;
46        }
47    };
48    
49    static RegionFeatures regionFeatures(gig::Region* rgn) {
50        RegionFeatures f;
51        for (int i = 0; i < rgn->DimensionRegions; ++i) {
52            gig::DimensionRegion* dr = rgn->pDimensionRegions[i];
53            if (dr->pSample) f.sampleRefs++;
54            // the user doesn't care about loop if there is no valid sample reference
55            if (dr->pSample && dr->SampleLoops) f.loops++;
56        }
57        return f;
58    }
59    
60  void SortedRegions::update(gig::Instrument* instrument) {  void SortedRegions::update(gig::Instrument* instrument) {
61      // Usually, the regions in a gig file are ordered after their key      // Usually, the regions in a gig file are ordered after their key
# Line 55  gig::Region* SortedRegions::first() { Line 78  gig::Region* SortedRegions::first() {
78  }  }
79    
80  gig::Region* SortedRegions::next() {  gig::Region* SortedRegions::next() {
81      region_iterator++;      ++region_iterator;
82      return region_iterator == regions.end() ? 0 : *region_iterator;      return region_iterator == regions.end() ? 0 : *region_iterator;
83  }  }
84    
# Line 63  gig::Region* SortedRegions::next() { Line 86  gig::Region* SortedRegions::next() {
86    
87  RegionChooser::RegionChooser() :  RegionChooser::RegionChooser() :
88      activeKeyColor("red"),      activeKeyColor("red"),
89      red("#8070ff"),      blue("#4796ff"),
90      grey1("grey69"),      grey1("grey69"),
91      white("white"),      white("white"),
92      black("black"),      black("black"),
93      m_VirtKeybModeChoice(_("Virtual Keyboard Mode")),      m_VirtKeybModeChoice(_("Virtual Keyboard Mode")),
94      currentActiveKey(-1)      currentActiveKey(-1),
95        modifyallregions(false)
96  {  {
97      set_size_request(500, KEYBOARD_HEIGHT + REGION_BLOCK_HEIGHT);      set_size_request(500, KEYBOARD_HEIGHT + REGION_BLOCK_HEIGHT);
98    
99        loadBuiltInPix();
100    
101        // create gray blue hatched pattern
102        {
103            const int width = grayBlueHatchedPattern->get_width();
104            const int height = grayBlueHatchedPattern->get_height();
105            const int stride = grayBlueHatchedPattern->get_rowstride();
106    
107            // manually convert from RGBA to ARGB
108            this->grayBlueHatchedPatternARGB = grayBlueHatchedPattern->copy();
109            const int pixelSize = stride / width;
110            const int totalPixels = width * height;
111            assert(pixelSize == 4);
112            unsigned char* ptr = this->grayBlueHatchedPatternARGB->get_pixels();
113            for (int iPixel = 0; iPixel < totalPixels; ++iPixel, ptr += pixelSize) {
114                const unsigned char r = ptr[0];
115                const unsigned char g = ptr[1];
116                const unsigned char b = ptr[2];
117                const unsigned char a = ptr[3];
118                ptr[0] = b;
119                ptr[1] = g;
120                ptr[2] = r;
121                ptr[3] = a;
122            }
123    
124            Cairo::RefPtr<Cairo::ImageSurface> imageSurface = Cairo::ImageSurface::create(
125                this->grayBlueHatchedPatternARGB->get_pixels(), Cairo::FORMAT_ARGB32, width, height, stride
126            );
127            this->grayBlueHatchedSurfacePattern = Cairo::SurfacePattern::create(imageSurface);
128            this->grayBlueHatchedSurfacePattern->set_extend(Cairo::EXTEND_REPEAT);
129        }
130    
131      instrument = 0;      instrument = 0;
132      region = 0;      region = 0;
133      resize.active = false;      resize.active = false;
# Line 81  RegionChooser::RegionChooser() : Line 137  RegionChooser::RegionChooser() :
137    
138      // properties of the virtual keyboard      // properties of the virtual keyboard
139      {      {
140          const char* choices[] = { _("normal"), _("chord"), NULL };          const char* choices[] = { _("normal"), _("chord"), 0 };
141          static const virt_keyboard_mode_t values[] = {          static const virt_keyboard_mode_t values[] = {
142              VIRT_KEYBOARD_MODE_NORMAL,              VIRT_KEYBOARD_MODE_NORMAL,
143              VIRT_KEYBOARD_MODE_CHORD              VIRT_KEYBOARD_MODE_CHORD
# Line 101  RegionChooser::RegionChooser() : Line 157  RegionChooser::RegionChooser() :
157      m_VirtKeybPropsBox.pack_start(m_VirtKeybOffVelocityLabel, Gtk::PACK_SHRINK);      m_VirtKeybPropsBox.pack_start(m_VirtKeybOffVelocityLabel, Gtk::PACK_SHRINK);
158      m_VirtKeybPropsBox.set_spacing(10);      m_VirtKeybPropsBox.set_spacing(10);
159      m_VirtKeybPropsBox.show();      m_VirtKeybPropsBox.show();
160        for (int i = 0 ; i < 128 ; i++) key_pressed[i] = false;
161    
162      actionGroup = Gtk::ActionGroup::create();      actionGroup = Gtk::ActionGroup::create();
163      actionGroup->add(Gtk::Action::create("Properties",      actionGroup->add(Gtk::Action::create("Properties",
# Line 154  RegionChooser::RegionChooser() : Line 211  RegionChooser::RegionChooser() :
211      keyboard_key_released_signal.connect(      keyboard_key_released_signal.connect(
212          sigc::mem_fun(*this, &RegionChooser::on_note_off_event)          sigc::mem_fun(*this, &RegionChooser::on_note_off_event)
213      );      );
214        set_tooltip_text(_("Right click here for adding new region. Use mouse pointer for moving (dragging) or resizing existing regions (by pointing at region's boundary). Right click on an existing region for more actions."));
215  }  }
216    
217  RegionChooser::~RegionChooser()  RegionChooser::~RegionChooser()
218  {  {
219  }  }
220    
221  template<class T> inline std::string ToString(T o) {  void RegionChooser::setModifyAllRegions(bool b) {
222      std::stringstream ss;      modifyallregions = b;
223      ss << o;      // redraw required parts
224      return ss.str();      queue_draw();
225    }
226    
227    void RegionChooser::invalidate_key(int key) {
228        const int h = KEYBOARD_HEIGHT;
229        const int w = get_width() - 1;
230        int x1 = key_to_x(key - 0.5, w);
231        int x2 = key_to_x(key + 1.5, w);
232    
233        Gdk::Rectangle rect(x1 + 1, h1 + 1, x2 - x1 - 1, h - 2);
234        get_window()->invalidate_rect(rect, false);
235  }  }
236    
237  void RegionChooser::on_note_on_event(int key, int velocity) {  void RegionChooser::on_note_on_event(int key, int velocity) {
238      draw_key(key, activeKeyColor);      key_pressed[key] = true;
239        invalidate_key(key);
240      m_VirtKeybVelocityLabel.set_text(ToString(velocity));      m_VirtKeybVelocityLabel.set_text(ToString(velocity));
241  }  }
242    
243  void RegionChooser::on_note_off_event(int key, int velocity) {  void RegionChooser::on_note_off_event(int key, int velocity) {
244      if (is_black_key(key)) {      key_pressed[key] = false;
245          draw_key(key, black);      invalidate_key(key);
     } else {  
         draw_key(key, key >= 21 && key <= 108 ? white : grey1);  
     }  
246      m_VirtKeybOffVelocityLabel.set_text(ToString(velocity));      m_VirtKeybOffVelocityLabel.set_text(ToString(velocity));
247  }  }
248    
249    
250  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
251  bool RegionChooser::on_expose_event(GdkEventExpose* e)  bool RegionChooser::on_expose_event(GdkEventExpose* e) {
252  {      double clipx1 = e->area.x;
253      return on_draw(get_window()->create_cairo_context());      double clipx2 = e->area.x + e->area.width;
254        double clipy1 = e->area.y;
255        double clipy2 = e->area.y + e->area.height;
256    
257        const Cairo::RefPtr<Cairo::Context>& cr =
258            get_window()->create_cairo_context();
259    #if 0
260  }  }
261  #endif  #endif
262    #else
263  bool RegionChooser::on_draw(const Cairo::RefPtr<Cairo::Context>& cr)  bool RegionChooser::on_draw(const Cairo::RefPtr<Cairo::Context>& cr) {
264  {      double clipx1, clipx2, clipy1, clipy2;
265      const int h = KEYBOARD_HEIGHT;      cr->get_clip_extents(clipx1, clipy1, clipx2, clipy2);
266      const int w = get_width() - 1;  #endif
     const int bh = int(h * 0.55);  
267    
268      cr->save();      cr->save();
269      cr->set_line_width(1);      cr->set_line_width(1);
# Line 204  bool RegionChooser::on_draw(const Cairo: Line 276  bool RegionChooser::on_draw(const Cairo:
276      Gdk::Cairo::set_source_rgba(cr, bg);      Gdk::Cairo::set_source_rgba(cr, bg);
277      cr->paint();      cr->paint();
278    
279        if (clipy2 > h1) {
280            draw_keyboard(cr, clipx1, clipx2);
281        }
282    
283        if (clipy1 < h1 && instrument) {
284            draw_regions(cr, clipx1, clipx2);
285        }
286    
287        cr->restore();
288    
289        return true;
290    }
291    
292    void RegionChooser::draw_keyboard(const Cairo::RefPtr<Cairo::Context>& cr,
293                                      int clip_low, int clip_high) {
294        const int h = KEYBOARD_HEIGHT;
295        const int w = get_width() - 1;
296        const int bh = int(h * 0.55);
297    
298      Gdk::Cairo::set_source_rgba(cr, black);      Gdk::Cairo::set_source_rgba(cr, black);
299      cr->rectangle(0.5, h1 + 0.5, w, h - 1);      cr->rectangle(0.5, h1 + 0.5, w, h - 1);
300      cr->stroke();      cr->stroke();
301    
302      int x1 = int(w * 20.5 / 128.0 + 0.5);      int x1 = key_to_x(20.5, w);
     int x2 = int(w * 109.5 / 128.0 + 0.5);  
   
303      Gdk::Cairo::set_source_rgba(cr, grey1);      Gdk::Cairo::set_source_rgba(cr, grey1);
304      cr->rectangle(1, h1 + 1, x1 - 1, h - 2);      cr->rectangle(1, h1 + 1, x1 - 1, h - 2);
305      cr->fill();      cr->fill();
306    
307        int x2 = key_to_x(109.5, w);
308      Gdk::Cairo::set_source_rgba(cr, white);      Gdk::Cairo::set_source_rgba(cr, white);
309      cr->rectangle(x1 + 1, h1 + 1, x2 - x1 - 1, h - 2);      cr->rectangle(x1 + 1, h1 + 1, x2 - x1 - 1, h - 2);
310      cr->fill();      cr->fill();
# Line 224  bool RegionChooser::on_draw(const Cairo: Line 314  bool RegionChooser::on_draw(const Cairo:
314      cr->fill();      cr->fill();
315    
316      Gdk::Cairo::set_source_rgba(cr, black);      Gdk::Cairo::set_source_rgba(cr, black);
317      for (int i = 0 ; i < 128 ; i++) {  
318        int clipkey1 = std::max(0, x_to_key_right(clip_low - 1, w));
319        int clipkey2 = std::min(x_to_key_right(clip_high - 1, w) + 1, 128);
320    
321        for (int i = clipkey1 ; i < clipkey2 ; i++) {
322          int note = (i + 3) % 12;          int note = (i + 3) % 12;
323          int x = int(w * i / 128.0 + 0.5);          int x = key_to_x(i, w);
324    
325          if (note == 1 || note == 4 || note == 6 ||          if (note == 1 || note == 4 || note == 6 ||
326              note == 9 || note == 11) {              note == 9 || note == 11) {
327              int x2 = int(w * (i + 0.5) / 128.0 + 0.5);              // black key: short line in the middle, with a rectangle
328                // on top
329                int x2 = key_to_x(i + 0.5, w);
330              cr->move_to(x2 + 0.5, h1 + bh + 0.5);              cr->move_to(x2 + 0.5, h1 + bh + 0.5);
331              cr->line_to(x2 + 0.5, h1 + h - 1);              cr->line_to(x2 + 0.5, h1 + h - 1);
332              cr->stroke();              cr->stroke();
333    
334              int x3 = int(w * (i + 1) / 128.0 + 0.5);              int x3 = key_to_x(i + 1, w);
335              cr->rectangle(x, h1 + 1, x3 - x + 1, bh);              cr->rectangle(x, h1 + 1, x3 - x + 1, bh);
336              cr->fill();              cr->fill();
337          } else if (note == 3 || note == 8) {          } else if (note == 3 || note == 8) {
338                // C or F: long line to the left
339              cr->move_to(x + 0.5, h1 + 1);              cr->move_to(x + 0.5, h1 + 1);
340              cr->line_to(x + 0.5, h1 + h - 1);              cr->line_to(x + 0.5, h1 + h - 1);
341              cr->stroke();              cr->stroke();
   
             if (note == 3) draw_digit(i);  
342          }          }
343    
344            if (key_pressed[i]) draw_key(cr, i);
345    
346            if (note == 3) draw_digit(cr, i);
347      }      }
348    }
349    
     if (instrument) {  
         int i = 0;  
         gig::Region* next_region;  
         int x3 = -1;  
         for (gig::Region* r = regions.first() ; r ; r = next_region) {  
350    
351              if (x3 < 0) x3 = int(w * (r->KeyRange.low) / 128.0 + 0.5);  void RegionChooser::draw_regions(const Cairo::RefPtr<Cairo::Context>& cr,
352              next_region = regions.next();                                   int clip_low, int clip_high) {
353              if (!next_region ||      const int w = get_width() - 1;
354                  r->KeyRange.high + 1 != next_region->KeyRange.low) {  
355                  int x2 = int(w * (r->KeyRange.high + 1) / 128.0 + 0.5);      Gdk::Cairo::set_source_rgba(cr, black);
356        gig::Region* next_region;
357        int x3 = -1;
358        for (gig::Region* r = regions.first() ; r ; r = next_region) {
359            next_region = regions.next();
360    
361            if (x3 < 0) {
362                x3 = key_to_x(r->KeyRange.low, w);
363                if (x3 >= clip_high) break;
364            }
365            if (!next_region ||
366                r->KeyRange.high + 1 != next_region->KeyRange.low ||
367                r == region || next_region == region) {
368    
369                int x2 = key_to_x(r->KeyRange.high + 1, w);
370                if (x2 >= clip_low) {
371                  cr->move_to(x3, 0.5);                  cr->move_to(x3, 0.5);
372                  cr->line_to(x2 + 0.5, 0.5);                  cr->line_to(x2 + 0.5, 0.5);
373                  cr->line_to(x2 + 0.5, h1 - 0.5);                  cr->line_to(x2 + 0.5, h1 - 0.5);
374                  cr->line_to(x3, h1 - 0.5);                  cr->line_to(x3, h1 - 0.5);
375                  cr->stroke();                  cr->stroke();
376    
377                  Gdk::Cairo::set_source_rgba(cr, white);                  if (region == r)
378                        Gdk::Cairo::set_source_rgba(cr, blue);
379                    else if (modifyallregions)
380                        cr->set_source(grayBlueHatchedSurfacePattern);
381                    else
382                        Gdk::Cairo::set_source_rgba(cr, white);
383    
384                  cr->rectangle(x3 + 1, 1, x2 - x3 - 1, h1 - 2);                  cr->rectangle(x3 + 1, 1, x2 - x3 - 1, h1 - 2);
385                  cr->fill();                  cr->fill();
386                  Gdk::Cairo::set_source_rgba(cr, black);                  Gdk::Cairo::set_source_rgba(cr, black);
   
                 x3 = -1;  
387              }              }
388              i++;              x3 = -1;
         }  
   
         for (gig::Region* r = regions.first() ; r ; r = regions.next()) {  
             int x = int(w * (r->KeyRange.low) / 128.0 + 0.5);  
             cr->move_to(x + 0.5, 1);  
             cr->line_to(x + 0.5, h1 - 1);  
             cr->stroke();  
389          }          }
390        }
391    
392          if (region) {      for (gig::Region* r = regions.first() ; r ; r = regions.next()) {
393              int x1 = int(w * (region->KeyRange.low) / 128.0 + 0.5);          int x = key_to_x(r->KeyRange.low, w);
394              int x2 = int(w * (region->KeyRange.high + 1) / 128.0 + 0.5);          int x2 = key_to_x(r->KeyRange.high + 1, w);
395              Gdk::Cairo::set_source_rgba(cr, red);  
396              cr->rectangle(x1 + 1, 1, x2 - x1 - 1, h1 - 2);          RegionFeatures features = regionFeatures(r);
397              cr->fill();  
398            const bool bShowLoopSymbol = features.loops > 0;
399            const bool bShowSampleRefSymbol = features.sampleRefs < r->DimensionRegions;
400            if (bShowLoopSymbol || bShowSampleRefSymbol) {
401                const int margin = 2;
402                const int wRgn = x2 - x;
403                //printf("x=%d x2=%d wRgn=%d\n", x, x2, wRgn);
404    
405                cr->save();
406                cr->set_line_width(1);
407                cr->rectangle(x, 1, wRgn, h1 - 1);
408                cr->clip();
409                if (bShowSampleRefSymbol) {
410                    const int wPic = 8;
411                    const int hPic = 8;
412                    Gdk::Cairo::set_source_pixbuf(
413                        cr, (features.sampleRefs) ? yellowDot : redDot,
414                        x + (wRgn-wPic)/2.f,
415                        (bShowLoopSymbol) ? margin : (h1-hPic)/2.f
416                    );
417                    cr->paint();
418                }
419                if (bShowLoopSymbol) {
420                    const int wPic = 12;
421                    const int hPic = 14;
422                    Gdk::Cairo::set_source_pixbuf(
423                        cr, (features.loops == r->DimensionRegions) ? blackLoop : grayLoop,
424                        x + (wRgn-wPic)/2.f,
425                        (bShowSampleRefSymbol) ? h1 - hPic - margin : (h1-hPic)/2.f
426                    );
427                    cr->paint();
428                }
429                cr->restore();
430          }          }
431      }      }
432    
433      cr->restore();      for (gig::Region* r = regions.first() ; r ; r = regions.next()) {
434            int x = key_to_x(r->KeyRange.low, w);
435    
436      return true;          if (x < clip_low) continue;
437  }          if (x >= clip_high) break;
438    
439            cr->move_to(x + 0.5, 1);
440            cr->line_to(x + 0.5, h1 - 1);
441            cr->stroke();
442        }
443    
444        // if there is no region yet, show the user some hint text that he may
445        // right click on this area to create a new region
446        if (!regions.first()) {
447            Glib::RefPtr<Pango::Context> context = get_pango_context();
448            Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
449            layout->set_alignment(Pango::ALIGN_CENTER);
450            layout->set_text(Glib::ustring("*** ") + _("Right click here to create a region.") + " ***");
451            layout->set_width(get_width() * Pango::SCALE);
452            //layout->set_height(get_height() * Pango::SCALE);
453            layout->set_spacing(10);
454            Gdk::Cairo::set_source_rgba(cr, blue);
455            // get the text dimensions
456            int text_width, text_height;
457            layout->get_pixel_size(text_width, text_height);
458            cr->move_to(0, (REGION_BLOCK_HEIGHT - text_height) / 2);
459    #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
460            pango_cairo_show_layout(cr->cobj(), layout->gobj());
461    #else
462            layout->show_in_cairo_context(cr);
463    #endif
464        }
465    }
466    
467  bool RegionChooser::is_black_key(int key) {  bool RegionChooser::is_black_key(int key) {
468      const int note = (key + 3) % 12;      const int note = (key + 3) % 12;
469      return note == 1 || note == 4 || note == 6 || note == 9 || note == 11;      return note == 1 || note == 4 || note == 6 || note == 9 || note == 11;
470  }  }
471    
472  void RegionChooser::draw_digit(int key) {  void RegionChooser::draw_digit(const Cairo::RefPtr<Cairo::Context>& cr,
473                                   int key) {
474      const int h = KEYBOARD_HEIGHT;      const int h = KEYBOARD_HEIGHT;
475      const int w = get_width() - 1;      const int w = get_width() - 1;
476      Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(get_pango_context());      Glib::RefPtr<Pango::Layout> layout =
477            Pango::Layout::create(get_pango_context());
478      char buf[30];      char buf[30];
479      sprintf(buf, "<span size=\"8000\">%d</span>", key / 12 - 1);      sprintf(buf, "<span size=\"8000\">%d</span>", key / 12 - 1);
480      layout->set_markup(buf);      layout->set_markup(buf);
# Line 312  void RegionChooser::draw_digit(int key) Line 482  void RegionChooser::draw_digit(int key)
482      double text_w = double(rectangle.get_width()) / Pango::SCALE;      double text_w = double(rectangle.get_width()) / Pango::SCALE;
483      double text_h = double(rectangle.get_height()) / Pango::SCALE;      double text_h = double(rectangle.get_height()) / Pango::SCALE;
484      double x = w * (key + 0.75) / 128.0;      double x = w * (key + 0.75) / 128.0;
     Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context();  
485      Gdk::Cairo::set_source_rgba(cr, black);      Gdk::Cairo::set_source_rgba(cr, black);
486      cr->move_to(int(x - text_w / 2 + 1), int(h1 + h - text_h + 0.5));      cr->move_to(int(x - text_w / 2 + 1), int(h1 + h - text_h + 0.5));
487  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 16) || GTKMM_MAJOR_VERSION < 2
# Line 322  void RegionChooser::draw_digit(int key) Line 491  void RegionChooser::draw_digit(int key)
491  #endif  #endif
492  }  }
493    
494  void RegionChooser::draw_key(int key, const Gdk::RGBA& color)  void RegionChooser::draw_key(const Cairo::RefPtr<Cairo::Context>& cr,
495  {                               int key) {
496      const int h = KEYBOARD_HEIGHT;      const int h = KEYBOARD_HEIGHT;
497      const int w = get_width() - 1;      const int w = get_width() - 1;
498      const int bh = int(h * 0.55);      const int bh = int(h * 0.55);
499    
500      Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context();      Gdk::Cairo::set_source_rgba(cr, activeKeyColor);
     Gdk::Cairo::set_source_rgba(cr, color);  
501    
502      int note = (key + 3) % 12;      int note = (key + 3) % 12;
503      int x = int(w * key / 128.0 + 0.5) + 1;      int x = key_to_x(key, w) + 1;
504      int x2 = int(w * (key + 1.5) / 128.0 + 0.5);      int x2 = key_to_x(key + 1.5, w);
505      int x3 = int(w * (key + 1) / 128.0 + 0.5);      int x3 = key_to_x(key + 1, w);
506      int x4 = int(w * (key - 0.5) / 128.0 + 0.5);      int x4 = key_to_x(key - 0.5, w);
507      int w1 = x3 - x;      int w1 = x3 - x;
508      switch (note) {      switch (note) {
509      case 0: case 5: case 10:      case 0: case 5: case 10:
# Line 355  void RegionChooser::draw_key(int key, co Line 523  void RegionChooser::draw_key(int key, co
523          cr->fill();          cr->fill();
524          cr->rectangle(x, h1 + bh + 1, x2 - x, h - bh - 2);          cr->rectangle(x, h1 + bh + 1, x2 - x, h - bh - 2);
525          cr->fill();          cr->fill();
         if (note == 3) draw_digit(key);  
526          break;          break;
527      default:      default:
528          cr->rectangle(x, h1 + 1, w1, bh - 1);          cr->rectangle(x, h1 + 1, w1, bh - 1);
529          cr->fill();          cr->fill();
530          break;          break;
531      }      }
532        Gdk::Cairo::set_source_rgba(cr, black);
533  }  }
534    
535  void RegionChooser::set_instrument(gig::Instrument* instrument)  void RegionChooser::set_instrument(gig::Instrument* instrument)
# Line 376  void RegionChooser::set_instrument(gig:: Line 544  void RegionChooser::set_instrument(gig::
544    
545  bool RegionChooser::on_button_release_event(GdkEventButton* event)  bool RegionChooser::on_button_release_event(GdkEventButton* event)
546  {  {
547      const int k = int(event->x / (get_width() - 1) * 128.0);      const int k = x_to_key(event->x, get_width() - 1);
548    
549      // handle-note off on virtual keyboard      // handle-note off on virtual keyboard
550      if (event->type == GDK_BUTTON_RELEASE) {      if (event->type == GDK_BUTTON_RELEASE) {
# Line 406  bool RegionChooser::on_button_release_ev Line 574  bool RegionChooser::on_button_release_ev
574  #endif  #endif
575          resize.active = false;          resize.active = false;
576    
         if (resize.mode == resize.moving_high_limit) {  
             if (resize.region->KeyRange.high != resize.pos - 1) {  
                 instrument_struct_to_be_changed_signal.emit(instrument);  
                 resize.region->SetKeyRange(  
                     resize.region->KeyRange.low, // low  
                     resize.pos - 1 // high  
                 );  
                 regions.update(instrument);  
                 instrument_changed.emit();  
                 instrument_struct_changed_signal.emit(instrument);  
             }  
         } else if (resize.mode == resize.moving_low_limit) {  
             if (resize.region->KeyRange.low != resize.pos) {  
                 instrument_struct_to_be_changed_signal.emit(instrument);  
                 resize.region->SetKeyRange(  
                     resize.pos, // low  
                     resize.region->KeyRange.high // high  
                 );  
                 regions.update(instrument);  
                 instrument_changed.emit();  
                 instrument_struct_changed_signal.emit(instrument);  
             }  
         }  
   
577          if (!is_in_resize_zone(event->x, event->y) && cursor_is_resize) {          if (!is_in_resize_zone(event->x, event->y) && cursor_is_resize) {
578              get_window()->set_cursor();              get_window()->set_cursor();
579              cursor_is_resize = false;              cursor_is_resize = false;
# Line 442  bool RegionChooser::on_button_release_ev Line 586  bool RegionChooser::on_button_release_ev
586  #endif  #endif
587          move.active = false;          move.active = false;
588    
         if (move.pos) {  
             instrument_struct_to_be_changed_signal.emit(instrument);  
             region->SetKeyRange(  
                 region->KeyRange.low  + move.pos,  
                 region->KeyRange.high + move.pos  
             );  
             regions.update(instrument);  
             instrument_changed.emit();  
             instrument_struct_changed_signal.emit(instrument);  
         }  
   
589          if (is_in_resize_zone(event->x, event->y)) {          if (is_in_resize_zone(event->x, event->y)) {
590  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2
591              get_window()->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));              get_window()->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));
# Line 465  bool RegionChooser::on_button_release_ev Line 598  bool RegionChooser::on_button_release_ev
598      return true;      return true;
599  }  }
600    
601    void RegionChooser::update_after_resize()
602    {
603        if (resize.mode == resize.moving_high_limit) {
604            if (resize.region->KeyRange.high != resize.pos - 1) {
605                instrument_struct_to_be_changed_signal.emit(instrument);
606                resize.region->SetKeyRange(resize.region->KeyRange.low,
607                                           resize.pos - 1);
608                regions.update(instrument);
609                instrument_changed.emit();
610                instrument_struct_changed_signal.emit(instrument);
611            }
612        } else if (resize.mode == resize.moving_low_limit) {
613            if (resize.region->KeyRange.low != resize.pos) {
614                instrument_struct_to_be_changed_signal.emit(instrument);
615                resize.region->SetKeyRange(resize.pos,
616                                           resize.region->KeyRange.high);
617                regions.update(instrument);
618                instrument_changed.emit();
619                instrument_struct_changed_signal.emit(instrument);
620            }
621        }
622    }
623    
624    void RegionChooser::update_after_move(int pos)
625    {
626        instrument_struct_to_be_changed_signal.emit(instrument);
627        const int range = region->KeyRange.high - region->KeyRange.low;
628        const int diff  = pos - int(region->KeyRange.low);
629        region->SetKeyRange(pos, pos + range);
630        if (Settings::singleton()->moveRootNoteWithRegionMoved) {
631            for (int i = 0; i < 256; ++i) {
632                gig::DimensionRegion* dimrgn = region->pDimensionRegions[i];
633                if (!dimrgn || !dimrgn->pSample || !dimrgn->PitchTrack) continue;
634                dimrgn->UnityNote += diff;
635            }
636        }
637        regions.update(instrument);
638        instrument_changed.emit();
639        instrument_struct_changed_signal.emit(instrument);
640    }
641    
642  bool RegionChooser::on_button_press_event(GdkEventButton* event)  bool RegionChooser::on_button_press_event(GdkEventButton* event)
643  {  {
644      if (!instrument) return true;      if (!instrument) return true;
645    
646      const int k = int(event->x / (get_width() - 1) * 128.0);      const int w = get_width() - 1;
647        const int k = x_to_key(event->x, w);
648    
649      if (event->type == GDK_BUTTON_PRESS) {      if (event->type == GDK_BUTTON_PRESS) {
650          if (event->y >= REGION_BLOCK_HEIGHT) {          if (event->y >= REGION_BLOCK_HEIGHT) {
# Line 480  bool RegionChooser::on_button_press_even Line 655  bool RegionChooser::on_button_press_even
655          }          }
656      }      }
657    
658        // left mouse button double click
659        if (event->type == GDK_2BUTTON_PRESS && event->button == 1) {
660            if (event->y < REGION_BLOCK_HEIGHT) {
661                // show dimension manager dialog for this region
662                manage_dimensions();
663            }
664        }
665    
666      if (event->y >= REGION_BLOCK_HEIGHT) return true;      if (event->y >= REGION_BLOCK_HEIGHT) return true;
667      if (event->type == GDK_BUTTON_PRESS && event->button == 3) {      if (event->type == GDK_BUTTON_PRESS && event->button == 3) {
668          gig::Region* r = get_region(k);          gig::Region* r = get_region(k);
# Line 539  bool RegionChooser::on_button_press_even Line 722  bool RegionChooser::on_button_press_even
722                                                        event->time);                                                        event->time);
723  #endif  #endif
724                  move.active = true;                  move.active = true;
725                  move.from_x = event->x;                  move.offset = event->x - key_to_x(region->KeyRange.low, w);
                 move.pos = 0;  
726              }              }
727          }          }
728      }      }
# Line 549  bool RegionChooser::on_button_press_even Line 731  bool RegionChooser::on_button_press_even
731    
732  gig::Region* RegionChooser::get_region(int key)  gig::Region* RegionChooser::get_region(int key)
733  {  {
734      gig::Region* prev_region = 0;      for (gig::Region* r = regions.first() ; r ; r = regions.next()) {
     gig::Region* next_region;  
     for (gig::Region* r = regions.first() ; r ; r = next_region) {  
         next_region = regions.next();  
   
735          if (key < r->KeyRange.low) return 0;          if (key < r->KeyRange.low) return 0;
736          if (key <= r->KeyRange.high) {          if (key <= r->KeyRange.high) return r;
             move.touch_left = prev_region && prev_region->KeyRange.high + 1 == r->KeyRange.low;  
             move.touch_right = next_region && r->KeyRange.high + 1 == next_region->KeyRange.low;  
             return r;  
         }  
         prev_region = r;  
737      }      }
738      return 0;      return 0;
739  }  }
740    
741    void RegionChooser::set_region(gig::Region* region) {
742        this->region = region;
743        queue_draw();
744        region_selected();
745        dimensionManager.set_region(region);
746    }
747    
748    void RegionChooser::select_next_region() {
749        if (!instrument) return;
750        if (!region) {
751            for (int i = 0; i < 128; ++i) {
752                ::gig::Region* rgn = instrument->GetRegion(i);
753                if (rgn) {
754                    set_region(rgn);
755                    return;
756                }
757            }
758        } else {
759            bool currentFound = false;
760            for (int i = 0; i < 128; ++i) {
761                ::gig::Region* rgn = instrument->GetRegion(i);
762                if (!rgn) continue;
763                if (currentFound) {
764                    if (rgn != region) {
765                        set_region(rgn);
766                        return;
767                    }
768                } else {
769                    if (rgn == region) currentFound = true;
770                }
771            }
772        }
773    }
774    
775    void RegionChooser::select_prev_region() {
776        if (!instrument) return;
777        if (!region) {
778            for (int i = 0; i < 128; ++i) {
779                ::gig::Region* rgn = instrument->GetRegion(i);
780                if (rgn) {
781                    set_region(rgn);
782                    return;
783                }
784            }
785        } else {
786            bool currentFound = false;
787            for (int i = 127; i >= 0; --i) {
788                ::gig::Region* rgn = instrument->GetRegion(i);
789                if (!rgn) continue;
790                if (currentFound) {
791                    if (rgn != region) {
792                        set_region(rgn);
793                        return;
794                    }
795                } else {
796                    if (rgn == region) currentFound = true;
797                }
798            }
799        }
800    }
801    
802  void RegionChooser::motion_resize_region(int x, int y)  void RegionChooser::motion_resize_region(int x, int y)
803  {  {
804      const int w = get_width() - 1;      const int w = get_width() - 1;
# Line 575  void RegionChooser::motion_resize_region Line 809  void RegionChooser::motion_resize_region
809      else if (k > resize.max) k = resize.max;      else if (k > resize.max) k = resize.max;
810    
811      if (k != resize.pos) {      if (k != resize.pos) {
         Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context();  
         cr->set_line_width(1);  
   
812          if (resize.mode == resize.undecided) {          if (resize.mode == resize.undecided) {
813              if (k < resize.pos) {              if (k < resize.pos) {
814                  // edit high limit of prev_region                  // edit high limit of prev_region
# Line 590  void RegionChooser::motion_resize_region Line 821  void RegionChooser::motion_resize_region
821                  resize.mode = resize.moving_low_limit;                  resize.mode = resize.moving_low_limit;
822              }              }
823          }          }
824          const Gdk::RGBA white = region == resize.region ? red : this->white;          resize.pos = k;
 #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2  
         const Gdk::Color bg = get_style()->get_bg(Gtk::STATE_NORMAL);  
 #else  
         const Gdk::RGBA bg = get_style_context()->get_background_color();  
 #endif  
   
         int prevx = int(w * resize.pos / 128.0 + 0.5);  
         x = int(w * k / 128.0 + 0.5);  
825    
826            int x1, x2;
827          if (resize.mode == resize.moving_high_limit) {          if (resize.mode == resize.moving_high_limit) {
828              if (k > resize.pos) {              if (resize.region->KeyRange.high < resize.pos - 1) {
829                  Gdk::Cairo::set_source_rgba(cr, white);                  x1 = resize.region->KeyRange.high;
830                  cr->rectangle(prevx, 1, x - prevx, h1 - 2);                  x2 = resize.pos - 1;
                 cr->fill();  
   
                 Gdk::Cairo::set_source_rgba(cr, black);  
                 cr->move_to(prevx, 0.5);  
                 cr->line_to(x + 1, 0.5);  
                 cr->move_to(prevx, h1 - 0.5);  
                 cr->line_to(x + 1, h1 - 0.5);  
                 cr->stroke();  
831              } else {              } else {
832                  int xx = (resize.pos == resize.max &&                  x1 = resize.pos - 1;
833                            resize.max != 128) ? 1 : 0;                  x2 = resize.region->KeyRange.high;
                 Gdk::Cairo::set_source_rgba(cr, bg);  
                 cr->rectangle(x + 1, 0, prevx - x - xx, h1);  
                 cr->fill();  
834              }              }
835          } else {          } else {
836              if (k < resize.pos) {              if (resize.region->KeyRange.low < resize.pos) {
837                  Gdk::Cairo::set_source_rgba(cr, white);                  x1 = resize.region->KeyRange.low;
838                  cr->rectangle(x + 1, 1, prevx - x, h1 - 2);                  x2 = resize.pos;
                 cr->fill();  
   
                 Gdk::Cairo::set_source_rgba(cr, black);  
                 cr->move_to(x, 0.5);  
                 cr->line_to(prevx, 0.5);  
                 cr->move_to(x, h1 - 0.5);  
                 cr->line_to(prevx, h1 - 0.5);  
                 cr->stroke();  
839              } else {              } else {
840                  int xx = (resize.pos == resize.min &&                  x1 = resize.pos;
841                            resize.min != 0) ? 1 : 0;                  x2 = resize.region->KeyRange.low;
                 Gdk::Cairo::set_source_rgba(cr, bg);  
                 cr->rectangle(prevx + xx, 0, x - prevx - xx, h1);  
                 cr->fill();  
842              }              }
843          }          }
844          Gdk::Cairo::set_source_rgba(cr, black);          x1 = key_to_x(x1, w);
845          cr->move_to(x + 0.5, 1);          x2 = key_to_x(x2 + 1, w) + 1;
846          cr->line_to(x + 0.5, h1 - 1);          Gdk::Rectangle rect(x1, 0, x2 - x1, h1);
847          cr->stroke();  
848          resize.pos = k;          update_after_resize();
849    
850            //get_window()->invalidate_rect(rect, false);
851            get_window()->invalidate(false); // repaint entire region, otherwise it would create visual artifacts
852      }      }
853  }  }
854    
855  void RegionChooser::motion_move_region(int x, int y)  void RegionChooser::motion_move_region(int x, int y)
856  {  {
857      const int w = get_width() - 1;      const int w = get_width() - 1;
     Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context();  
     cr->set_line_width(1);  
858    
859      int k = int(double(x - move.from_x) / w * 128.0 + 0.5);      int l = int(double(x - move.offset) / w * 128.0 + 0.5);
860      if (k == move.pos) return;  
861      int new_k;      if (l == region->KeyRange.low) return;
862      bool new_touch_left;      int new_l;
863      bool new_touch_right;      int regionsize = region->KeyRange.high - region->KeyRange.low;
864      int a = 0;      int a = 0;
865      if (k > move.pos) {      if (l > region->KeyRange.low) {
866          for (gig::Region* r = regions.first() ; ; r = regions.next()) {          for (gig::Region* r = regions.first() ; ; r = regions.next()) {
867              if (r != region) {              if (r != region) {
868                  int b = r ? r->KeyRange.low : 128;                  int b = r ? r->KeyRange.low : 128;
869    
870                  // gap: from a to b (not inclusive b)                  // gap: from a to b (not inclusive b)
871    
872                  if (region->KeyRange.high + move.pos >= b) {                  if (region->KeyRange.high >= b) {
873                      // not found the current gap yet, just continue                      // not found the current gap yet, just continue
874                  } else {                  } else {
875    
876                      if (a > region->KeyRange.low + k) {                      if (a > l) {
877                          // this gap is too far to the right, break                          // this gap is too far to the right, break
878                          break;                          break;
879                      }                      }
880    
881                      int newhigh = std::min(region->KeyRange.high + k, b - 1);                      int newhigh = std::min(l + regionsize, b - 1);
882                      int newlo = newhigh - (region->KeyRange.high - region->KeyRange.low);                      int newlo = newhigh - regionsize;
883    
884                      if (newlo >= a) {                      if (newlo >= a) {
885                          // yes it fits - it's a candidate                          // yes it fits - it's a candidate
886                          new_k = newlo - region->KeyRange.low;                          new_l = newlo;
                         new_touch_left = a > 0 && a == newlo;  
                         new_touch_right = b < 128 && newhigh + 1 == b;  
887                      }                      }
888                  }                  }
889                  if (!r) break;                  if (!r) break;
# Line 696  void RegionChooser::motion_move_region(i Line 897  void RegionChooser::motion_move_region(i
897    
898                  // gap from a to b (not inclusive b)                  // gap from a to b (not inclusive b)
899    
900                  if (region->KeyRange.high + k >= b) {                  if (l + regionsize >= b) {
901                      // not found the current gap yet, just continue                      // not found the current gap yet, just continue
902                  } else {                  } else {
903    
904                      if (a > region->KeyRange.low + move.pos) {                      if (a > region->KeyRange.low) {
905                          // this gap is too far to the right, break                          // this gap is too far to the right, break
906                          break;                          break;
907                      }                      }
908    
909                      int newlo = std::max(region->KeyRange.low + k, a);                      int newlo = std::max(l, a);
910                      int newhigh = newlo + (region->KeyRange.high - region->KeyRange.low);                      int newhigh = newlo + regionsize;
911    
912                      if (newhigh < b) {                      if (newhigh < b) {
913                          // yes it fits - break as the first one is the best                          // yes it fits - break as the first one is the best
914                          new_k = newlo - region->KeyRange.low;                          new_l = newlo;
                         new_touch_left = a > 0 && a == newlo;  
                         new_touch_right = b < 128 && newhigh + 1 == b;  
915                          break;                          break;
916                      }                      }
917                  }                  }
# Line 721  void RegionChooser::motion_move_region(i Line 920  void RegionChooser::motion_move_region(i
920              }              }
921          }          }
922      }      }
923      k = new_k;      if (new_l == region->KeyRange.low) return;
     if (k == move.pos) return;  
924    
925  #if (GTKMM_MAJOR_VERSION == 2 && GTKMM_MINOR_VERSION < 90) || GTKMM_MAJOR_VERSION < 2      int x1 = key_to_x(std::min(int(region->KeyRange.low), new_l), w);
926      const Gdk::Color bg = get_style()->get_bg(Gtk::STATE_NORMAL);      int x2 = key_to_x(std::max(int(region->KeyRange.high),
927  #else                                 new_l + regionsize) + 1, w) + 1;
     const Gdk::RGBA bg = get_style_context()->get_background_color();  
 #endif  
   
     int prevx = int(w * (move.pos + region->KeyRange.low) / 128.0 + 0.5);  
     x = int(w * (k + region->KeyRange.low) / 128.0 + 0.5);  
     int prevx2 = int(w * (move.pos + region->KeyRange.high + 1) / 128.0 + 0.5);  
     int x2 = int(w * (k + region->KeyRange.high + 1) / 128.0 + 0.5);  
   
     if (!new_touch_left) {  
         Gdk::Cairo::set_source_rgba(cr, black);  
         cr->move_to(x + 0.5, 1);  
         cr->line_to(x + 0.5, h1 - 1);  
         cr->stroke();  
     }  
     if (!new_touch_right) {  
         Gdk::Cairo::set_source_rgba(cr, black);  
         cr->move_to(x2 + 0.5, 1);  
         cr->line_to(x2 + 0.5, h1 - 1);  
         cr->stroke();  
     }  
928    
929      if (k > move.pos) {      Gdk::Rectangle rect(x1, 0, x2 - x1, h1);
930          Gdk::Cairo::set_source_rgba(cr, bg);      update_after_move(new_l);
         cr->rectangle(prevx + (move.touch_left ? 1 : 0), 0,  
                       std::min(x, prevx2 + 1 - (move.touch_right ? 1 : 0)) -  
                       (prevx + (move.touch_left ? 1 : 0)), h1);  
         cr->fill();  
   
         Gdk::Cairo::set_source_rgba(cr, black);  
         cr->move_to(std::max(x, prevx2 + 1), 0.5);  
         cr->line_to(x2 + 1, 0.5);  
         cr->move_to(std::max(x, prevx2 + 1), h1 - 0.5);  
         cr->line_to(x2 + 1, h1 - 0.5);  
         cr->stroke();  
   
         Gdk::Cairo::set_source_rgba(cr, red);  
         cr->rectangle(std::max(x + 1, prevx2), 1,  
                       x2 - std::max(x + 1, prevx2), h1 - 2);  
         cr->fill();  
     } else {  
         Gdk::Cairo::set_source_rgba(cr, bg);  
         cr->rectangle(std::max(x2 + 1, prevx + (move.touch_left ? 1 : 0)), 0,  
                       prevx2 + 1 - (move.touch_right ? 1 : 0) -  
                       std::max(x2 + 1, prevx + (move.touch_left ? 1 : 0)), h1);  
         cr->fill();  
   
         Gdk::Cairo::set_source_rgba(cr, black);  
         cr->move_to(x, 0.5);  
         cr->line_to(std::min(x2, prevx - 1) + 1, 0.5);  
         cr->move_to(x, h1 - 0.5);  
         cr->line_to(std::min(x2, prevx - 1) + 1, h1 - 0.5);  
         cr->stroke();  
   
         Gdk::Cairo::set_source_rgba(cr, red);  
         cr->rectangle(x + 1, 1, std::min(x2 - 1, prevx) - x, h1 - 2);  
         cr->fill();  
     }  
931    
932      move.pos = k;      get_window()->invalidate_rect(rect, false);
     move.touch_left = new_touch_left;  
     move.touch_right = new_touch_right;  
933  }  }
934    
935    
# Line 804  bool RegionChooser::on_motion_notify_eve Line 946  bool RegionChooser::on_motion_notify_eve
946          event->y >= REGION_BLOCK_HEIGHT &&          event->y >= REGION_BLOCK_HEIGHT &&
947          event->y < REGION_BLOCK_HEIGHT + KEYBOARD_HEIGHT)          event->y < REGION_BLOCK_HEIGHT + KEYBOARD_HEIGHT)
948      {      {
949          const int k = int(event->x / (get_width() - 1) * 128.0);          const int k = x_to_key(event->x, get_width() - 1);
950          if (k != currentActiveKey) {          if (k != currentActiveKey) {
951              int velocity =              int velocity =
952                  (event->y >= REGION_BLOCK_HEIGHT + KEYBOARD_HEIGHT - 1) ? 127 :                  (event->y >= REGION_BLOCK_HEIGHT + KEYBOARD_HEIGHT - 1) ? 127 :
# Line 849  bool RegionChooser::is_in_resize_zone(do Line 991  bool RegionChooser::is_in_resize_zone(do
991          for (gig::Region* r = regions.first(); r ; r = next_region) {          for (gig::Region* r = regions.first(); r ; r = next_region) {
992              next_region = regions.next();              next_region = regions.next();
993    
994              int lo = int(w * (r->KeyRange.low) / 128.0 + 0.5);              int lo = key_to_x(r->KeyRange.low, w);
995              if (x <= lo - 2) break;              if (x <= lo - 2) break;
996              if (x < lo + 2) {              if (x < lo + 2) {
997                  resize.region = r;                  resize.region = r;
# Line 872  bool RegionChooser::is_in_resize_zone(do Line 1014  bool RegionChooser::is_in_resize_zone(do
1014                  return resize.min != resize.max;                  return resize.min != resize.max;
1015              }              }
1016              if (!next_region || r->KeyRange.high + 1 != next_region->KeyRange.low) {              if (!next_region || r->KeyRange.high + 1 != next_region->KeyRange.low) {
1017                  int hi = int(w * (r->KeyRange.high + 1) / 128.0 + 0.5);                  int hi = key_to_x(r->KeyRange.high + 1, w);
1018                  if (x <= hi - 2) break;                  if (x <= hi - 2) break;
1019                  if (x < hi + 2) {                  if (x < hi + 2) {
1020                      // edit high limit                      // edit high limit
# Line 922  void RegionChooser::show_region_properti Line 1064  void RegionChooser::show_region_properti
1064      // add OK and CANCEL buttons to the dialog      // add OK and CANCEL buttons to the dialog
1065      dialog.add_button(Gtk::Stock::OK, 0);      dialog.add_button(Gtk::Stock::OK, 0);
1066      dialog.add_button(Gtk::Stock::CANCEL, 1);      dialog.add_button(Gtk::Stock::CANCEL, 1);
1067        dialog.set_position(Gtk::WIN_POS_MOUSE);
1068      dialog.show_all_children();      dialog.show_all_children();
1069      if (!dialog.run()) { // OK selected ...      if (!dialog.run()) { // OK selected ...
1070          region->KeyGroup =          region->KeyGroup =

Legend:
Removed from v.2169  
changed lines
  Added in v.3226

  ViewVC Help
Powered by ViewVC