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

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

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

revision 1078 by persson, Tue Mar 6 20:15:23 2007 UTC revision 1104 by persson, Sun Mar 18 17:15:00 2007 UTC
# Line 18  Line 18 
18   */   */
19    
20  #include "dimregionchooser.h"  #include "dimregionchooser.h"
21    #include <gdkmm/cursor.h>
22    
23  DimRegionChooser::DimRegionChooser()  DimRegionChooser::DimRegionChooser()
24  {  {
# Line 40  DimRegionChooser::DimRegionChooser() Line 41  DimRegionChooser::DimRegionChooser()
41      region = 0;      region = 0;
42      dimregno = -1;      dimregno = -1;
43      focus_line = 0;      focus_line = 0;
44        resize.active = false;
45        cursor_is_resize = false;
46        h = 20;
47        w = 800;
48      set_flags(Gtk::CAN_FOCUS);      set_flags(Gtk::CAN_FOCUS);
49      add_events(Gdk::BUTTON_PRESS_MASK);      add_events(Gdk::BUTTON_PRESS_MASK | Gdk::POINTER_MOTION_MASK |
50                   Gdk::POINTER_MOTION_HINT_MASK);
51    
52      for (int i = 0 ; i < 256 ; i++) {      for (int i = 0 ; i < 256 ; i++) {
53          dimvalue_from[i] = 0;          dimvalue_from[i] = 0;
# Line 74  bool DimRegionChooser::on_expose_event(G Line 80  bool DimRegionChooser::on_expose_event(G
80      Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);      Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(context);
81    
82      window->clear();      window->clear();
     const int h = 20;  
     const int w = 800;  
83      int y = 0;      int y = 0;
84      double maxwidth = 0;      double maxwidth = 0;
85      for (int i = 0 ; i < region->Dimensions ; i++) {      for (int i = 0 ; i < region->Dimensions ; i++) {
# Line 127  bool DimRegionChooser::on_expose_event(G Line 131  bool DimRegionChooser::on_expose_event(G
131              }              }
132              layout->set_text(dstr);              layout->set_text(dstr);
133    
134              Pango::Rectangle rectangle = layout->get_logical_extents();              Pango::Rectangle rectangle = layout->get_logical_extents();
135              double text_w = double(rectangle.get_width()) / Pango::SCALE;              double text_w = double(rectangle.get_width()) / Pango::SCALE;
136              if (text_w > maxwidth) maxwidth = text_w;              if (text_w > maxwidth) maxwidth = text_w;
137              double text_h = double(rectangle.get_height()) / Pango::SCALE;              double text_h = double(rectangle.get_height()) / Pango::SCALE;
138              Glib::RefPtr<const Gdk::GC> fg = get_style()->get_fg_gc(get_state());              Glib::RefPtr<const Gdk::GC> fg = get_style()->get_fg_gc(get_state());
139              window->draw_layout(fg, 4, int(y + (h - text_h) / 2 + 0.5), layout);              window->draw_layout(fg, 4, int(y + (h - text_h) / 2 + 0.5), layout);
140    
# Line 146  bool DimRegionChooser::on_expose_event(G Line 150  bool DimRegionChooser::on_expose_event(G
150    
151              if (has_focus() && focus_line == i) {              if (has_focus() && focus_line == i) {
152                  Gdk::Rectangle farea(0, y, 150, 20);                  Gdk::Rectangle farea(0, y, 150, 20);
153                  get_style()->paint_focus(window, get_state(), farea, *this, "", 0, y, label_width, 20);                  get_style()->paint_focus(window, get_state(), farea, *this, "",
154                                             0, y, label_width, 20);
155              }              }
156    
157              Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();              Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();
158              window->draw_line(black, label_width, y, w - 1, y);              window->draw_line(black, label_width, y, w - 1, y);
159              window->draw_line(black, w - 1, y + h - 1, label_width, y + h - 1);              window->draw_line(black, w - 1, y + h - 1, label_width, y + h - 1);
160              window->draw_rectangle(get_style()->get_white_gc(), true, label_width + 1, y + 1, (w - label_width - 2), h - 2);              window->draw_rectangle(get_style()->get_white_gc(), true,
161                                       label_width + 1, y + 1, (w - label_width - 2), h - 2);
162    
163              int c = 0;              int c = 0;
164              if (dimregno >= 0) {              if (dimregno >= 0) {
# Line 194  bool DimRegionChooser::on_expose_event(G Line 200  bool DimRegionChooser::on_expose_event(G
200                          int v = upperLimit + 1;                          int v = upperLimit + 1;
201                          int x2 = int((w - label_width - 1) * v / 128.0 + 0.5);                          int x2 = int((w - label_width - 1) * v / 128.0 + 0.5);
202                          if (j == dr && x1 < x2) {                          if (j == dr && x1 < x2) {
203                              window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1, (x2 - x1) - 1, h - 2);                              window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1,
204                                                       (x2 - x1) - 1, h - 2);
205                              break;                              break;
206                          }                          }
207                          x1 = x2;                          x1 = x2;
# Line 203  bool DimRegionChooser::on_expose_event(G Line 210  bool DimRegionChooser::on_expose_event(G
210                      if (dr < nbZones) {                      if (dr < nbZones) {
211                          int x1 = int((w - label_width - 1) * dr / double(nbZones) + 0.5);                          int x1 = int((w - label_width - 1) * dr / double(nbZones) + 0.5);
212                          int x2 = int((w - label_width - 1) * (dr + 1) / double(nbZones) + 0.5);                          int x2 = int((w - label_width - 1) * (dr + 1) / double(nbZones) + 0.5);
213                          window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1, (x2 - x1) - 1, h - 2);                          window->draw_rectangle(gc, true, label_width + x1 + 1, y + 1,
214                                                   (x2 - x1) - 1, h - 2);
215                      }                      }
216                  }                  }
217              }              }
# Line 216  bool DimRegionChooser::on_expose_event(G Line 224  bool DimRegionChooser::on_expose_event(G
224      return true;      return true;
225  }  }
226    
   
