2013-04-15 22:10:18 -04:00
|
|
|
/*
|
|
|
|
Copyright (C) 2011-2013 Paul Davis
|
|
|
|
Author: Carl Hetherington <cth@carlh.net>
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2013-04-09 14:22:58 -04:00
|
|
|
#include <gdk/gdk.h>
|
|
|
|
|
2013-04-06 18:35:32 -04:00
|
|
|
#include <cairomm/cairomm.h>
|
2015-04-29 08:19:22 -04:00
|
|
|
#include <gtkmm/window.h>
|
2013-04-04 00:32:52 -04:00
|
|
|
#include <gtkmm/label.h>
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2013-04-09 14:22:58 -04:00
|
|
|
#include "pbd/stacktrace.h"
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
#include "canvas/text.h"
|
|
|
|
#include "canvas/canvas.h"
|
|
|
|
#include "canvas/utils.h"
|
2014-11-07 14:24:27 -05:00
|
|
|
#include "canvas/colors.h"
|
2013-04-04 00:32:52 -04:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace ArdourCanvas;
|
|
|
|
|
2015-04-29 08:19:22 -04:00
|
|
|
|
2014-06-12 14:53:44 -04:00
|
|
|
Text::Text (Canvas* c)
|
|
|
|
: Item (c)
|
2013-04-04 00:32:52 -04:00
|
|
|
, _color (0x000000ff)
|
2013-04-06 18:35:32 -04:00
|
|
|
, _font_description (0)
|
2013-04-04 00:32:52 -04:00
|
|
|
, _alignment (Pango::ALIGN_LEFT)
|
2013-04-06 18:35:32 -04:00
|
|
|
, _width (0)
|
|
|
|
, _height (0)
|
2013-04-06 19:04:34 -04:00
|
|
|
, _need_redraw (false)
|
2014-09-19 13:17:45 -04:00
|
|
|
, _width_correction (-1)
|
2013-04-15 14:45:22 -04:00
|
|
|
, _clamped_width (COORD_MAX)
|
2013-04-04 00:32:52 -04:00
|
|
|
{
|
2014-06-30 10:39:26 -04:00
|
|
|
_outline = false;
|
2014-06-12 14:53:44 -04:00
|
|
|
}
|
2013-04-04 00:32:52 -04:00
|
|
|
|
2014-06-21 11:43:42 -04:00
|
|
|
Text::Text (Item* parent)
|
|
|
|
: Item (parent)
|
2014-06-12 14:53:44 -04:00
|
|
|
, _color (0x000000ff)
|
|
|
|
, _font_description (0)
|
|
|
|
, _alignment (Pango::ALIGN_LEFT)
|
|
|
|
, _width (0)
|
|
|
|
, _height (0)
|
|
|
|
, _need_redraw (false)
|
2014-09-19 13:17:45 -04:00
|
|
|
, _width_correction (-1)
|
2014-06-12 14:53:44 -04:00
|
|
|
, _clamped_width (COORD_MAX)
|
|
|
|
{
|
2014-06-30 10:39:26 -04:00
|
|
|
_outline = false;
|
2013-04-04 00:32:52 -04:00
|
|
|
}
|
|
|
|
|
2013-04-04 18:45:27 -04:00
|
|
|
Text::~Text ()
|
|
|
|
{
|
|
|
|
delete _font_description;
|
|
|
|
}
|
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
void
|
|
|
|
Text::set (string const & text)
|
|
|
|
{
|
|
|
|
begin_change ();
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
_text = text;
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2013-04-06 19:04:34 -04:00
|
|
|
_need_redraw = true;
|
2013-04-04 00:32:52 -04:00
|
|
|
_bounding_box_dirty = true;
|
2013-04-06 19:04:34 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
end_change ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-09-06 10:39:08 -04:00
|
|
|
Text::_redraw () const
|
2013-04-04 00:32:52 -04:00
|
|
|
{
|
2015-09-06 10:39:08 -04:00
|
|
|
assert (!_text.empty());
|
|
|
|
Glib::RefPtr<Pango::Context> context = Glib::wrap (gdk_pango_context_get());
|
2013-04-04 00:32:52 -04:00
|
|
|
Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create (context);
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2014-09-19 13:17:45 -04:00
|
|
|
#ifdef __APPLE__
|
2015-06-13 16:05:29 -04:00
|
|
|
if (_width_correction < 0.0) {
|
|
|
|
// Pango returns incorrect text width on some OS X
|
|
|
|
// So we have to make a correction
|
|
|
|
// To determine the correct indent take the largest symbol for which the width is correct
|
|
|
|
// and make the calculation
|
|
|
|
Gtk::Window win;
|
|
|
|
Gtk::Label foo;
|
|
|
|
win.add (foo);
|
2015-09-06 10:39:08 -04:00
|
|
|
win.ensure_style ();
|
2015-06-13 16:05:29 -04:00
|
|
|
|
|
|
|
int width = 0;
|
|
|
|
int height = 0;
|
|
|
|
Glib::RefPtr<Pango::Layout> test_layout = foo.create_pango_layout ("H");
|
2015-06-13 16:04:31 -04:00
|
|
|
if (_font_description) {
|
|
|
|
test_layout->set_font_description (*_font_description);
|
|
|
|
}
|
2015-06-13 16:05:29 -04:00
|
|
|
test_layout->get_pixel_size (width, height);
|
|
|
|
|
|
|
|
_width_correction = width*1.5;
|
|
|
|
}
|
2014-09-19 13:17:45 -04:00
|
|
|
#else
|
|
|
|
/* don't bother with a conditional here */
|
|
|
|
_width_correction = 0.0;
|
|
|
|
#endif
|
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
layout->set_text (_text);
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
if (_font_description) {
|
|
|
|
layout->set_font_description (*_font_description);
|
|
|
|
}
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
layout->set_alignment (_alignment);
|
2015-10-04 14:51:05 -04:00
|
|
|
|
2013-10-24 17:12:28 -04:00
|
|
|
int w;
|
|
|
|
int h;
|
2013-04-08 19:48:09 -04:00
|
|
|
|
2014-06-17 11:34:47 -04:00
|
|
|
layout->get_pixel_size (w, h);
|
|
|
|
|
2015-04-29 08:19:22 -04:00
|
|
|
_width = w + _width_correction;
|
2014-06-17 11:34:47 -04:00
|
|
|
_height = h;
|
|
|
|
|
2015-06-16 07:51:50 -04:00
|
|
|
#ifdef __APPLE__
|
|
|
|
_image = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, _width * 2, _height * 2);
|
|
|
|
#else
|
2013-04-06 19:04:34 -04:00
|
|
|
_image = Cairo::ImageSurface::create (Cairo::FORMAT_ARGB32, _width, _height);
|
2015-06-16 07:51:50 -04:00
|
|
|
#endif
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-06 19:04:34 -04:00
|
|
|
Cairo::RefPtr<Cairo::Context> img_context = Cairo::Context::create (_image);
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2015-06-16 07:51:50 -04:00
|
|
|
#ifdef __APPLE__
|
|
|
|
/* Below, the rendering scaling is set to support retina display
|
|
|
|
*/
|
|
|
|
img_context->scale (2, 2);
|
|
|
|
#endif
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-06 19:04:34 -04:00
|
|
|
/* and draw, in the appropriate color of course */
|
2013-04-06 18:35:32 -04:00
|
|
|
|
2014-06-30 10:39:26 -04:00
|
|
|
if (_outline) {
|
|
|
|
set_source_rgba (img_context, _outline_color);
|
|
|
|
layout->update_from_cairo_context (img_context);
|
|
|
|
pango_cairo_layout_path (img_context->cobj(), layout->gobj());
|
|
|
|
img_context->stroke_preserve ();
|
|
|
|
set_source_rgba (img_context, _color);
|
|
|
|
img_context->fill ();
|
|
|
|
} else {
|
|
|
|
set_source_rgba (img_context, _color);
|
|
|
|
layout->show_in_cairo_context (img_context);
|
|
|
|
}
|
2013-04-06 19:04:34 -04:00
|
|
|
|
|
|
|
/* text has now been rendered in _image and is ready for blit in
|
2015-10-04 14:51:05 -04:00
|
|
|
* ::render
|
2013-04-06 19:04:34 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
_need_redraw = false;
|
2013-04-06 18:35:32 -04:00
|
|
|
}
|
|
|
|
|
2013-06-18 13:46:24 -04:00
|
|
|
void
|
2014-06-17 11:34:47 -04:00
|
|
|
Text::render (Rect const & area, Cairo::RefPtr<Cairo::Context> context) const
|
2013-06-18 13:46:24 -04:00
|
|
|
{
|
|
|
|
if (_text.empty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-06-17 11:34:47 -04:00
|
|
|
Rect self = item_to_window (Rect (0, 0, min (_clamped_width, (double)_image->get_width ()), _image->get_height ()));
|
|
|
|
boost::optional<Rect> i = self.intersection (area);
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2014-06-17 11:34:47 -04:00
|
|
|
if (!i) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-18 13:46:24 -04:00
|
|
|
if (_need_redraw) {
|
2015-09-06 10:39:08 -04:00
|
|
|
_redraw ();
|
2013-06-18 13:46:24 -04:00
|
|
|
}
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2014-06-17 11:34:47 -04:00
|
|
|
Rect intersection (i.get());
|
|
|
|
|
|
|
|
context->rectangle (intersection.x0, intersection.y0, intersection.width(), intersection.height());
|
2015-06-16 07:51:50 -04:00
|
|
|
#ifdef __APPLE__
|
|
|
|
/* Below, the rendering scaling is set to support retina display
|
|
|
|
*/
|
|
|
|
Cairo::Matrix original_matrix = context->get_matrix();
|
|
|
|
context->scale (0.5, 0.5);
|
|
|
|
context->set_source (_image, self.x0 * 2, self.y0 * 2);
|
|
|
|
context->fill ();
|
|
|
|
context->set_matrix (original_matrix);
|
|
|
|
#else
|
2013-06-26 17:55:42 -04:00
|
|
|
context->set_source (_image, self.x0, self.y0);
|
2013-06-18 13:46:24 -04:00
|
|
|
context->fill ();
|
2015-06-16 07:51:50 -04:00
|
|
|
#endif
|
2013-06-18 13:46:24 -04:00
|
|
|
}
|
|
|
|
|
2013-04-15 14:45:22 -04:00
|
|
|
void
|
|
|
|
Text::clamp_width (double w)
|
|
|
|
{
|
2015-09-06 10:39:08 -04:00
|
|
|
if (_clamped_width == w) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
begin_change ();
|
2013-04-15 14:45:22 -04:00
|
|
|
_clamped_width = w;
|
2015-09-06 10:39:08 -04:00
|
|
|
_bounding_box_dirty = true;
|
|
|
|
end_change ();
|
2013-04-15 14:45:22 -04:00
|
|
|
}
|
|
|
|
|
2013-04-06 18:35:32 -04:00
|
|
|
void
|
|
|
|
Text::compute_bounding_box () const
|
|
|
|
{
|
2013-04-09 14:22:58 -04:00
|
|
|
if (!_canvas || _text.empty()) {
|
2013-04-08 19:48:09 -04:00
|
|
|
_bounding_box = boost::optional<Rect> ();
|
|
|
|
_bounding_box_dirty = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-26 17:55:42 -04:00
|
|
|
if (_bounding_box_dirty) {
|
2015-09-06 10:39:08 -04:00
|
|
|
#ifdef __APPLE__
|
|
|
|
const float retina_factor = 0.5;
|
|
|
|
#else
|
|
|
|
const float retina_factor = 1.0;
|
|
|
|
#endif
|
2013-10-24 17:12:28 -04:00
|
|
|
if (_need_redraw || !_image) {
|
2015-09-06 10:39:08 -04:00
|
|
|
_redraw ();
|
2013-06-26 17:55:42 -04:00
|
|
|
}
|
2015-09-06 10:39:08 -04:00
|
|
|
_bounding_box = Rect (0, 0, min (_clamped_width, (double) _image->get_width() * retina_factor), _image->get_height() * retina_factor);
|
2013-06-26 17:55:42 -04:00
|
|
|
_bounding_box_dirty = false;
|
2013-04-16 10:07:52 -04:00
|
|
|
}
|
2013-04-04 00:32:52 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Text::set_alignment (Pango::Alignment alignment)
|
|
|
|
{
|
|
|
|
begin_change ();
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-04 00:32:52 -04:00
|
|
|
_alignment = alignment;
|
2013-04-06 19:04:34 -04:00
|
|
|
_need_redraw = true;
|
2013-04-04 00:32:52 -04:00
|
|
|
_bounding_box_dirty = true;
|
|
|
|
end_change ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-04-04 18:45:27 -04:00
|
|
|
Text::set_font_description (Pango::FontDescription font_description)
|
2013-04-04 00:32:52 -04:00
|
|
|
{
|
|
|
|
begin_change ();
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-04 18:45:27 -04:00
|
|
|
_font_description = new Pango::FontDescription (font_description);
|
2013-04-06 19:04:34 -04:00
|
|
|
_need_redraw = true;
|
2014-09-19 13:17:45 -04:00
|
|
|
_width_correction = -1.0;
|
2013-04-04 00:32:52 -04:00
|
|
|
|
|
|
|
_bounding_box_dirty = true;
|
|
|
|
end_change ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Text::set_color (Color color)
|
|
|
|
{
|
|
|
|
begin_change ();
|
|
|
|
|
|
|
|
_color = color;
|
2014-06-30 10:39:26 -04:00
|
|
|
if (_outline) {
|
|
|
|
set_outline_color (contrasting_text_color (_color));
|
|
|
|
}
|
2013-04-06 19:04:34 -04:00
|
|
|
_need_redraw = true;
|
2013-04-04 00:32:52 -04:00
|
|
|
|
|
|
|
end_change ();
|
|
|
|
}
|
|
|
|
|
2015-10-05 10:17:49 -04:00
|
|
|
|
2013-04-08 19:48:09 -04:00
|
|
|
void
|
|
|
|
Text::dump (ostream& o) const
|
|
|
|
{
|
|
|
|
Item::dump (o);
|
|
|
|
|
|
|
|
o << _canvas->indent() << '\t' << " text = " << _text << endl
|
|
|
|
<< _canvas->indent() << " color = " << _color;
|
|
|
|
|
|
|
|
o << endl;
|
|
|
|
}
|
2014-12-15 06:17:44 -05:00
|
|
|
|
|
|
|
|
|
|
|
double
|
|
|
|
Text::text_width() const
|
|
|
|
{
|
|
|
|
if (_need_redraw) {
|
|
|
|
redraw ();
|
|
|
|
}
|
2015-10-04 14:51:05 -04:00
|
|
|
|
2014-12-15 06:17:44 -05:00
|
|
|
return _width;
|
|
|
|
}
|