e66d007627
- closes #1298 - allows for much faster halo drawing and simliar quality for radius values in the 1-2 px range - also moves grid_renderer away from using freetype stroker completely since halo quality is not critical for this renderer
499 lines
12 KiB
C++
499 lines
12 KiB
C++
/*****************************************************************************
|
|
*
|
|
* This file is part of Mapnik (c++ mapping toolkit)
|
|
*
|
|
* Copyright (C) 2011 Artem Pavlenko
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*****************************************************************************/
|
|
|
|
//mapnik
|
|
#include <mapnik/debug.hpp>
|
|
#include <mapnik/text_symbolizer.hpp>
|
|
#include <mapnik/enumeration.hpp>
|
|
#include <mapnik/formatting/text.hpp>
|
|
|
|
|
|
// boost
|
|
#include <boost/scoped_ptr.hpp>
|
|
|
|
|
|
namespace mapnik
|
|
{
|
|
|
|
static const char * halo_rasterizer_strings[] = {
|
|
"full",
|
|
"fast",
|
|
""
|
|
};
|
|
|
|
IMPLEMENT_ENUM( halo_rasterizer_e, halo_rasterizer_strings )
|
|
|
|
|
|
static const char * label_placement_strings[] = {
|
|
"point",
|
|
"line",
|
|
"vertex",
|
|
"interior",
|
|
""
|
|
};
|
|
|
|
|
|
IMPLEMENT_ENUM( label_placement_e, label_placement_strings )
|
|
|
|
static const char * vertical_alignment_strings[] = {
|
|
"top",
|
|
"middle",
|
|
"bottom",
|
|
"auto",
|
|
""
|
|
};
|
|
|
|
|
|
IMPLEMENT_ENUM( vertical_alignment_e, vertical_alignment_strings )
|
|
|
|
static const char * horizontal_alignment_strings[] = {
|
|
"left",
|
|
"middle",
|
|
"right",
|
|
"auto",
|
|
""
|
|
};
|
|
|
|
|
|
IMPLEMENT_ENUM( horizontal_alignment_e, horizontal_alignment_strings )
|
|
|
|
static const char * justify_alignment_strings[] = {
|
|
"left",
|
|
"center", // not 'middle' in order to match CSS
|
|
"right",
|
|
"auto",
|
|
""
|
|
};
|
|
|
|
|
|
IMPLEMENT_ENUM( justify_alignment_e, justify_alignment_strings )
|
|
|
|
static const char * text_transform_strings[] = {
|
|
"none",
|
|
"uppercase",
|
|
"lowercase",
|
|
"capitalize",
|
|
""
|
|
};
|
|
|
|
|
|
IMPLEMENT_ENUM( text_transform_e, text_transform_strings )
|
|
|
|
|
|
text_symbolizer::text_symbolizer(text_placements_ptr placements)
|
|
: symbolizer_base(),
|
|
placement_options_(placements),
|
|
halo_rasterizer_(HALO_RASTERIZER_FULL)
|
|
{
|
|
|
|
}
|
|
|
|
text_symbolizer::text_symbolizer(expression_ptr name, std::string const& face_name,
|
|
float size, color const& fill,
|
|
text_placements_ptr placements)
|
|
: symbolizer_base(),
|
|
placement_options_(placements),
|
|
halo_rasterizer_(HALO_RASTERIZER_FULL)
|
|
{
|
|
set_name(name);
|
|
set_face_name(face_name);
|
|
set_text_size(size);
|
|
set_fill(fill);
|
|
}
|
|
|
|
text_symbolizer::text_symbolizer(expression_ptr name, float size, color const& fill,
|
|
text_placements_ptr placements)
|
|
: symbolizer_base(),
|
|
placement_options_(placements),
|
|
halo_rasterizer_(HALO_RASTERIZER_FULL)
|
|
{
|
|
set_name(name);
|
|
set_text_size(size);
|
|
set_fill(fill);
|
|
}
|
|
|
|
text_symbolizer::text_symbolizer(text_symbolizer const& rhs)
|
|
: symbolizer_base(rhs),
|
|
placement_options_(rhs.placement_options_),
|
|
halo_rasterizer_(rhs.halo_rasterizer_)
|
|
/*TODO: Copy options! */
|
|
{
|
|
}
|
|
|
|
text_symbolizer& text_symbolizer::operator=(text_symbolizer const& other)
|
|
{
|
|
if (this == &other)
|
|
return *this;
|
|
placement_options_ = other.placement_options_; /*TODO: Copy options? */
|
|
halo_rasterizer_ = other.halo_rasterizer_;
|
|
return *this;
|
|
}
|
|
|
|
expression_ptr text_symbolizer::get_name() const
|
|
{
|
|
formatting::text_node *node = dynamic_cast<formatting::text_node *>(placement_options_->defaults.format_tree().get());
|
|
if (!node) return expression_ptr();
|
|
return node->get_text();
|
|
}
|
|
|
|
void text_symbolizer::set_name(expression_ptr name)
|
|
{
|
|
placement_options_->defaults.set_old_style_expression(name);
|
|
}
|
|
|
|
expression_ptr text_symbolizer::get_orientation() const
|
|
{
|
|
return placement_options_->defaults.orientation;
|
|
}
|
|
|
|
void text_symbolizer::set_orientation(expression_ptr orientation)
|
|
{
|
|
placement_options_->defaults.orientation = orientation;
|
|
}
|
|
|
|
std::string const& text_symbolizer::get_face_name() const
|
|
{
|
|
return placement_options_->defaults.format.face_name;
|
|
}
|
|
|
|
void text_symbolizer::set_face_name(std::string face_name)
|
|
{
|
|
placement_options_->defaults.format.face_name = face_name;
|
|
}
|
|
|
|
void text_symbolizer::set_fontset(font_set const& fontset)
|
|
{
|
|
placement_options_->defaults.format.fontset = fontset;
|
|
}
|
|
|
|
boost::optional<font_set> const& text_symbolizer::get_fontset() const
|
|
{
|
|
return placement_options_->defaults.format.fontset;
|
|
}
|
|
|
|
double text_symbolizer::get_text_ratio() const
|
|
{
|
|
return placement_options_->defaults.text_ratio;
|
|
}
|
|
|
|
void text_symbolizer::set_text_ratio(double ratio)
|
|
{
|
|
placement_options_->defaults.text_ratio = ratio;
|
|
}
|
|
|
|
double text_symbolizer::get_wrap_width() const
|
|
{
|
|
return placement_options_->defaults.wrap_width;
|
|
}
|
|
|
|
void text_symbolizer::set_wrap_width(double width)
|
|
{
|
|
placement_options_->defaults.wrap_width = width;
|
|
}
|
|
|
|
bool text_symbolizer::get_wrap_before() const
|
|
{
|
|
return placement_options_->defaults.format.wrap_before;
|
|
}
|
|
|
|
void text_symbolizer::set_wrap_before(bool wrap_before)
|
|
{
|
|
placement_options_->defaults.format.wrap_before = wrap_before;
|
|
}
|
|
|
|
unsigned char text_symbolizer::get_wrap_char() const
|
|
{
|
|
return placement_options_->defaults.format.wrap_char;
|
|
}
|
|
|
|
std::string text_symbolizer::get_wrap_char_string() const
|
|
{
|
|
return std::string(1, placement_options_->defaults.format.wrap_char);
|
|
}
|
|
|
|
void text_symbolizer::set_wrap_char(unsigned char character)
|
|
{
|
|
placement_options_->defaults.format.wrap_char = character;
|
|
}
|
|
|
|
void text_symbolizer::set_wrap_char_from_string(std::string const& character)
|
|
{
|
|
placement_options_->defaults.format.wrap_char = (character)[0];
|
|
}
|
|
|
|
text_transform_e text_symbolizer::get_text_transform() const
|
|
{
|
|
return placement_options_->defaults.format.text_transform;
|
|
}
|
|
|
|
void text_symbolizer::set_text_transform(text_transform_e convert)
|
|
{
|
|
placement_options_->defaults.format.text_transform = convert;
|
|
}
|
|
|
|
double text_symbolizer::get_line_spacing() const
|
|
{
|
|
return placement_options_->defaults.format.line_spacing;
|
|
}
|
|
|
|
void text_symbolizer::set_line_spacing(double spacing)
|
|
{
|
|
placement_options_->defaults.format.line_spacing = spacing;
|
|
}
|
|
|
|
double text_symbolizer::get_character_spacing() const
|
|
{
|
|
return placement_options_->defaults.format.character_spacing;
|
|
}
|
|
|
|
void text_symbolizer::set_character_spacing(double spacing)
|
|
{
|
|
placement_options_->defaults.format.character_spacing = spacing;
|
|
}
|
|
|
|
double text_symbolizer::get_label_spacing() const
|
|
{
|
|
return placement_options_->defaults.label_spacing;
|
|
}
|
|
|
|
void text_symbolizer::set_label_spacing(double spacing)
|
|
{
|
|
placement_options_->defaults.label_spacing = spacing;
|
|
}
|
|
|
|
unsigned text_symbolizer::get_label_position_tolerance() const
|
|
{
|
|
return placement_options_->defaults.label_position_tolerance;
|
|
}
|
|
|
|
void text_symbolizer::set_label_position_tolerance(unsigned tolerance)
|
|
{
|
|
placement_options_->defaults.label_position_tolerance = tolerance;
|
|
}
|
|
|
|
bool text_symbolizer::get_force_odd_labels() const
|
|
{
|
|
return placement_options_->defaults.force_odd_labels;
|
|
}
|
|
|
|
void text_symbolizer::set_force_odd_labels(bool force)
|
|
{
|
|
placement_options_->defaults.force_odd_labels = force;
|
|
}
|
|
|
|
double text_symbolizer::get_max_char_angle_delta() const
|
|
{
|
|
return placement_options_->defaults.max_char_angle_delta;
|
|
}
|
|
|
|
void text_symbolizer::set_max_char_angle_delta(double angle)
|
|
{
|
|
placement_options_->defaults.max_char_angle_delta = angle;
|
|
}
|
|
|
|
void text_symbolizer::set_text_size(double size)
|
|
{
|
|
placement_options_->defaults.format.text_size = size;
|
|
}
|
|
|
|
double text_symbolizer::get_text_size() const
|
|
{
|
|
return placement_options_->defaults.format.text_size;
|
|
}
|
|
|
|
void text_symbolizer::set_fill(color const& fill)
|
|
{
|
|
placement_options_->defaults.format.fill = fill;
|
|
}
|
|
|
|
color const& text_symbolizer::get_fill() const
|
|
{
|
|
return placement_options_->defaults.format.fill;
|
|
}
|
|
|
|
void text_symbolizer::set_halo_fill(color const& fill)
|
|
{
|
|
placement_options_->defaults.format.halo_fill = fill;
|
|
}
|
|
|
|
color const& text_symbolizer::get_halo_fill() const
|
|
{
|
|
return placement_options_->defaults.format.halo_fill;
|
|
}
|
|
|
|
void text_symbolizer::set_halo_radius(double radius)
|
|
{
|
|
placement_options_->defaults.format.halo_radius = radius;
|
|
}
|
|
|
|
double text_symbolizer::get_halo_radius() const
|
|
{
|
|
return placement_options_->defaults.format.halo_radius;
|
|
}
|
|
|
|
void text_symbolizer::set_halo_rasterizer(halo_rasterizer_e rasterizer_p)
|
|
{
|
|
halo_rasterizer_ = rasterizer_p;
|
|
}
|
|
|
|
halo_rasterizer_e text_symbolizer::get_halo_rasterizer() const
|
|
{
|
|
return halo_rasterizer_;
|
|
}
|
|
|
|
void text_symbolizer::set_label_placement(label_placement_e label_p)
|
|
{
|
|
placement_options_->defaults.label_placement = label_p;
|
|
}
|
|
|
|
label_placement_e text_symbolizer::get_label_placement() const
|
|
{
|
|
return placement_options_->defaults.label_placement;
|
|
}
|
|
|
|
void text_symbolizer::set_displacement(double x, double y)
|
|
{
|
|
placement_options_->defaults.displacement = std::make_pair(x,y);
|
|
}
|
|
|
|
void text_symbolizer::set_displacement(position const& p)
|
|
{
|
|
placement_options_->defaults.displacement = p;
|
|
}
|
|
|
|
position const& text_symbolizer::get_displacement() const
|
|
{
|
|
return placement_options_->defaults.displacement;
|
|
}
|
|
|
|
bool text_symbolizer::get_avoid_edges() const
|
|
{
|
|
return placement_options_->defaults.avoid_edges;
|
|
}
|
|
|
|
void text_symbolizer::set_avoid_edges(bool avoid)
|
|
{
|
|
placement_options_->defaults.avoid_edges = avoid;
|
|
}
|
|
|
|
bool text_symbolizer::largest_bbox_only() const
|
|
{
|
|
return placement_options_->defaults.largest_bbox_only;
|
|
}
|
|
|
|
void text_symbolizer::set_largest_bbox_only(bool v)
|
|
{
|
|
placement_options_->defaults.largest_bbox_only = v;
|
|
}
|
|
|
|
double text_symbolizer::get_minimum_distance() const
|
|
{
|
|
return placement_options_->defaults.minimum_distance;
|
|
}
|
|
|
|
void text_symbolizer::set_minimum_distance(double distance)
|
|
{
|
|
placement_options_->defaults.minimum_distance = distance;
|
|
}
|
|
|
|
double text_symbolizer::get_minimum_padding() const
|
|
{
|
|
return placement_options_->defaults.minimum_padding;
|
|
}
|
|
|
|
void text_symbolizer::set_minimum_padding(double distance)
|
|
{
|
|
placement_options_->defaults.minimum_padding = distance;
|
|
}
|
|
|
|
double text_symbolizer::get_minimum_path_length() const
|
|
{
|
|
return placement_options_->defaults.minimum_path_length;
|
|
}
|
|
|
|
void text_symbolizer::set_minimum_path_length(double size)
|
|
{
|
|
placement_options_->defaults.minimum_path_length = size;
|
|
}
|
|
|
|
void text_symbolizer::set_allow_overlap(bool overlap)
|
|
{
|
|
placement_options_->defaults.allow_overlap = overlap;
|
|
}
|
|
|
|
bool text_symbolizer::get_allow_overlap() const
|
|
{
|
|
return placement_options_->defaults.allow_overlap;
|
|
}
|
|
|
|
void text_symbolizer::set_text_opacity(double text_opacity)
|
|
{
|
|
placement_options_->defaults.format.text_opacity = text_opacity;
|
|
}
|
|
|
|
double text_symbolizer::get_text_opacity() const
|
|
{
|
|
return placement_options_->defaults.format.text_opacity;
|
|
}
|
|
|
|
void text_symbolizer::set_vertical_alignment(vertical_alignment_e valign)
|
|
{
|
|
placement_options_->defaults.valign = valign;
|
|
}
|
|
|
|
vertical_alignment_e text_symbolizer::get_vertical_alignment() const
|
|
{
|
|
return placement_options_->defaults.valign;
|
|
}
|
|
|
|
void text_symbolizer::set_horizontal_alignment(horizontal_alignment_e halign)
|
|
{
|
|
placement_options_->defaults.halign = halign;
|
|
}
|
|
|
|
horizontal_alignment_e text_symbolizer::get_horizontal_alignment() const
|
|
{
|
|
return placement_options_->defaults.halign;
|
|
}
|
|
|
|
void text_symbolizer::set_justify_alignment(justify_alignment_e jalign)
|
|
{
|
|
placement_options_->defaults.jalign = jalign;
|
|
}
|
|
|
|
justify_alignment_e text_symbolizer::get_justify_alignment() const
|
|
{
|
|
return placement_options_->defaults.jalign;
|
|
}
|
|
|
|
text_placements_ptr text_symbolizer::get_placement_options() const
|
|
{
|
|
return placement_options_;
|
|
}
|
|
|
|
void text_symbolizer::set_placement_options(text_placements_ptr placement_options)
|
|
{
|
|
placement_options_ = placement_options;
|
|
}
|
|
|
|
|
|
}
|