141 lines
3.9 KiB
C++
141 lines
3.9 KiB
C++
|
/*****************************************************************************
|
||
|
*
|
||
|
* This file is part of Mapnik (c++ mapping toolkit)
|
||
|
*
|
||
|
* Copyright (C) 2013 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/text/face.hpp>
|
||
|
#include <mapnik/debug.hpp>
|
||
|
|
||
|
extern "C"
|
||
|
{
|
||
|
#include FT_GLYPH_H
|
||
|
}
|
||
|
|
||
|
namespace mapnik
|
||
|
{
|
||
|
|
||
|
font_face::font_face(FT_Face face)
|
||
|
: face_(face), dimension_cache_(), char_height_(0.0)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
double font_face::get_char_height() const
|
||
|
{
|
||
|
if (char_height_ != 0.0) return char_height_;
|
||
|
glyph_info tmp;
|
||
|
tmp.glyph_index = FT_Get_Char_Index(face_, 'X');
|
||
|
glyph_dimensions(tmp);
|
||
|
char_height_ = tmp.height();
|
||
|
return char_height_;
|
||
|
}
|
||
|
|
||
|
bool font_face::set_character_sizes(double size)
|
||
|
{
|
||
|
char_height_ = 0.0;
|
||
|
return !FT_Set_Char_Size(face_,0,(FT_F26Dot6)(size * (1<<6)),0,0);
|
||
|
}
|
||
|
|
||
|
void font_face::glyph_dimensions(glyph_info & glyph) const
|
||
|
{
|
||
|
//TODO
|
||
|
//Check if char is already in cache
|
||
|
// std::map<glyph_index_t, glyph_info>::const_iterator itr;
|
||
|
// itr = dimension_cache_.find(glyph.glyph_index);
|
||
|
// if (itr != dimension_cache_.end()) {
|
||
|
// glyph = itr->second;
|
||
|
// return;
|
||
|
// }
|
||
|
|
||
|
FT_Vector pen;
|
||
|
pen.x = 0;
|
||
|
pen.y = 0;
|
||
|
/*
|
||
|
FT_Matrix matrix;
|
||
|
matrix.xx = (FT_Fixed)( 1 * 0x10000L );
|
||
|
matrix.xy = (FT_Fixed)( 0 * 0x10000L );
|
||
|
matrix.yx = (FT_Fixed)( 0 * 0x10000L );
|
||
|
matrix.yy = (FT_Fixed)( 1 * 0x10000L );
|
||
|
FT_Set_Transform(face_, &matrix, &pen);
|
||
|
*/
|
||
|
// TODO - any benefit to using a matrix here?
|
||
|
FT_Set_Transform(face_, 0, &pen);
|
||
|
|
||
|
if (FT_Load_Glyph (face_, glyph.glyph_index, FT_LOAD_NO_HINTING)) return;
|
||
|
|
||
|
FT_Glyph image;
|
||
|
if (FT_Get_Glyph(face_->glyph, &image)) return;
|
||
|
FT_BBox glyph_bbox;
|
||
|
FT_Glyph_Get_CBox(image, ft_glyph_bbox_pixels, &glyph_bbox);
|
||
|
FT_Done_Glyph(image);
|
||
|
|
||
|
glyph.ymin = glyph_bbox.yMin; //pixels!
|
||
|
glyph.ymax = glyph_bbox.yMax;
|
||
|
glyph.line_height = face_->size->metrics.height/64.0;
|
||
|
// TODO: we round to integers for now to maintain
|
||
|
// back compatibility with Mapnik 2.x
|
||
|
//glyph.width = face_->glyph->advance.x/64.0;
|
||
|
glyph.width = face_->glyph->advance.x >> 6;
|
||
|
|
||
|
//TODO: dimension_cache_.insert(std::pair<unsigned, char_info>(c, dim));
|
||
|
}
|
||
|
|
||
|
font_face::~font_face()
|
||
|
{
|
||
|
MAPNIK_LOG_DEBUG(font_face) <<
|
||
|
"font_face: Clean up face \"" << family_name() <<
|
||
|
" " << style_name() << "\"";
|
||
|
|
||
|
FT_Done_Face(face_);
|
||
|
}
|
||
|
|
||
|
/******************************************************************************/
|
||
|
|
||
|
void font_face_set::add(face_ptr face)
|
||
|
{
|
||
|
faces_.push_back(face);
|
||
|
}
|
||
|
|
||
|
void font_face_set::set_character_sizes(double size)
|
||
|
{
|
||
|
for (face_ptr const& face : faces_)
|
||
|
{
|
||
|
face->set_character_sizes(size);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/******************************************************************************/
|
||
|
|
||
|
void stroker::init(double radius)
|
||
|
{
|
||
|
FT_Stroker_Set(s_, (FT_Fixed) (radius * (1<<6)),
|
||
|
FT_STROKER_LINECAP_ROUND,
|
||
|
FT_STROKER_LINEJOIN_ROUND,
|
||
|
0);
|
||
|
}
|
||
|
|
||
|
stroker::~stroker()
|
||
|
{
|
||
|
MAPNIK_LOG_DEBUG(font_engine_freetype) << "stroker: Destroy stroker=" << s_;
|
||
|
|
||
|
FT_Stroker_Done(s_);
|
||
|
}
|
||
|
|
||
|
}//ns mapnik
|