2009-06-20 09:47:10 -04:00
|
|
|
/*
|
2009-10-14 12:10:01 -04:00
|
|
|
Copyright (C) 2009 Paul Davis
|
2009-06-20 09:47:10 -04:00
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
*/
|
2015-05-01 22:54:59 -04:00
|
|
|
#if !defined USE_CAIRO_IMAGE_SURFACE && !defined NDEBUG
|
|
|
|
#define OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
#endif
|
2009-06-20 09:47:10 -04:00
|
|
|
|
2011-12-07 08:57:45 -05:00
|
|
|
#include "gtkmm2ext/cairo_widget.h"
|
|
|
|
#include "gtkmm2ext/gui_thread.h"
|
2009-06-20 09:44:47 -04:00
|
|
|
|
2014-03-06 09:44:33 -05:00
|
|
|
#include "i18n.h"
|
|
|
|
|
2011-11-18 10:35:58 -05:00
|
|
|
static const char* has_cairo_widget_background_info = "has_cairo_widget_background_info";
|
|
|
|
|
2014-08-27 11:55:01 -04:00
|
|
|
bool CairoWidget::_flat_buttons = false;
|
2015-01-27 20:22:52 -05:00
|
|
|
bool CairoWidget::_widget_prelight = true;
|
2014-08-27 11:55:01 -04:00
|
|
|
|
2016-03-15 12:34:26 -04:00
|
|
|
sigc::slot<void,Gtk::Widget*> CairoWidget::focus_handler;
|
2014-09-16 12:42:39 -04:00
|
|
|
|
2014-08-27 11:55:01 -04:00
|
|
|
void CairoWidget::set_source_rgb_a( cairo_t* cr, Gdk::Color col, float a) //ToDo: this one and the Canvas version should be in a shared file (?)
|
|
|
|
{
|
|
|
|
float r = col.get_red_p ();
|
|
|
|
float g = col.get_green_p ();
|
|
|
|
float b = col.get_blue_p ();
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2014-08-27 11:55:01 -04:00
|
|
|
cairo_set_source_rgba(cr, r, g, b, a);
|
|
|
|
}
|
|
|
|
|
2009-06-20 09:44:47 -04:00
|
|
|
CairoWidget::CairoWidget ()
|
2012-02-07 12:43:55 -05:00
|
|
|
: _active_state (Gtkmm2ext::Off)
|
|
|
|
, _visual_state (Gtkmm2ext::NoVisualState)
|
2011-11-18 10:35:58 -05:00
|
|
|
, _need_bg (true)
|
2014-08-21 12:13:23 -04:00
|
|
|
, _grabbed (false)
|
2014-03-06 09:44:33 -05:00
|
|
|
, _name_proxy (this, X_("name"))
|
2014-08-29 19:42:37 -04:00
|
|
|
, _current_parent (0)
|
2009-06-20 09:44:47 -04:00
|
|
|
{
|
2014-03-06 09:44:33 -05:00
|
|
|
_name_proxy.connect (sigc::mem_fun (*this, &CairoWidget::on_name_changed));
|
2009-06-20 09:44:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
CairoWidget::~CairoWidget ()
|
|
|
|
{
|
2014-08-29 19:42:37 -04:00
|
|
|
if (_parent_style_change) _parent_style_change.disconnect();
|
2009-06-20 09:44:47 -04:00
|
|
|
}
|
|
|
|
|
2014-09-16 12:42:39 -04:00
|
|
|
bool
|
|
|
|
CairoWidget::on_button_press_event (GdkEventButton*)
|
|
|
|
{
|
2016-03-15 12:34:26 -04:00
|
|
|
focus_handler (this);
|
2014-09-16 12:42:39 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-05-06 14:06:10 -04:00
|
|
|
|
|
|
|
#ifdef USE_TRACKS_CODE_FEATURES
|
|
|
|
|
|
|
|
/* This is Tracks version of this method.
|
|
|
|
|
|
|
|
The use of get_visible_window() in this method is an abuse of the GDK/GTK
|
|
|
|
semantics. It can and may break on different GDK backends, and uses a
|
|
|
|
side-effect/unintended behaviour in GDK/GTK to try to accomplish something
|
|
|
|
which should be done differently. I (Paul) have confirmed this with the GTK
|
|
|
|
development team.
|
|
|
|
|
|
|
|
For this reason, this code is not acceptable for ordinary merging into the Ardour libraries.
|
|
|
|
|
|
|
|
Ardour Developers: you are not obligated to maintain the internals of this
|
|
|
|
implementation in the face of build-time environment changes (e.g. -D
|
|
|
|
defines etc).
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
|
|
|
CairoWidget::on_expose_event (GdkEventExpose *ev)
|
|
|
|
{
|
|
|
|
cairo_rectangle_t expose_area;
|
|
|
|
expose_area.width = ev->area.width;
|
|
|
|
expose_area.height = ev->area.height;
|
|
|
|
|
|
|
|
#ifdef USE_CAIRO_IMAGE_SURFACE_FOR_CAIRO_WIDGET
|
|
|
|
Cairo::RefPtr<Cairo::Context> cr;
|
|
|
|
if (get_visible_window ()) {
|
|
|
|
expose_area.x = 0;
|
|
|
|
expose_area.y = 0;
|
|
|
|
if (!_image_surface) {
|
|
|
|
_image_surface = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, get_width(), get_height());
|
|
|
|
}
|
|
|
|
cr = Cairo::Context::create (_image_surface);
|
|
|
|
} else {
|
|
|
|
expose_area.x = ev->area.x;
|
|
|
|
expose_area.y = ev->area.y;
|
|
|
|
cr = get_window()->create_cairo_context ();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
expose_area.x = ev->area.x;
|
|
|
|
expose_area.y = ev->area.y;
|
|
|
|
Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context ();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
cr->rectangle (expose_area.x, expose_area.y, expose_area.width, expose_area.height);
|
|
|
|
cr->clip ();
|
|
|
|
|
|
|
|
/* paint expose area the color of the parent window bg
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (get_visible_window ()) {
|
|
|
|
Gdk::Color bg (get_parent_bg());
|
|
|
|
cr->rectangle (expose_area.x, expose_area.y, expose_area.width, expose_area.height);
|
|
|
|
cr->set_source_rgb (bg.get_red_p(), bg.get_green_p(), bg.get_blue_p());
|
|
|
|
cr->fill ();
|
|
|
|
}
|
|
|
|
|
|
|
|
render (cr->cobj(), &expose_area);
|
|
|
|
|
|
|
|
#ifdef USE_CAIRO_IMAGE_SURFACE_FOR_CAIRO_WIDGET
|
|
|
|
if(get_visible_window ()) {
|
|
|
|
_image_surface->flush();
|
|
|
|
/* now blit our private surface back to the GDK one */
|
|
|
|
|
|
|
|
Cairo::RefPtr<Cairo::Context> cairo_context = get_window()->create_cairo_context ();
|
|
|
|
|
|
|
|
cairo_context->rectangle (ev->area.x, ev->area.y, ev->area.width, ev->area.height);
|
|
|
|
cairo_context->clip ();
|
|
|
|
cairo_context->set_source (_image_surface, ev->area.x, ev->area.y);
|
|
|
|
cairo_context->set_operator (Cairo::OPERATOR_OVER);
|
|
|
|
cairo_context->paint ();
|
|
|
|
}
|
|
|
|
#endif
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2015-05-06 14:06:10 -04:00
|
|
|
Gtk::Widget* child = get_child ();
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2015-05-06 14:06:10 -04:00
|
|
|
if (child) {
|
|
|
|
propagate_expose (*child, ev);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
/* Ardour mainline: not using Tracks code features.
|
|
|
|
|
|
|
|
Tracks Developers: please do not modify this version of
|
|
|
|
::on_expose_event(). The version used by Tracks is before the preceding
|
|
|
|
#else and contains hacks required for the Tracks GUI to work.
|
|
|
|
*/
|
|
|
|
|
2009-06-20 09:44:47 -04:00
|
|
|
bool
|
2011-10-27 16:24:29 -04:00
|
|
|
CairoWidget::on_expose_event (GdkEventExpose *ev)
|
2009-06-20 09:44:47 -04:00
|
|
|
{
|
2015-05-01 22:54:59 -04:00
|
|
|
#ifdef OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
Cairo::RefPtr<Cairo::Context> cr;
|
|
|
|
if (getenv("ARDOUR_IMAGE_SURFACE")) {
|
|
|
|
if (!image_surface) {
|
|
|
|
image_surface = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, get_width(), get_height());
|
|
|
|
}
|
|
|
|
cr = Cairo::Context::create (image_surface);
|
|
|
|
} else {
|
|
|
|
cr = get_window()->create_cairo_context ();
|
|
|
|
}
|
|
|
|
#elif defined USE_CAIRO_IMAGE_SURFACE
|
2014-10-27 21:15:10 -04:00
|
|
|
|
|
|
|
if (!image_surface) {
|
|
|
|
image_surface = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, get_width(), get_height());
|
|
|
|
}
|
|
|
|
|
|
|
|
Cairo::RefPtr<Cairo::Context> cr = Cairo::Context::create (image_surface);
|
|
|
|
#else
|
|
|
|
Cairo::RefPtr<Cairo::Context> cr = get_window()->create_cairo_context ();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
cr->rectangle (ev->area.x, ev->area.y, ev->area.width, ev->area.height);
|
2011-11-18 10:35:58 -05:00
|
|
|
|
2015-07-23 17:38:04 -04:00
|
|
|
if (_need_bg) {
|
|
|
|
cr->clip_preserve ();
|
2015-10-26 14:35:06 -04:00
|
|
|
|
2015-07-23 17:38:04 -04:00
|
|
|
/* paint expose area the color of the parent window bg
|
|
|
|
*/
|
2015-10-26 14:35:06 -04:00
|
|
|
|
2015-07-23 17:38:04 -04:00
|
|
|
Gdk::Color bg (get_parent_bg());
|
2015-10-26 14:35:06 -04:00
|
|
|
|
2015-07-23 17:38:04 -04:00
|
|
|
cr->set_source_rgb (bg.get_red_p(), bg.get_green_p(), bg.get_blue_p());
|
|
|
|
cr->fill ();
|
|
|
|
} else {
|
|
|
|
std::cerr << get_name() << " skipped bg fill\n";
|
|
|
|
cr->clip ();
|
|
|
|
}
|
2011-11-18 10:35:58 -05:00
|
|
|
|
2014-03-20 18:50:40 -04:00
|
|
|
cairo_rectangle_t expose_area;
|
|
|
|
expose_area.x = ev->area.x;
|
|
|
|
expose_area.y = ev->area.y;
|
|
|
|
expose_area.width = ev->area.width;
|
|
|
|
expose_area.height = ev->area.height;
|
|
|
|
|
2014-10-27 21:15:10 -04:00
|
|
|
render (cr->cobj(), &expose_area);
|
2011-11-18 10:35:58 -05:00
|
|
|
|
2015-05-01 22:54:59 -04:00
|
|
|
#ifdef OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
if (getenv("ARDOUR_IMAGE_SURFACE")) {
|
|
|
|
#endif
|
|
|
|
#if defined USE_CAIRO_IMAGE_SURFACE || defined OPTIONAL_CAIRO_IMAGE_SURFACE
|
2014-10-27 21:15:10 -04:00
|
|
|
image_surface->flush();
|
|
|
|
/* now blit our private surface back to the GDK one */
|
|
|
|
|
|
|
|
Cairo::RefPtr<Cairo::Context> cairo_context = get_window()->create_cairo_context ();
|
|
|
|
|
|
|
|
cairo_context->rectangle (ev->area.x, ev->area.y, ev->area.width, ev->area.height);
|
|
|
|
cairo_context->clip ();
|
|
|
|
cairo_context->set_source (image_surface, 0, 0);
|
|
|
|
cairo_context->set_operator (Cairo::OPERATOR_SOURCE);
|
|
|
|
cairo_context->paint ();
|
|
|
|
#endif
|
2015-05-01 22:54:59 -04:00
|
|
|
#ifdef OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
}
|
|
|
|
#endif
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2009-06-20 09:44:47 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-06 14:06:10 -04:00
|
|
|
#endif
|
|
|
|
|
2009-06-21 11:56:16 -04:00
|
|
|
/** Marks the widget as dirty, so that render () will be called on
|
|
|
|
* the next GTK expose event.
|
|
|
|
*/
|
|
|
|
|
2009-06-20 09:44:47 -04:00
|
|
|
void
|
2016-02-12 16:57:46 -05:00
|
|
|
CairoWidget::set_dirty (cairo_rectangle_t *area)
|
2009-06-20 09:44:47 -04:00
|
|
|
{
|
2011-11-03 17:35:20 -04:00
|
|
|
ENSURE_GUI_THREAD (*this, &CairoWidget::set_dirty);
|
2016-02-12 16:57:46 -05:00
|
|
|
if (!area) {
|
|
|
|
queue_draw ();
|
|
|
|
} else {
|
|
|
|
queue_draw_area (area->x, area->y, area->width, area->height);
|
|
|
|
}
|
2009-06-20 09:44:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Handle a size allocation.
|
|
|
|
* @param alloc GTK allocation.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
CairoWidget::on_size_allocate (Gtk::Allocation& alloc)
|
|
|
|
{
|
|
|
|
Gtk::EventBox::on_size_allocate (alloc);
|
|
|
|
|
2015-05-01 22:54:59 -04:00
|
|
|
#ifdef OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
if (getenv("ARDOUR_IMAGE_SURFACE")) {
|
|
|
|
#endif
|
|
|
|
#if defined USE_CAIRO_IMAGE_SURFACE || defined OPTIONAL_CAIRO_IMAGE_SURFACE
|
2014-10-27 21:15:10 -04:00
|
|
|
image_surface = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, alloc.get_width(), alloc.get_height());
|
|
|
|
#endif
|
2015-05-01 22:54:59 -04:00
|
|
|
#ifdef OPTIONAL_CAIRO_IMAGE_SURFACE
|
|
|
|
}
|
|
|
|
#endif
|
2014-10-27 21:15:10 -04:00
|
|
|
|
2009-06-20 09:44:47 -04:00
|
|
|
set_dirty ();
|
|
|
|
}
|
2011-10-26 17:01:14 -04:00
|
|
|
|
|
|
|
Gdk::Color
|
|
|
|
CairoWidget::get_parent_bg ()
|
|
|
|
{
|
2014-08-29 19:42:37 -04:00
|
|
|
Widget* parent;
|
2011-10-26 17:01:14 -04:00
|
|
|
|
|
|
|
parent = get_parent ();
|
|
|
|
|
2014-08-29 19:42:37 -04:00
|
|
|
while (parent) {
|
2011-11-18 10:35:58 -05:00
|
|
|
void* p = g_object_get_data (G_OBJECT(parent->gobj()), has_cairo_widget_background_info);
|
|
|
|
|
|
|
|
if (p) {
|
|
|
|
Glib::RefPtr<Gtk::Style> style = parent->get_style();
|
2014-08-29 19:42:37 -04:00
|
|
|
if (_current_parent != parent) {
|
|
|
|
if (_parent_style_change) _parent_style_change.disconnect();
|
|
|
|
_current_parent = parent;
|
|
|
|
_parent_style_change = parent->signal_style_changed().connect (mem_fun (*this, &CairoWidget::on_style_changed));
|
|
|
|
}
|
2011-11-18 10:35:58 -05:00
|
|
|
return style->get_bg (get_state());
|
|
|
|
}
|
2014-08-29 19:42:37 -04:00
|
|
|
|
2011-11-18 10:35:58 -05:00
|
|
|
if (!parent->get_has_window()) {
|
|
|
|
parent = parent->get_parent();
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
2014-08-29 19:42:37 -04:00
|
|
|
}
|
2011-10-26 17:01:14 -04:00
|
|
|
|
2014-08-29 19:42:37 -04:00
|
|
|
if (parent && parent->get_has_window()) {
|
|
|
|
if (_current_parent != parent) {
|
|
|
|
if (_parent_style_change) _parent_style_change.disconnect();
|
|
|
|
_current_parent = parent;
|
|
|
|
_parent_style_change = parent->signal_style_changed().connect (mem_fun (*this, &CairoWidget::on_style_changed));
|
|
|
|
}
|
2011-10-26 17:01:14 -04:00
|
|
|
return parent->get_style ()->get_bg (parent->get_state());
|
2014-08-29 19:42:37 -04:00
|
|
|
}
|
2011-10-26 17:01:14 -04:00
|
|
|
|
|
|
|
return get_style ()->get_bg (get_state());
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-10-31 21:22:08 -04:00
|
|
|
CairoWidget::set_active_state (Gtkmm2ext::ActiveState s)
|
2011-10-26 17:01:14 -04:00
|
|
|
{
|
2011-10-27 07:24:43 -04:00
|
|
|
if (_active_state != s) {
|
|
|
|
_active_state = s;
|
|
|
|
StateChanged ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-10-31 21:22:08 -04:00
|
|
|
CairoWidget::set_visual_state (Gtkmm2ext::VisualState s)
|
2011-10-27 07:24:43 -04:00
|
|
|
{
|
|
|
|
if (_visual_state != s) {
|
|
|
|
_visual_state = s;
|
|
|
|
StateChanged ();
|
2011-10-26 17:01:14 -04:00
|
|
|
}
|
|
|
|
}
|
2011-11-03 10:50:42 -04:00
|
|
|
|
2011-11-17 17:49:13 -05:00
|
|
|
void
|
|
|
|
CairoWidget::set_active (bool yn)
|
|
|
|
{
|
|
|
|
/* this is an API simplification for buttons
|
|
|
|
that only use the Active and Normal states.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (yn) {
|
2012-02-07 12:43:55 -05:00
|
|
|
set_active_state (Gtkmm2ext::ExplicitActive);
|
2011-11-17 17:49:13 -05:00
|
|
|
} else {
|
|
|
|
unset_active_state ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-29 19:42:37 -04:00
|
|
|
void
|
|
|
|
CairoWidget::on_style_changed (const Glib::RefPtr<Gtk::Style>&)
|
|
|
|
{
|
|
|
|
queue_draw();
|
|
|
|
}
|
|
|
|
|
2011-11-03 10:50:42 -04:00
|
|
|
void
|
|
|
|
CairoWidget::on_state_changed (Gtk::StateType)
|
|
|
|
{
|
|
|
|
/* this will catch GTK-level state changes from calls like
|
2014-08-29 19:42:37 -04:00
|
|
|
::set_sensitive()
|
2011-11-03 10:50:42 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
if (get_state() == Gtk::STATE_INSENSITIVE) {
|
|
|
|
set_visual_state (Gtkmm2ext::VisualState (visual_state() | Gtkmm2ext::Insensitive));
|
|
|
|
} else {
|
|
|
|
set_visual_state (Gtkmm2ext::VisualState (visual_state() & ~Gtkmm2ext::Insensitive));
|
|
|
|
}
|
|
|
|
|
|
|
|
queue_draw ();
|
|
|
|
}
|
2011-11-18 10:35:58 -05:00
|
|
|
|
|
|
|
void
|
|
|
|
CairoWidget::set_draw_background (bool yn)
|
|
|
|
{
|
|
|
|
_need_bg = yn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CairoWidget::provide_background_for_cairo_widget (Gtk::Widget& w, const Gdk::Color& bg)
|
|
|
|
{
|
2014-08-29 19:42:37 -04:00
|
|
|
/* set up @w to be able to provide bg information to
|
2011-11-18 10:35:58 -05:00
|
|
|
any CairoWidgets that are packed inside it.
|
|
|
|
*/
|
|
|
|
|
|
|
|
w.modify_bg (Gtk::STATE_NORMAL, bg);
|
|
|
|
w.modify_bg (Gtk::STATE_INSENSITIVE, bg);
|
|
|
|
w.modify_bg (Gtk::STATE_ACTIVE, bg);
|
|
|
|
w.modify_bg (Gtk::STATE_SELECTED, bg);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT(w.gobj()), has_cairo_widget_background_info, (void*) 0xfeedface);
|
|
|
|
}
|
2014-08-27 11:55:01 -04:00
|
|
|
|
|
|
|
void
|
|
|
|
CairoWidget::set_flat_buttons (bool yn)
|
|
|
|
{
|
|
|
|
_flat_buttons = yn;
|
|
|
|
}
|
2014-09-16 12:42:39 -04:00
|
|
|
|
2015-01-27 20:22:52 -05:00
|
|
|
void
|
|
|
|
CairoWidget::set_widget_prelight (bool yn)
|
|
|
|
{
|
|
|
|
_widget_prelight = yn;
|
|
|
|
}
|
|
|
|
|
2014-09-16 12:42:39 -04:00
|
|
|
void
|
2016-03-15 12:34:26 -04:00
|
|
|
CairoWidget::set_focus_handler (sigc::slot<void,Gtk::Widget*> s)
|
2014-09-16 12:42:39 -04:00
|
|
|
{
|
|
|
|
focus_handler = s;
|
|
|
|
}
|