227  void DimRegionChooser::on_size_request(GtkRequisition* requisition)  void DimRegionChooser::on_size_request(GtkRequisition* requisition)
228  {  {
229      printf("DimRegionChooser::on_size_request\n");      printf("DimRegionChooser::on_size_request\n");
230      *requisition = GtkRequisition();      *requisition = GtkRequisition();
231      requisition->height = region ? region->Dimensions * 20 : 0;      requisition->height = region ? nbDimensions * 20 : 0;
232      requisition->width = 800;      requisition->width = 800;
233  }  }
234    
 void DimRegionChooser::set_instrument(gig::Instrument* instrument)  
 {  
     this->instrument = instrument;  
     this->region = 0;  
     this->dimregno = -1;  
     queue_draw();  
 }  
   
235  void DimRegionChooser::set_region(gig::Region* region)  void DimRegionChooser::set_region(gig::Region* region)
236  {  {
237      this->region = region;      this->region = region;
238      dimregno = 0;      dimregno = 0;
239      int bitcount = 0;      nbDimensions = 0;
240      for (int dim = 0 ; dim < region->Dimensions ; dim++) {      if (region) {
241          int from = dimvalue_from[region->pDimensionDefinitions[dim].dimension];          int bitcount = 0;
242          int to = dimvalue_to[region->pDimensionDefinitions[dim].dimension];          for (int dim = 0 ; dim < region->Dimensions ; dim++) {
243          int z;              if (region->pDimensionDefinitions[dim].bits == 0) continue;
244          switch (region->pDimensionDefinitions[dim].split_type) {              nbDimensions++;
245          case gig::split_type_normal:  
246              z = int((to + from) / 2.0 / region->pDimensionDefinitions[dim].zone_size);              int from = dimvalue_from[region->pDimensionDefinitions[dim].dimension];
247              break;              int to = dimvalue_to[region->pDimensionDefinitions[dim].dimension];
248          case gig::split_type_bit:              int z;
249              z = std::min(from, region->pDimensionDefinitions[dim].zones - 1);              switch (region->pDimensionDefinitions[dim].split_type) {
250              break;              case gig::split_type_normal:
251          }                  z = int((to + from) / 2.0 / region->pDimensionDefinitions[dim].zone_size);
252          int mask =                  break;
253              ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) <<              case gig::split_type_bit:
254                bitcount);                  z = std::min(from, region->pDimensionDefinitions[dim].zones - 1);
255          dimregno &= mask;                  break;
256          dimregno |= (z << bitcount);              }
257          bitcount += region->pDimensionDefinitions[dim].bits;              int mask =
258                    ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) <<
259                      bitcount);
260                dimregno &= mask;
261                dimregno |= (z << bitcount);
262                bitcount += region->pDimensionDefinitions[dim].bits;
263            }
264            dimreg = region->pDimensionRegions[dimregno];
265        } else {
266            dimreg = 0;
267      }      }
     dimreg = region->pDimensionRegions[dimregno];  
