2014-06-03 23:57:12 -04:00
|
|
|
/*
|
|
|
|
Copyright (C) 2014 Paul Davis
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <cairomm/context.h>
|
|
|
|
#include "pbd/stacktrace.h"
|
|
|
|
#include "pbd/compose.h"
|
|
|
|
|
|
|
|
#include "canvas/canvas.h"
|
|
|
|
#include "canvas/widget.h"
|
|
|
|
#include "canvas/debug.h"
|
|
|
|
#include "canvas/utils.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace ArdourCanvas;
|
|
|
|
|
2014-06-12 14:53:44 -04:00
|
|
|
Widget::Widget (Canvas* c, CairoWidget& w)
|
|
|
|
: Item (c)
|
|
|
|
, _widget (w)
|
|
|
|
{
|
|
|
|
Event.connect (sigc::mem_fun (*this, &Widget::event_proxy));
|
2016-12-19 07:36:42 -05:00
|
|
|
w.set_canvas_widget ();
|
|
|
|
w.QueueDraw.connect (sigc::mem_fun(*this, &Widget::queue_draw));
|
|
|
|
w.QueueResize.connect (sigc::mem_fun(*this, &Widget::queue_resize));
|
2014-06-12 14:53:44 -04:00
|
|
|
}
|
|
|
|
|
2014-06-21 11:43:42 -04:00
|
|
|
Widget::Widget (Item* parent, CairoWidget& w)
|
|
|
|
: Item (parent)
|
2014-06-03 23:57:12 -04:00
|
|
|
, _widget (w)
|
|
|
|
{
|
|
|
|
Event.connect (sigc::mem_fun (*this, &Widget::event_proxy));
|
2016-12-19 07:36:42 -05:00
|
|
|
w.set_canvas_widget ();
|
|
|
|
w.QueueDraw.connect (sigc::mem_fun(*this, &Widget::queue_draw));
|
|
|
|
w.QueueResize.connect (sigc::mem_fun(*this, &Widget::queue_resize));
|
2014-06-03 23:57:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Widget::event_proxy (GdkEvent* ev)
|
|
|
|
{
|
2014-06-12 14:53:44 -04:00
|
|
|
/* XXX need to translate coordinate into widget's own coordinate space */
|
2014-06-03 23:57:12 -04:00
|
|
|
return _widget.event (ev);
|
|
|
|
}
|
|
|
|
|
2016-12-19 07:36:42 -05:00
|
|
|
bool
|
|
|
|
Widget::queue_draw ()
|
|
|
|
{
|
|
|
|
begin_visual_change ();
|
|
|
|
end_visual_change ();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Widget::queue_resize ()
|
|
|
|
{
|
|
|
|
begin_change ();
|
|
|
|
end_change ();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-06-03 23:57:12 -04:00
|
|
|
void
|
|
|
|
Widget::render (Rect const & area, Cairo::RefPtr<Cairo::Context> context) const
|
|
|
|
{
|
2017-01-18 18:23:10 -05:00
|
|
|
//std::cerr << "Render widget " << name << " @ " << position() << endl;
|
2014-06-03 23:57:12 -04:00
|
|
|
|
|
|
|
if (!_bounding_box) {
|
|
|
|
std::cerr << "no bbox\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-19 14:54:24 -05:00
|
|
|
Rect self = item_to_window (_bounding_box);
|
|
|
|
Rect r = self.intersection (area);
|
2014-06-03 23:57:12 -04:00
|
|
|
|
|
|
|
if (!r) {
|
|
|
|
std::cerr << "no intersection\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-19 14:54:24 -05:00
|
|
|
Rect draw = r;
|
2014-06-03 23:57:12 -04:00
|
|
|
cairo_rectangle_t crect;
|
|
|
|
crect.x = draw.x0;
|
|
|
|
crect.y = draw.y0;
|
|
|
|
crect.height = draw.height();
|
|
|
|
crect.width = draw.width();
|
|
|
|
|
2017-01-18 18:23:10 -05:00
|
|
|
Duple p = position_offset();
|
|
|
|
|
2014-06-03 23:57:12 -04:00
|
|
|
context->save ();
|
2017-01-18 18:23:10 -05:00
|
|
|
context->translate (p.x, p.y);
|
2014-06-03 23:57:12 -04:00
|
|
|
//context->rectangle (draw.x0, draw.y0, draw.width(), draw.height());
|
2017-01-18 18:23:10 -05:00
|
|
|
//context->clip ();
|
2014-06-03 23:57:12 -04:00
|
|
|
|
2017-03-20 12:11:56 -04:00
|
|
|
_widget.render (context, &crect);
|
2014-06-03 23:57:12 -04:00
|
|
|
|
|
|
|
context->restore ();
|
|
|
|
}
|
|
|
|
|
2017-01-30 10:31:35 -05:00
|
|
|
void
|
|
|
|
Widget::size_allocate (Rect const & r)
|
|
|
|
{
|
|
|
|
Item::size_allocate (r);
|
|
|
|
Gtk::Allocation alloc;
|
|
|
|
alloc.set_x (0);
|
|
|
|
alloc.set_y (0);
|
|
|
|
alloc.set_width (r.width());
|
|
|
|
alloc.set_height (r.height());
|
|
|
|
_widget.size_allocate (alloc);
|
|
|
|
}
|
|
|
|
|
2014-06-03 23:57:12 -04:00
|
|
|
void
|
|
|
|
Widget::compute_bounding_box () const
|
|
|
|
{
|
|
|
|
std::cerr << "cbbox for widget\n";
|
|
|
|
|
2014-11-18 02:37:30 -05:00
|
|
|
GtkRequisition req = { 0, 0 };
|
2014-06-03 23:57:12 -04:00
|
|
|
Gtk::Allocation alloc;
|
|
|
|
|
|
|
|
_widget.size_request (req);
|
|
|
|
|
|
|
|
std::cerr << "widget wants " << req.width << " x " << req.height << "\n";
|
|
|
|
|
|
|
|
_bounding_box = Rect (0, 0, req.width, req.height);
|
|
|
|
|
|
|
|
/* make sure the widget knows that it got what it asked for */
|
|
|
|
alloc.set_x (0);
|
|
|
|
alloc.set_y (0);
|
|
|
|
alloc.set_width (req.width);
|
|
|
|
alloc.set_height (req.height);
|
|
|
|
|
|
|
|
_widget.size_allocate (alloc);
|
|
|
|
|
|
|
|
_bounding_box_dirty = false;
|
|
|
|
}
|