+class CimgPlayheadArea : public CimgArea
+{
+public:
+ CimgPlayheadArea (Cairo::RefPtr<Cairo::ImageSurface> sf, float x0, float w, bool h = false)
+ : CimgArea (sf)
+ , _playhead(-1)
+ , _x0 (x0)
+ , _aw (w)
+ , _highlight (h)
+ {
+ }
+
+ void set_playhead (float pos) {
+ if (rint (_playhead * _aw) == rint (pos * _aw)) {
+ return;
+ }
+ if (_playhead == -1 || pos == -1) {
+ set_dirty ();
+ } else {
+ invalidate (_playhead);
+ invalidate (pos);
+ }
+ _playhead = pos;
+ }
+
+ sigc::signal<void, float> seek_playhead;
+
+protected:
+
+ virtual void overlay (cairo_t* cr, cairo_rectangle_t* r) {
+ if (_playhead > 0 && _playhead < 1.0 && _aw > 0) {
+ if (_highlight) {
+ cairo_rectangle (cr, _x0, 0, _aw, _surface->get_height());
+ cairo_set_source_rgba (cr, .4, .4, .6, .4);
+ cairo_fill (cr);
+ }
+
+ const float x = _playhead * _aw;
+ const float h = _surface->get_height();
+ cairo_set_source_rgba (cr, 1, 0, 0, 1);
+ cairo_set_line_width (cr, 1.5);
+ cairo_move_to (cr, _x0 + x, 0);
+ cairo_line_to (cr, _x0 + x, h);
+ cairo_stroke (cr);
+ }
+ }
+
+ bool on_button_press_event (GdkEventButton *ev) {
+ CairoWidget::on_button_press_event (ev);
+ if (ev->button == 1 && _aw > 0 && ev->x >= _x0 && ev->x <= _x0 + _aw) {
+ seek_playhead (((float) ev->x - _x0) / (float)_aw);
+ }
+ return true;
+ }
+
+private:
+ float _playhead;
+ float _x0, _aw;
+ bool _highlight;
+
+ void invalidate (float pos) {
+ if (pos < 0 || pos > 1) { return; }
+ const float x = pos * _aw;
+ cairo_rectangle_t r;
+ r.y = 0;
+ r.x = _x0 + x - 1;
+ r.width = 3;
+ r.height = _surface->get_height();
+ set_dirty (&r);
+ }
+};
+
+class CimgWaveArea : public CimgPlayheadArea
+{
+public:
+ CimgWaveArea (
+ Cairo::RefPtr<Cairo::ImageSurface> sf,
+ Cairo::RefPtr<Cairo::ImageSurface> sf_log,
+ Cairo::RefPtr<Cairo::ImageSurface> sf_rect,
+ Cairo::RefPtr<Cairo::ImageSurface> sf_logrec,
+ float x0, float w)
+ : CimgPlayheadArea (sf, x0, w)
+ , _sf_log (sf_log)
+ , _sf_rect (sf_rect)
+ , _sf_logrec (sf_logrec)
+ , _logscale (false)
+ , _rectified (false)
+ {
+ }
+
+ void set_logscale (bool en) {
+ _logscale = en;
+ set_dirty ();
+ }
+
+ void set_rectified (bool en) {
+ _rectified = en;
+ set_dirty ();
+ }
+
+protected:
+
+ virtual void background (cairo_t* cr, cairo_rectangle_t* r) {
+ if (_logscale && _rectified) {
+ cairo_set_source_surface (cr, _sf_logrec->cobj(), 0, 0);
+ } else if (_logscale) {
+ cairo_set_source_surface (cr, _sf_log->cobj(), 0, 0);
+ } else if (_rectified) {
+ cairo_set_source_surface (cr, _sf_rect->cobj(), 0, 0);
+ } else {
+ cairo_set_source_surface (cr, _surface->cobj(), 0, 0);
+ }
+ cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
+ cairo_paint (cr);
+ }
+
+private:
+ Cairo::RefPtr<Cairo::ImageSurface> _sf_log;
+ Cairo::RefPtr<Cairo::ImageSurface> _sf_rect;
+ Cairo::RefPtr<Cairo::ImageSurface> _sf_logrec;
+ bool _logscale;
+ bool _rectified;
+};
+