268      sel_changed_signal.emit();      sel_changed_signal.emit();
269      queue_resize();      queue_resize();
270  }  }
271    
272  /*  bool DimRegionChooser::on_button_release_event(GdkEventButton* event)
273    void DimRegionChooser::set_dimregno(int x) {  {
274    this->dimregno = x;      if (resize.active) {
275    queue_draw();          get_window()->pointer_ungrab(event->time);
276    }          resize.active = false;
277  */  
278            if (region->pDimensionDefinitions[resize.dimension].dimension == gig::dimension_velocity) {
279    
280                int bitpos = 0;
281                for (int j = 0 ; j < resize.dimension ; j++) {
282                    bitpos += region->pDimensionDefinitions[j].bits;
283                }
284                int mask =
285                         ~(((1 << region->pDimensionDefinitions[resize.dimension].bits) - 1) << bitpos);
286                int c = dimregno & mask; // mask away this dimension
287    
288                gig::DimensionRegion *d = region->pDimensionRegions[c + resize.offset];
289                if (d->DimensionUpperLimits[resize.dimension]) {
290                    d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
291                } else {
292                    d->VelocityUpperLimit = resize.pos - 1;
293                }
294    
295            } else {
296                for (int i = 0 ; i < region->DimensionRegions ; ) {
297    
298                    gig::DimensionRegion *d = region->pDimensionRegions[i + resize.offset];
299                    d->DimensionUpperLimits[resize.dimension] = resize.pos - 1;
300    
301                    int bitpos = 0;
302                    int j;
303                    for (j = 0 ; j < region->Dimensions ; j++) {
304                        if (j != resize.dimension) {
305                            int maxzones = 1 << region->pDimensionDefinitions[j].bits;
306                            int dimj = (i >> bitpos) & (maxzones - 1);
307                            if (dimj + 1 < region->pDimensionDefinitions[j].zones) break;
308                        }
309                        bitpos += region->pDimensionDefinitions[j].bits;
310                    }
311                    if (j == region->Dimensions) break;
312                    i = (i & ~((1 << bitpos) - 1)) + (1 << bitpos);
313                }
314            }
315    
316            if (!is_in_resize_zone(event->x, event->y) && cursor_is_resize) {
317                get_window()->set_cursor();
318                cursor_is_resize = false;
319            }
320        }
321        return true;
322    }
323    
324  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)  bool DimRegionChooser::on_button_press_event(GdkEventButton* event)
325  {  {
326      const int h = 20;      if (region && event->y < nbDimensions * h &&
327      const int w = 800;          event->x >= label_width && event->x < w) {
328    
329      if (region) {          if (is_in_resize_zone(event->x, event->y)) {
330          if (event->y < region->Dimensions * h &&              Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);
331              event->x >= label_width && event->x < w) {              get_window()->pointer_grab(false,
332                                           Gdk::BUTTON_RELEASE_MASK |
333              int dim = int(event->y / h);                                         Gdk::POINTER_MOTION_MASK |
334                                           Gdk::POINTER_MOTION_HINT_MASK,
335                                           double_arrow, event->time);
336                resize.active = true;
337            } else {
338                int ydim = int(event->y / h);
339                int dim;
340                for (dim = 0 ; dim < region->Dimensions ; dim++) {
341                    if (region->pDimensionDefinitions[dim].bits == 0) continue;
342                    if (ydim == 0) break;
343                    ydim--;
344                }
345              int nbZones = region->pDimensionDefinitions[dim].zones;              int nbZones = region->pDimensionDefinitions[dim].zones;
346    
347              int z = -1;              int z = -1;
# Line 347  bool DimRegionChooser::on_button_press_e Line 410  bool DimRegionChooser::on_button_press_e
410      return true;      return true;
411  }  }
412    
413    bool DimRegionChooser::on_motion_notify_event(GdkEventMotion* event)
414    {
415        Glib::RefPtr<Gdk::Window> window = get_window();
416        int x, y;
417        Gdk::ModifierType state = Gdk::ModifierType(0);
418        window->get_pointer(x, y, state);
419    
420        if (resize.active) {
421            int k = int((x - label_width) * 128.0 / (w - label_width - 1) + 0.5);
422    
423            if (k < resize.min) k = resize.min;
424            else if (k > resize.max) k = resize.max;
425    
426            if (k < 2) k = 2; // k is upper limit + 1, upper limit 0 is forbidden
427    
428            if (k != resize.pos) {
429                Glib::RefPtr<const Gdk::GC> black = get_style()->get_black_gc();
430                Glib::RefPtr<const Gdk::GC> white = get_style()->get_white_gc();
431    
432                int prevx = int((w - label_width - 1) * resize.pos / 128.0 + 0.5) + label_width;
433                int x = int((w - label_width - 1) * k / 128.0 + 0.5) + label_width;
434                int y = resize.dimension * h;
435    
436                if (resize.selected == resize.none) {
437                    if (resize.pos != resize.min && resize.pos != resize.max) {
438                        window->draw_line(white, prevx, y + 1, prevx, y + h - 2);
439                    }
440                } else {
441                    gc->set_foreground(red);
442    
443                    Glib::RefPtr<const Gdk::GC> left;
444                    Glib::RefPtr<const Gdk::GC> right;
445                    if (resize.selected == resize.left) {
446                        left = gc;
447                        right = white;
448                    } else {
449                        left = white;
450                        right = gc;
451                    }
452    
453                    if (k > resize.pos) {
454                        int xx = resize.pos == resize.min ? 1 : 0;
455                        window->draw_rectangle(left, true,
456                                               prevx + xx, y + 1, x - prevx - xx, h - 2);
457                    } else {
458                        int xx = resize.pos == resize.max ? 0 : 1;
459                        window->draw_rectangle(right, true,
460                                               x, y + 1, prevx - x + xx, h - 2);
461                    }
462                }
463                window->draw_line(black, x, y + 1, x, y + h - 2);
464                resize.pos = k;
465            }
466        } else {
467            if (is_in_resize_zone(x, y)) {
468                if (!cursor_is_resize) {
469                    Gdk::Cursor double_arrow(Gdk::SB_H_DOUBLE_ARROW);
470                    window->set_cursor(double_arrow);
471                    cursor_is_resize = true;
472                }
473            } else if (cursor_is_resize) {
474                window->set_cursor();
475                cursor_is_resize = false;
476            }
477        }
478        return true;
479    }
480    
481    bool DimRegionChooser::is_in_resize_zone(double x, double y)
482    {
483        if (region && y < nbDimensions * h && x >= label_width && x < w) {
484            int ydim = int(y / h);
485            int dim;
486            int bitpos = 0;
487            for (dim = 0 ; dim < region->Dimensions ; dim++) {
488                if (region->pDimensionDefinitions[dim].bits == 0) continue;
489                if (ydim == 0) break;
490                ydim--;
491                bitpos += region->pDimensionDefinitions[dim].bits;
492            }
493            int nbZones = region->pDimensionDefinitions[dim].zones;
494    
495            int c = 0;
496            if (dimregno >= 0) {
497                int mask = ~(((1 << region->pDimensionDefinitions[dim].bits) - 1) << bitpos);
498                c = dimregno & mask; // mask away this dimension
499            }
500            bool customsplits =
501                ((region->pDimensionDefinitions[dim].split_type == gig::split_type_normal &&
502                  region->pDimensionRegions[c]->DimensionUpperLimits[dim]) ||
503                 (region->pDimensionDefinitions[dim].dimension == gig::dimension_velocity &&
504                  region->pDimensionRegions[c]->VelocityUpperLimit));
505    
506            if (customsplits) {
507                int prev_limit = 0;
508                for (int j = 0 ; j < nbZones - 1 ; j++) {
509                    gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];
510                    int upperLimit = d->DimensionUpperLimits[dim];
511                    if (!upperLimit) upperLimit = d->VelocityUpperLimit;
512                    int limit = upperLimit + 1;
513                    int limitx = int((w - label_width - 1) * limit / 128.0 + 0.5) + label_width;
514    
515                    if (x <= limitx - 2) break;
516                    if (x <= limitx + 2) {
517                        resize.dimension = dim;
518                        resize.offset = j << bitpos;
519                        resize.pos = limit;
520                        resize.min = prev_limit;
521    
522                        int dr = (dimregno >> bitpos) &
523                            ((1 << region->pDimensionDefinitions[dim].bits) - 1);
524                        resize.selected = dr == j ? resize.left :
525                            dr == j + 1 ? resize.right : resize.none;
526    
527                        j++;
528                        gig::DimensionRegion *d = region->pDimensionRegions[c + (j << bitpos)];
529                        int upperLimit = d->DimensionUpperLimits[dim];
530                        if (!upperLimit) upperLimit = d->VelocityUpperLimit;
531                        int limit = upperLimit + 1;
532                        resize.max = limit;
533                        return true;
534                    }
535                    prev_limit = limit;
536                }
537            }
538        }
539        return false;
540    }
541    
542  sigc::signal<void> DimRegionChooser::signal_sel_changed()  sigc::signal<void> DimRegionChooser::signal_sel_changed()
543  {  {
544      return sel_changed_signal;      return sel_changed_signal;

Legend:
Removed from v.1078  
changed lines
  Added in v.1104

  ViewVC Help
Powered by ViewVC