/[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 1261 by persson, Thu Jul 5 17:12:20 2007 UTC revision 1303 by persson, Sun Aug 26 09:29:52 2007 UTC
# Line 31  RegionChooser::RegionChooser() Line 31  RegionChooser::RegionChooser()
31  {  {
32      Glib::RefPtr<Gdk::Colormap> colormap = get_default_colormap();      Glib::RefPtr<Gdk::Colormap> colormap = get_default_colormap();
33    
     black = Gdk::Color("black");  
     white = Gdk::Color("white");  
34      red = Gdk::Color("#8070ff");      red = Gdk::Color("#8070ff");
35      blue = Gdk::Color("#c098ff");      grey1 = Gdk::Color("#b0b0b0");
     green = Gdk::Color("#a088ff");  
     grey1 = Gdk::Color("red");  
36    
     colormap->alloc_color(black);  
     colormap->alloc_color(white);  
37      colormap->alloc_color(red);      colormap->alloc_color(red);
     colormap->alloc_color(blue);  
     colormap->alloc_color(green);  
38      colormap->alloc_color(grey1);      colormap->alloc_color(grey1);
39      instrument = 0;      instrument = 0;
40      region = 0;      region = 0;
41      resize.active = false;      resize.active = false;
42        move.active = false;
43      cursor_is_resize = false;      cursor_is_resize = false;
44      h1 = 20;      h1 = 20;
45      width = 800;      width = 800;
# Line 117  bool RegionChooser::on_expose_event(GdkE Line 110  bool RegionChooser::on_expose_event(GdkE
110      Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();      Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();
111      Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();      Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();
112    
113        Glib::RefPtr<Pango::Context> context = get_pango_context();
114        Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
115    
116      window->draw_rectangle(black, false, 0, h1, w, h - 1);      window->draw_rectangle(black, false, 0, h1, w, h - 1);
117      window->draw_rectangle(white, true, 1, h1 + 1, w - 1, h - 2);      gc->set_foreground(grey1);
118        int x1 = int(w * 20.5 / 128.0 + 0.5);
119        int x2 = int(w * 109.5 / 128.0 + 0.5);
120        window->draw_rectangle(gc, true, 1, h1 + 1,
121                               x1 - 1, h - 2);
122        window->draw_rectangle(white, true, x1 + 1, h1 + 1, x2 - x1 - 1, h - 2);
123        window->draw_rectangle(gc, true, x2 + 1, h1 + 1,
124                               w - x2 - 1, h - 2);
125        int octave = -1;
126      for (int i = 0 ; i < 128 ; i++) {      for (int i = 0 ; i < 128 ; i++) {
127          int note = (i + 3) % 12;          int note = (i + 3) % 12;
128          int x = int(w * i / 128.0 + 0.5);          int x = int(w * i / 128.0 + 0.5);
# Line 132  bool RegionChooser::on_expose_event(GdkE Line 136  bool RegionChooser::on_expose_event(GdkE
136          } else if (note == 3 || note == 8) {          } else if (note == 3 || note == 8) {
137              window->draw_line(black, x, h1 + 1, x, h1 + h);              window->draw_line(black, x, h1 + 1, x, h1 + h);
138          }          }
139            if (note == 3) {
140                char buf[30];
141                sprintf(buf, "<span size=\"x-small\">%d</span>", octave);
142                layout->set_markup(buf);
143                Pango::Rectangle rectangle = layout->get_logical_extents();
144                double text_w = double(rectangle.get_width()) / Pango::SCALE;
145                double text_h = double(rectangle.get_height()) / Pango::SCALE;
146                double x2 = w * (i + 0.75) / 128.0;
147                window->draw_layout(black, int(x2 - text_w / 2 + 1),
148                                    int(h1 + h - text_h + 0.5), layout);
149                octave++;
150            }
151      }      }
152    
153      if (instrument) {      if (instrument) {
154          int i = 0;          int i = 0;
155          gig::Region *nextRegion;          gig::Region *next_region;
156          int x3 = -1;          int x3 = -1;
157          for (gig::Region *r = instrument->GetFirstRegion() ;          for (gig::Region *r = instrument->GetFirstRegion() ;
158               r ;               r ;
159               r = nextRegion) {               r = next_region) {
160    
161              if (x3 < 0) x3 = int(w * (r->KeyRange.low) / 128.0 + 0.5);              if (x3 < 0) x3 = int(w * (r->KeyRange.low) / 128.0 + 0.5);
162              nextRegion = instrument->GetNextRegion();              next_region = instrument->GetNextRegion();
163              if (!nextRegion || r->KeyRange.high + 1 != nextRegion->KeyRange.low) {              if (!next_region || r->KeyRange.high + 1 != next_region->KeyRange.low) {
164                  int x2 = int(w * (r->KeyRange.high + 1) / 128.0 + 0.5);                  int x2 = int(w * (r->KeyRange.high + 1) / 128.0 + 0.5);
165                  window->draw_line(black, x3, 0, x2, 0);                  window->draw_line(black, x3, 0, x2, 0);
166                  window->draw_line(black, x3, h1 - 1, x2, h1 - 1);                  window->draw_line(black, x3, h1 - 1, x2, h1 - 1);
# Line 248  bool RegionChooser::on_button_release_ev Line 264  bool RegionChooser::on_button_release_ev
264              get_window()->set_cursor();              get_window()->set_cursor();
265              cursor_is_resize = false;              cursor_is_resize = false;
266          }          }
267        } else if (move.active) {
268            get_window()->pointer_ungrab(event->time);
269            move.active = false;
270    
271            if (move.pos) {
272                region->KeyRange.low += move.pos;
273                region->KeyRange.high += move.pos;
274    
275                // find the r which is the first one to the right of region
276                // at its new position
277                gig::Region* r;
278                gig::Region* prev_region = 0;
279                for (r = instrument->GetFirstRegion() ; r ; r = instrument->GetNextRegion()) {
280                    if (r->KeyRange.low > region->KeyRange.low) break;
281                    prev_region = r;
282                }
283    
284                // place region before r if it's not already there
285                if (prev_region != region) {
286                    instrument->MoveRegion(region, r);
287                }
288            }
289    
290            if (is_in_resize_zone(event->x, event->y)) {
291                get_window()->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));
292                cursor_is_resize = true;
293            }
294      }      }
295      return true;      return true;
296  }  }
# Line 271  bool RegionChooser::on_button_press_even Line 314  bool RegionChooser::on_button_press_even
314          }          }
315      } else {      } else {
316          if (is_in_resize_zone(event->x, event->y)) {          if (is_in_resize_zone(event->x, event->y)) {
             Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);  
317              get_window()->pointer_grab(false,              get_window()->pointer_grab(false,
318                                         Gdk::BUTTON_RELEASE_MASK |                                         Gdk::BUTTON_RELEASE_MASK |
319                                         Gdk::POINTER_MOTION_MASK |                                         Gdk::POINTER_MOTION_MASK |
320                                         Gdk::POINTER_MOTION_HINT_MASK,                                         Gdk::POINTER_MOTION_HINT_MASK,
321                                         double_arrow, event->time);                                         Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW), event->time);
322              resize.active = true;              resize.active = true;
323          } else {          } else {
324              gig::Region* r = get_region(k);              gig::Region* r = get_region(k);
# Line 284  bool RegionChooser::on_button_press_even Line 326  bool RegionChooser::on_button_press_even
326                  region = r;                  region = r;
327                  queue_draw();                  queue_draw();
328                  region_selected();                  region_selected();
329    
330                    get_window()->pointer_grab(false,
331                                               Gdk::BUTTON_RELEASE_MASK |
332                                               Gdk::POINTER_MOTION_MASK |
333                                               Gdk::POINTER_MOTION_HINT_MASK,
334                                               Gdk::Cursor(Gdk::FLEUR), event->time);
335                    move.active = true;
336                    move.from_x = event->x;
337                    move.pos = 0;
338              }              }
339          }          }
340      }      }
# Line 292  bool RegionChooser::on_button_press_even Line 343  bool RegionChooser::on_button_press_even
343    
344  gig::Region* RegionChooser::get_region(int key)  gig::Region* RegionChooser::get_region(int key)
345  {  {
346        gig::Region* prev_region = 0;
347        gig::Region* next_region;
348      for (gig::Region *r = instrument->GetFirstRegion() ; r ;      for (gig::Region *r = instrument->GetFirstRegion() ; r ;
349           r = instrument->GetNextRegion()) {           r = next_region) {
350            next_region = instrument->GetNextRegion();
351    
352          if (key < r->KeyRange.low) return 0;          if (key < r->KeyRange.low) return 0;
353          if (key <= r->KeyRange.high) return r;          if (key <= r->KeyRange.high) {
354                move.touch_left = prev_region && prev_region->KeyRange.high + 1 == r->KeyRange.low;
355                move.touch_right = next_region && r->KeyRange.high + 1 == next_region->KeyRange.low;
356                return r;
357            }
358            prev_region = r;
359      }      }
360      return 0;      return 0;
361  }  }
362    
363  bool RegionChooser::on_motion_notify_event(GdkEventMotion* event)  void RegionChooser::motion_resize_region(int x, int y)
364  {  {
365      const int w = width - 1;      const int w = width - 1;
366      Glib::RefPtr<Gdk::Window> window = get_window();      Glib::RefPtr<Gdk::Window> window = get_window();
     int x, y;  
     Gdk::ModifierType state = Gdk::ModifierType(0);  
     window->get_pointer(x, y, state);  
     if (resize.active) {  
         int k = int(double(x) / w * 128.0 + 0.5);  
367    
368          if (k < resize.min) k = resize.min;      int k = int(double(x) / w * 128.0 + 0.5);
         else if (k > resize.max) k = resize.max;  
369    
370          if (k != resize.pos) {      if (k < resize.min) k = resize.min;
371              if (resize.mode == resize.undecided) {      else if (k > resize.max) k = resize.max;
372                  if (k < resize.pos) {  
373                      // edit high limit of prev_region      if (k != resize.pos) {
374                      resize.max = resize.region->KeyRange.low;          if (resize.mode == resize.undecided) {
375                      resize.region = resize.prev_region;              if (k < resize.pos) {
376                      resize.mode = resize.moving_high_limit;                  // edit high limit of prev_region
377                  } else {                  resize.max = resize.region->KeyRange.low;
378                      // edit low limit of region                  resize.region = resize.prev_region;
379                      resize.min = resize.prev_region->KeyRange.high + 1;                  resize.mode = resize.moving_high_limit;
380                      resize.mode = resize.moving_low_limit;              } else {
381                  }                  // edit low limit of region
382                    resize.min = resize.prev_region->KeyRange.high + 1;
383                    resize.mode = resize.moving_low_limit;
384              }              }
385              Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();          }
386              Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();          Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();
387              if (region == resize.region) {          Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();
388                  gc->set_foreground(red);          if (region == resize.region) {
389                  white = gc;              gc->set_foreground(red);
390              }              white = gc;
391              Glib::RefPtr<const Gdk::GC> bg = get_style()->get_bg_gc(Gtk::STATE_NORMAL);          }
392              int prevx = int(w * resize.pos / 128.0 + 0.5);          Glib::RefPtr<const Gdk::GC> bg = get_style()->get_bg_gc(Gtk::STATE_NORMAL);
393              x = int(w * k / 128.0 + 0.5);          int prevx = int(w * resize.pos / 128.0 + 0.5);
394            x = int(w * k / 128.0 + 0.5);
395              if (resize.mode == resize.moving_high_limit) {  
396                  if (k > resize.pos) {          if (resize.mode == resize.moving_high_limit) {
397                      window->draw_rectangle(white, true, prevx, 1, x - prevx, h1 - 2);              if (k > resize.pos) {
398                      window->draw_line(black, prevx, 0, x, 0);                  window->draw_rectangle(white, true, prevx, 1, x - prevx, h1 - 2);
399                      window->draw_line(black, prevx, h1 - 1, x, h1 - 1);                  window->draw_line(black, prevx, 0, x, 0);
400                    window->draw_line(black, prevx, h1 - 1, x, h1 - 1);
401                } else {
402                    int xx = ((resize.pos == resize.max && resize.max != 128) ? 1 : 0);
403                    window->draw_rectangle(bg, true, x + 1, 0, prevx - x - xx, h1);
404                }
405            } else {
406                if (k < resize.pos) {
407                    window->draw_rectangle(white, true, x + 1, 1, prevx - x, h1 - 2);
408                    window->draw_line(black, x, 0, prevx, 0);
409                    window->draw_line(black, x, h1 - 1, prevx, h1 - 1);
410                } else {
411                    int xx = ((resize.pos == resize.min && resize.min != 0) ? 1 : 0);
412                    window->draw_rectangle(bg, true, prevx + xx, 0, x - prevx - xx, h1);
413                }
414            }
415            window->draw_line(black, x, 1, x, h1 - 2);
416            resize.pos = k;
417        }
418    }
419    
420    void RegionChooser::motion_move_region(int x, int y)
421    {
422        const int w = width - 1;
423        Glib::RefPtr<Gdk::Window> window = get_window();
424    
425        int k = int(double(x - move.from_x) / w * 128.0 + 0.5);
426        if (k == move.pos) return;
427        int new_k;
428        bool new_touch_left;
429        bool new_touch_right;
430        int a = 0;
431        if (k > move.pos) {
432            for (gig::Region* r = instrument->GetFirstRegion() ; ;
433                 r = instrument->GetNextRegion()) {
434                if (r != region) {
435                    int b = r ? r->KeyRange.low : 128;
436    
437                    // gap: from a to b (not inclusive b)
438    
439                    if (region->KeyRange.high + move.pos >= b) {
440                        // not found the current gap yet, just continue
441                  } else {                  } else {
442                      int xx = ((resize.pos == resize.max && resize.max != 128) ? 1 : 0);  
443                      window->draw_rectangle(bg, true, x + 1, 0, prevx - x - xx, h1);                      if (a > region->KeyRange.low + k) {
444                            // this gap is too far to the right, break
445                            break;
446                        }
447    
448                        int newhigh = std::min(region->KeyRange.high + k, b - 1);
449                        int newlo = newhigh - (region->KeyRange.high - region->KeyRange.low);
450    
451                        if (newlo >= a) {
452                            // yes it fits - it's a candidate
453                            new_k = newlo - region->KeyRange.low;
454                            new_touch_left = a > 0 && a == newlo;
455                            new_touch_right = b < 128 && newhigh + 1 == b;
456                        }
457                  }                  }
458              } else {                  if (!r) break;
459                  if (k < resize.pos) {                  a = r->KeyRange.high + 1;
460                      window->draw_rectangle(white, true, x + 1, 1, prevx - x, h1 - 2);              }
461                      window->draw_line(black, x, 0, prevx, 0);          }
462                      window->draw_line(black, x, h1 - 1, prevx, h1 - 1);      } else {
463            for (gig::Region* r = instrument->GetFirstRegion() ; ;
464                 r = instrument->GetNextRegion()) {
465                if (r != region) {
466                    int b = r ? r->KeyRange.low : 128;
467    
468                    // gap from a to b (not inclusive b)
469    
470                    if (region->KeyRange.high + k >= b) {
471                        // not found the current gap yet, just continue
472                  } else {                  } else {
473                      int xx = ((resize.pos == resize.min && resize.min != 0) ? 1 : 0);  
474                      window->draw_rectangle(bg, true, prevx + xx, 0, x - prevx - xx, h1);                      if (a > region->KeyRange.low + move.pos) {
475                            // this gap is too far to the right, break
476                            break;
477                        }
478    
479                        int newlo = std::max(region->KeyRange.low + k, a);
480                        int newhigh = newlo + (region->KeyRange.high - region->KeyRange.low);
481    
482                        if (newhigh < b) {
483                            // yes it fits - break as the first one is the best
484                            new_k = newlo - region->KeyRange.low;
485                            new_touch_left = a > 0 && a == newlo;
486                            new_touch_right = b < 128 && newhigh + 1 == b;
487                            break;
488                        }
489                  }                  }
490                    if (!r) break;
491                    a = r->KeyRange.high + 1;
492              }              }
             window->draw_line(black, x, 1, x, h1 - 2);  
             resize.pos = k;  
493          }          }
494        }
495        k = new_k;
496        if (k == move.pos) return;
497    
498        Glib::RefPtr<const Gdk::GC> bg = get_style()->get_bg_gc(Gtk::STATE_NORMAL);
499        int prevx = int(w * (move.pos + region->KeyRange.low) / 128.0 + 0.5);
500        x = int(w * (k + region->KeyRange.low) / 128.0 + 0.5);
501        int prevx2 = int(w * (move.pos + region->KeyRange.high + 1) / 128.0 + 0.5);
502        int x2 = int(w * (k + region->KeyRange.high + 1) / 128.0 + 0.5);
503        Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();
504        gc->set_foreground(red);
505    
506        if (!new_touch_left) window->draw_line(black, x, 1, x, h1 - 2);
507        if (!new_touch_right) window->draw_line(black, x2, 1, x2, h1 - 2);
508    
509        if (k > move.pos) {
510            window->draw_rectangle(bg, true, prevx + (move.touch_left ? 1 : 0), 0,
511                                   std::min(x, prevx2 + 1 - (move.touch_right ? 1 : 0)) -
512                                   (prevx + (move.touch_left ? 1 : 0)), h1);
513    
514            window->draw_line(black, std::max(x, prevx2 + 1), 0, x2, 0);
515            window->draw_line(black, std::max(x, prevx2 + 1), h1 - 1, x2, h1 - 1);
516            window->draw_rectangle(gc, true, std::max(x + 1, prevx2), 1,
517                                   x2 - std::max(x + 1, prevx2), h1 - 2);
518        } else {
519            window->draw_rectangle(bg, true, std::max(x2 + 1, prevx + (move.touch_left ? 1 : 0)), 0,
520                                   prevx2 + 1 - (move.touch_right ? 1 : 0) -
521                                   std::max(x2 + 1, prevx + (move.touch_left ? 1 : 0)), h1);
522    
523            window->draw_line(black, x, 0, std::min(x2, prevx - 1), 0);
524            window->draw_line(black, x, h1 - 1, std::min(x2, prevx - 1), h1 - 1);
525    
526            window->draw_rectangle(gc, true, x + 1, 1, std::min(x2 - 1, prevx) - x, h1 - 2);
527        }
528    
529        move.pos = k;
530        move.touch_left = new_touch_left;
531        move.touch_right = new_touch_right;
532    }
533    
534    
535    bool RegionChooser::on_motion_notify_event(GdkEventMotion* event)
536    {
537        Glib::RefPtr<Gdk::Window> window = get_window();
538        int x, y;
539        Gdk::ModifierType state = Gdk::ModifierType(0);
540        window->get_pointer(x, y, state);
541    
542        if (resize.active) {
543            motion_resize_region(x, y);
544        } else if (move.active) {
545            motion_move_region(x, y);
546      } else {      } else {
547          if (is_in_resize_zone(x, y)) {          if (is_in_resize_zone(x, y)) {
548              if (!cursor_is_resize) {              if (!cursor_is_resize) {
549                  Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);                  window->set_cursor(Gdk::Cursor(Gdk::SB_H_DOUBLE_ARROW));
                 window->set_cursor(double_arrow);  
550                  cursor_is_resize = true;                  cursor_is_resize = true;
551              }              }
552          } else if (cursor_is_resize) {          } else if (cursor_is_resize) {

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

  ViewVC Help
Powered by ViewVC