summary | shortlog | log | commit | commitdiff | tree
raw | patch | inline | side by side (parent: 71160df)
raw | patch | inline | side by side (parent: 71160df)
author | Jon A. Cruz <jon@joncruz.org> | |
Sat, 17 Apr 2010 22:48:56 +0000 (15:48 -0700) | ||
committer | Jon A. Cruz <jon@joncruz.org> | |
Sat, 17 Apr 2010 22:48:56 +0000 (15:48 -0700) |
src/libnrtype/FontFactory.cpp | patch | blob | history | |
src/libnrtype/FontInstance.cpp | patch | blob | history | |
src/libnrtype/Layout-TNG-Output.cpp | patch | blob | history |
index 1e4bb3285e1858085d457e6d50510d7726ea564c..83b9bf9d19c5b3232df4c45824981a07a0ed27a8 100644 (file)
g_free(tc);
} else {
loadedFaces.erase(loadedFaces.find(who->descr));
- // printf("unrefFace %p: success\n",who);
+ // printf("unrefFace %p: success\n",who);
}
}
}
for (int i = 0;i < nbEnt;i++) ents[i].age *= 0.9;
for (int i = 0;i < nbEnt;i++) {
if ( ents[i].f == who ) {
- // printf("present\n");
+ // printf("present\n");
ents[i].age += 1.0;
return;
}
index 5a9aad206b4457aae91144bb45884efc20bc3f19..1b02ace23aa2cd0553fda9d330aa04b77d3dd76d 100644 (file)
@@ -39,57 +39,59 @@ struct font_style_equal : public std::binary_function<font_style, font_style, bo
typedef INK_UNORDERED_MAP<font_style, raster_font*, font_style_hash, font_style_equal> StyleMap;
-
-
-size_t font_style_hash::operator()(const font_style &x) const {
- int h=0,n;
- n=(int)floor(100*x.stroke_width);
- h*=12186;
- h+=n;
- n=(x.vertical)?1:0;
- h*=12186;
- h+=n;
- if ( x.stroke_width >= 0.01 ) {
- n=x.stroke_cap*10+x.stroke_join+(int)(x.stroke_miter_limit*100);
- h*=12186;
- h+=n;
- if ( x.nbDash > 0 ) {
- n=x.nbDash;
- h*=12186;
- h+=n;
- n=(int)floor(100*x.dash_offset);
- h*=12186;
- h+=n;
- for (int i=0;i<x.nbDash;i++) {
- n=(int)floor(100*x.dashes[i]);
- h*=12186;
- h+=n;
- }
- }
- }
- return h;
+static const double STROKE_WIDTH_THREASHOLD = 0.01;
+
+
+
+size_t font_style_hash::operator()(const font_style &x) const {
+ int h = 0;
+ int n = static_cast<int>(floor(100 * x.stroke_width));
+ h *= 12186;
+ h += n;
+ n = (x.vertical) ? 1:0;
+ h *= 12186;
+ h += n;
+ if ( x.stroke_width >= STROKE_WIDTH_THREASHOLD ) {
+ n = x.stroke_cap * 10 + x.stroke_join + static_cast<int>(x.stroke_miter_limit * 100);
+ h *= 12186;
+ h += n;
+ if ( x.nbDash > 0 ) {
+ n = x.nbDash;
+ h *= 12186;
+ h += n;
+ n = static_cast<int>(floor(100 * x.dash_offset));
+ h *= 12186;
+ h += n;
+ for (int i = 0; i < x.nbDash; i++) {
+ n = static_cast<int>(floor(100 * x.dashes[i]));
+ h *= 12186;
+ h += n;
+ }
+ }
+ }
+ return h;
}
-bool font_style_equal::operator()(const font_style &a,const font_style &b) const {
- for (int i=0;i<6;i++) {
- if ( (int)(100*a.transform[i]) != (int)(100*b.transform[i]) ) return false;
- }
- if ( a.vertical && b.vertical == false ) return false;
- if ( a.vertical == false && b.vertical ) return false;
- if ( a.stroke_width > 0.01 && b.stroke_width <= 0.01 ) return false;
- if ( a.stroke_width <= 0.01 && b.stroke_width > 0.01 ) return false;
- if ( a.stroke_width <= 0.01 && b.stroke_width <= 0.01 ) return true;
-
- if ( a.stroke_cap != b.stroke_cap ) return false;
- if ( a.stroke_join != b.stroke_join ) return false;
- if ( (int)(a.stroke_miter_limit*100) != (int)(b.stroke_miter_limit*100) ) return false;
- if ( a.nbDash != b.nbDash ) return false;
- if ( a.nbDash <= 0 ) return true;
- if ( (int)floor(100*a.dash_offset) != (int)floor(100*b.dash_offset) ) return false;
- for (int i=0;i<a.nbDash;i++) {
- if ( (int)floor(100*a.dashes[i]) != (int)floor(100*b.dashes[i]) ) return false;
- }
- return true;
+bool font_style_equal::operator()(const font_style &a,const font_style &b) const {
+ bool same = true;
+ for (int i = 0; (i < 6) && same; i++) {
+ same = ( static_cast<int>(100 * a.transform[i]) == static_cast<int>(100 * b.transform[i]) );
+ }
+ same &= ( a.vertical == b.vertical )
+ && ( a.stroke_width > STROKE_WIDTH_THREASHOLD ) == ( b.stroke_width > STROKE_WIDTH_THREASHOLD );
+ if ( same && ( a.stroke_width > STROKE_WIDTH_THREASHOLD ) ) {
+ same = ( a.stroke_cap == b.stroke_cap )
+ && ( a.stroke_join == b.stroke_join )
+ && ( static_cast<int>(a.stroke_miter_limit * 100) == static_cast<int>(b.stroke_miter_limit * 100) )
+ && ( a.nbDash == b.nbDash );
+ if ( same && ( a.nbDash > 0 ) ) {
+ same = ( static_cast<int>(floor(100 * a.dash_offset)) == static_cast<int>(floor(100 * b.dash_offset)) );
+ for (int i = 0; (i < a.nbDash) && same; i++) {
+ same = ( static_cast<int>(floor(100 * a.dashes[i])) == static_cast<int>(floor(100 * b.dashes[i])) );
+ }
+ }
+ }
+ return same;
}
#ifndef USE_PANGO_WIN32
* Outline extraction
*/
typedef struct ft2_to_liv {
- Path* theP;
- double scale;
- Geom::Point last;
+ Path* theP;
+ double scale;
+ Geom::Point last;
} ft2_to_liv;
// Note: Freetype 2.2.1 redefined function signatures for functions to be placed in an
// outline as returned by freetype -> livarot Path
// see nr-type-ft2.cpp for the freetype -> artBPath on which this code is based
static int ft2_move_to(FREETYPE_VECTOR *to, void * i_user) {
- ft2_to_liv* user=(ft2_to_liv*)i_user;
- Geom::Point p(user->scale*to->x,user->scale*to->y);
- // printf("m t=%f %f\n",p[0],p[1]);
- user->theP->MoveTo(p);
- user->last=p;
- return 0;
+ ft2_to_liv* user=(ft2_to_liv*)i_user;
+ Geom::Point p(user->scale*to->x,user->scale*to->y);
+ // printf("m t=%f %f\n",p[0],p[1]);
+ user->theP->MoveTo(p);
+ user->last=p;
+ return 0;
}
static int ft2_line_to(FREETYPE_VECTOR *to, void *i_user)
{
- ft2_to_liv* user=(ft2_to_liv*)i_user;
- Geom::Point p(user->scale*to->x,user->scale*to->y);
- // printf("l t=%f %f\n",p[0],p[1]);
- user->theP->LineTo(p);
- user->last=p;
- return 0;
+ ft2_to_liv* user=(ft2_to_liv*)i_user;
+ Geom::Point p(user->scale*to->x,user->scale*to->y);
+ // printf("l t=%f %f\n",p[0],p[1]);
+ user->theP->LineTo(p);
+ user->last=p;
+ return 0;
}
static int ft2_conic_to(FREETYPE_VECTOR *control, FREETYPE_VECTOR *to, void *i_user)
{
- ft2_to_liv* user=(ft2_to_liv*)i_user;
- Geom::Point p(user->scale*to->x,user->scale*to->y),c(user->scale*control->x,user->scale*control->y);
- // printf("b c=%f %f t=%f %f\n",c[0],c[1],p[0],p[1]);
- user->theP->BezierTo(p);
- user->theP->IntermBezierTo(c);
- user->theP->EndBezierTo();
- user->last=p;
- return 0;
+ ft2_to_liv* user=(ft2_to_liv*)i_user;
+ Geom::Point p(user->scale*to->x,user->scale*to->y),c(user->scale*control->x,user->scale*control->y);
+ // printf("b c=%f %f t=%f %f\n",c[0],c[1],p[0],p[1]);
+ user->theP->BezierTo(p);
+ user->theP->IntermBezierTo(c);
+ user->theP->EndBezierTo();
+ user->last=p;
+ return 0;
}
static int ft2_cubic_to(FREETYPE_VECTOR *control1, FREETYPE_VECTOR *control2, FREETYPE_VECTOR *to, void *i_user)
{
- ft2_to_liv* user=(ft2_to_liv*)i_user;
- Geom::Point p(user->scale*to->x,user->scale*to->y),
- c1(user->scale*control1->x,user->scale*control1->y),
- c2(user->scale*control2->x,user->scale*control2->y);
- // printf("c c1=%f %f c2=%f %f t=%f %f\n",c1[0],c1[1],c2[0],c2[1],p[0],p[1]);
- user->theP->CubicTo(p,3*(c1-user->last),3*(p-c2));
- user->last=p;
- return 0;
+ ft2_to_liv* user=(ft2_to_liv*)i_user;
+ Geom::Point p(user->scale*to->x,user->scale*to->y);
+ Geom::Point c1(user->scale*control1->x,user->scale*control1->y);
+ Geom::Point c2(user->scale*control2->x,user->scale*control2->y);
+ // printf("c c1=%f %f c2=%f %f t=%f %f\n",c1[0],c1[1],c2[0],c2[1],p[0],p[1]);
+ user->theP->CubicTo(p,3*(c1-user->last),3*(p-c2));
+ user->last=p;
+ return 0;
}
#endif
descr = 0;
}
- // if ( theFace ) FT_Done_Face(theFace); // owned by pFont. don't touch
+ // if ( theFace ) FT_Done_Face(theFace); // owned by pFont. don't touch
theFace = 0;
for (int i=0;i<nbGlyph;i++) {
void font_instance::Ref(void)
{
- refCount++;
- //char *tc=pango_font_description_to_string(descr);
- //printf("font %x %s ref'd %i\n",this,tc,refCount);
- //free(tc);
+ refCount++;
+ //char *tc=pango_font_description_to_string(descr);
+ //printf("font %x %s ref'd %i\n",this,tc,refCount);
+ //free(tc);
}
void font_instance::Unref(void)
{
- refCount--;
- //char *tc=pango_font_description_to_string(descr);
- //printf("font %x %s unref'd %i\n",this,tc,refCount);
- //free(tc);
- if ( refCount <= 0 ) {
- if ( daddy ) daddy->UnrefFace(this);
- daddy=NULL;
- delete this;
- }
+ refCount--;
+ //char *tc=pango_font_description_to_string(descr);
+ //printf("font %x %s unref'd %i\n",this,tc,refCount);
+ //free(tc);
+ if ( refCount <= 0 ) {
+ if ( daddy ) {
+ daddy->UnrefFace(this);
+ }
+ daddy=NULL;
+ delete this;
+ }
}
unsigned int font_instance::Name(gchar *str, unsigned int size)
{
- return Attribute("name", str, size);
+ return Attribute("name", str, size);
}
unsigned int font_instance::Family(gchar *str, unsigned int size)
{
- return Attribute("family", str, size);
+ return Attribute("family", str, size);
}
unsigned int font_instance::PSName(gchar *str, unsigned int size)
{
- return Attribute("psname", str, size);
+ return Attribute("psname", str, size);
}
unsigned int font_instance::Attribute(const gchar *key, gchar *str, unsigned int size)
{
- if ( descr == NULL ) {
- if ( size > 0 ) str[0]=0;
- return 0;
- }
- char* res=NULL;
- bool free_res=false;
-
- if ( strcmp(key,"name") == 0 ) {
- PangoFontDescription* td=pango_font_description_copy(descr);
- pango_font_description_unset_fields (td, PANGO_FONT_MASK_SIZE);
- res=pango_font_description_to_string (td);
- pango_font_description_free(td);
- free_res=true;
- } else if ( strcmp(key,"psname") == 0 ) {
+ if ( descr == NULL ) {
+ if ( size > 0 ) {
+ str[0]=0;
+ }
+ return 0;
+ }
+ char* res=NULL;
+ bool free_res=false;
+
+ if ( strcmp(key,"name") == 0 ) {
+ PangoFontDescription* td=pango_font_description_copy(descr);
+ pango_font_description_unset_fields (td, PANGO_FONT_MASK_SIZE);
+ res=pango_font_description_to_string (td);
+ pango_font_description_free(td);
+ free_res=true;
+ } else if ( strcmp(key,"psname") == 0 ) {
#ifndef USE_PANGO_WIN32
res = (char *) FT_Get_Postscript_Name (theFace); // that's the main method, seems to always work
#endif
@@ -292,89 +298,97 @@ unsigned int font_instance::Attribute(const gchar *key, gchar *str, unsigned int
(i) ? "Italic" : ((o) ? "Oblique" : "") );
free_res = true;
}
- } else if ( strcmp(key,"family") == 0 ) {
- res=(char*)pango_font_description_get_family(descr);
- free_res=false;
- } else if ( strcmp(key,"style") == 0 ) {
- PangoStyle v=pango_font_description_get_style(descr);
- if ( v == PANGO_STYLE_ITALIC ) {
- res=(char*)"italic";
- } else if ( v == PANGO_STYLE_OBLIQUE ) {
- res=(char*)"oblique";
- } else {
- res=(char*)"normal";
- }
- free_res=false;
- } else if ( strcmp(key,"weight") == 0 ) {
- PangoWeight v=pango_font_description_get_weight(descr);
- if ( v <= PANGO_WEIGHT_THIN ) {
- res=(char*)"100";
- } else if ( v <= PANGO_WEIGHT_ULTRALIGHT ) {
- res=(char*)"200";
- } else if ( v <= PANGO_WEIGHT_LIGHT ) {
- res=(char*)"300";
- } else if ( v <= PANGO_WEIGHT_BOOK ) {
- res=(char*)"380";
- } else if ( v <= PANGO_WEIGHT_NORMAL ) {
- res=(char*)"normal";
- } else if ( v <= PANGO_WEIGHT_MEDIUM ) {
- res=(char*)"500";
- } else if ( v <= PANGO_WEIGHT_SEMIBOLD ) {
- res=(char*)"600";
- } else if ( v <= PANGO_WEIGHT_BOLD ) {
- res=(char*)"bold";
- } else if ( v <= PANGO_WEIGHT_ULTRABOLD ) {
- res=(char*)"800";
- } else { // HEAVY NB: Pango defines ULTRAHEAVY = 1000 but not CSS2
- res=(char*)"900";
- }
- free_res=false;
- } else if ( strcmp(key,"stretch") == 0 ) {
- PangoStretch v=pango_font_description_get_stretch(descr);
- if ( v <= PANGO_STRETCH_EXTRA_CONDENSED ) {
- res=(char*)"extra-condensed";
- } else if ( v <= PANGO_STRETCH_CONDENSED ) {
- res=(char*)"condensed";
- } else if ( v <= PANGO_STRETCH_SEMI_CONDENSED ) {
- res=(char*)"semi-condensed";
- } else if ( v <= PANGO_STRETCH_NORMAL ) {
- res=(char*)"normal";
- } else if ( v <= PANGO_STRETCH_SEMI_EXPANDED ) {
- res=(char*)"semi-expanded";
- } else if ( v <= PANGO_STRETCH_EXPANDED ) {
- res=(char*)"expanded";
- } else {
- res=(char*)"extra-expanded";
- }
- free_res=false;
- } else if ( strcmp(key,"variant") == 0 ) {
- PangoVariant v=pango_font_description_get_variant(descr);
- if ( v == PANGO_VARIANT_SMALL_CAPS ) {
- res=(char*)"small-caps";
- } else {
- res=(char*)"normal";
- }
- free_res=false;
- } else {
- res = NULL;
- free_res=false;
- }
- if ( res == NULL ) {
- if ( size > 0 ) str[0]=0;
- return 0;
- }
-
- if (res) {
- unsigned int len=strlen(res);
- unsigned int rlen=(size-1<len)?size-1:len;
- if ( str ) {
- if ( rlen > 0 ) memcpy(str,res,rlen);
- if ( size > 0 ) str[rlen]=0;
- }
- if (free_res) free(res);
- return len;
- }
- return 0;
+ } else if ( strcmp(key,"family") == 0 ) {
+ res=(char*)pango_font_description_get_family(descr);
+ free_res=false;
+ } else if ( strcmp(key,"style") == 0 ) {
+ PangoStyle v=pango_font_description_get_style(descr);
+ if ( v == PANGO_STYLE_ITALIC ) {
+ res=(char*)"italic";
+ } else if ( v == PANGO_STYLE_OBLIQUE ) {
+ res=(char*)"oblique";
+ } else {
+ res=(char*)"normal";
+ }
+ free_res=false;
+ } else if ( strcmp(key,"weight") == 0 ) {
+ PangoWeight v=pango_font_description_get_weight(descr);
+ if ( v <= PANGO_WEIGHT_THIN ) {
+ res=(char*)"100";
+ } else if ( v <= PANGO_WEIGHT_ULTRALIGHT ) {
+ res=(char*)"200";
+ } else if ( v <= PANGO_WEIGHT_LIGHT ) {
+ res=(char*)"300";
+ } else if ( v <= PANGO_WEIGHT_BOOK ) {
+ res=(char*)"380";
+ } else if ( v <= PANGO_WEIGHT_NORMAL ) {
+ res=(char*)"normal";
+ } else if ( v <= PANGO_WEIGHT_MEDIUM ) {
+ res=(char*)"500";
+ } else if ( v <= PANGO_WEIGHT_SEMIBOLD ) {
+ res=(char*)"600";
+ } else if ( v <= PANGO_WEIGHT_BOLD ) {
+ res=(char*)"bold";
+ } else if ( v <= PANGO_WEIGHT_ULTRABOLD ) {
+ res=(char*)"800";
+ } else { // HEAVY NB: Pango defines ULTRAHEAVY = 1000 but not CSS2
+ res=(char*)"900";
+ }
+ free_res=false;
+ } else if ( strcmp(key,"stretch") == 0 ) {
+ PangoStretch v=pango_font_description_get_stretch(descr);
+ if ( v <= PANGO_STRETCH_EXTRA_CONDENSED ) {
+ res=(char*)"extra-condensed";
+ } else if ( v <= PANGO_STRETCH_CONDENSED ) {
+ res=(char*)"condensed";
+ } else if ( v <= PANGO_STRETCH_SEMI_CONDENSED ) {
+ res=(char*)"semi-condensed";
+ } else if ( v <= PANGO_STRETCH_NORMAL ) {
+ res=(char*)"normal";
+ } else if ( v <= PANGO_STRETCH_SEMI_EXPANDED ) {
+ res=(char*)"semi-expanded";
+ } else if ( v <= PANGO_STRETCH_EXPANDED ) {
+ res=(char*)"expanded";
+ } else {
+ res=(char*)"extra-expanded";
+ }
+ free_res=false;
+ } else if ( strcmp(key,"variant") == 0 ) {
+ PangoVariant v=pango_font_description_get_variant(descr);
+ if ( v == PANGO_VARIANT_SMALL_CAPS ) {
+ res=(char*)"small-caps";
+ } else {
+ res=(char*)"normal";
+ }
+ free_res=false;
+ } else {
+ res = NULL;
+ free_res=false;
+ }
+ if ( res == NULL ) {
+ if ( size > 0 ) {
+ str[0] = 0;
+ }
+ return 0;
+ }
+
+ if (res) {
+ unsigned int len=strlen(res);
+ unsigned int rlen=(size-1<len)?size-1:len;
+ if ( str ) {
+ if ( rlen > 0 ) {
+ memcpy(str, res, rlen);
+ }
+ if ( size > 0 ) {
+ str[rlen] = 0;
+ }
+ }
+ if (free_res) {
+ free(res);
+ }
+ return len;
+ }
+ return 0;
}
void font_instance::InitTheFace()
SetGraphicsMode(daddy->hScreenDC, GM_COMPATIBLE);
SelectObject(daddy->hScreenDC,theFace);
#else
- theFace=pango_ft2_font_get_face(pFont);
- if ( theFace )
+ theFace=pango_ft2_font_get_face(pFont);
+ if ( theFace ) {
FT_Select_Charmap(theFace,ft_encoding_unicode) && FT_Select_Charmap(theFace,ft_encoding_symbol);
+ }
#endif
}
void font_instance::InstallFace(PangoFont* iFace)
{
- if ( !iFace )
- return;
- pFont=iFace;
+ if ( !iFace ) {
+ return;
+ }
+ pFont=iFace;
InitTheFace();
- if ( pFont && IsOutlineFont() == false ) {
+ if ( pFont && IsOutlineFont() == false ) {
FreeTheFace();
- if ( pFont ) g_object_unref(pFont);
- pFont=NULL;
- }
+ if ( pFont ) {
+ g_object_unref(pFont);
+ }
+ pFont=NULL;
+ }
}
-bool font_instance::IsOutlineFont(void)
+bool font_instance::IsOutlineFont(void)
{
- if ( pFont == NULL ) return false;
+ if ( pFont == NULL ) {
+ return false;
+ }
InitTheFace();
#ifdef USE_PANGO_WIN32
TEXTMETRIC tm;
return GetTextMetrics(daddy->hScreenDC,&tm) && tm.tmPitchAndFamily&(TMPF_TRUETYPE|TMPF_DEVICE);
#else
- return FT_IS_SCALABLE(theFace);
+ return FT_IS_SCALABLE(theFace);
#endif
}
int font_instance::MapUnicodeChar(gunichar c)
{
- if ( pFont == NULL ) return 0;
+ int res = 0;
+ if ( pFont ) {
#ifdef USE_PANGO_WIN32
- return pango_win32_font_get_glyph_index(pFont,c);
+ res = pango_win32_font_get_glyph_index(pFont, c);
#else
- int res=0;
- theFace=pango_ft2_font_get_face(pFont);
- if ( c > 0xf0000 ) {
- res=CLAMP(c,0xf0000,0x1fffff)-0xf0000;
- } else {
- res=FT_Get_Char_Index(theFace, c);
- }
- return res;
+ theFace = pango_ft2_font_get_face(pFont);
+ if ( c > 0xf0000 ) {
+ res = CLAMP(c, 0xf0000, 0x1fffff) - 0xf0000;
+ } else {
+ res = FT_Get_Char_Index(theFace, c);
+ }
+ }
#endif
+ return res;
}
void font_instance::LoadGlyph(int glyph_id)
{
- if ( pFont == NULL ) return;
+ if ( pFont == NULL ) {
+ return;
+ }
InitTheFace();
#ifndef USE_PANGO_WIN32
- if ( !FT_IS_SCALABLE(theFace) ) return; // bitmap font
+ if ( !FT_IS_SCALABLE(theFace) ) {
+ return; // bitmap font
+ }
#endif
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- if ( nbGlyph >= maxGlyph ) {
- maxGlyph=2*nbGlyph+1;
- glyphs=(font_glyph*)realloc(glyphs,maxGlyph*sizeof(font_glyph));
- }
- font_glyph n_g;
- n_g.outline=NULL;
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ if ( nbGlyph >= maxGlyph ) {
+ maxGlyph=2*nbGlyph+1;
+ glyphs=(font_glyph*)realloc(glyphs,maxGlyph*sizeof(font_glyph));
+ }
+ font_glyph n_g;
+ n_g.outline=NULL;
n_g.pathvector=NULL;
- n_g.bbox[0]=n_g.bbox[1]=n_g.bbox[2]=n_g.bbox[3]=0;
- bool doAdd=false;
+ n_g.bbox[0]=n_g.bbox[1]=n_g.bbox[2]=n_g.bbox[3]=0;
+ bool doAdd=false;
#ifdef USE_PANGO_WIN32
Geom::Point this_mid=pointfx_to_nrpoint(p[0], scale);
while ( p != endp ) {
Geom::Point next_mid=pointfx_to_nrpoint(p[1], scale);
- n_g.outline->BezierTo((next_mid+this_mid)/2);
- n_g.outline->IntermBezierTo(this_mid);
- n_g.outline->EndBezierTo();
+ n_g.outline->BezierTo((next_mid+this_mid)/2);
+ n_g.outline->IntermBezierTo(this_mid);
+ n_g.outline->EndBezierTo();
++p;
this_mid=next_mid;
}
- n_g.outline->BezierTo(pointfx_to_nrpoint(p[1], scale));
- n_g.outline->IntermBezierTo(this_mid);
- n_g.outline->EndBezierTo();
+ n_g.outline->BezierTo(pointfx_to_nrpoint(p[1], scale));
+ n_g.outline->IntermBezierTo(this_mid);
+ n_g.outline->EndBezierTo();
break;
}
delete [] buffer;
}
#else
- if (FT_Load_Glyph (theFace, glyph_id, FT_LOAD_NO_SCALE | FT_LOAD_NO_HINTING | FT_LOAD_NO_BITMAP)) {
- // shit happened
- } else {
- if ( FT_HAS_HORIZONTAL(theFace) ) {
- n_g.h_advance=((double)theFace->glyph->metrics.horiAdvance)/((double)theFace->units_per_EM);
- n_g.h_width=((double)theFace->glyph->metrics.width)/((double)theFace->units_per_EM);
- } else {
- n_g.h_width=n_g.h_advance=((double)(theFace->bbox.xMax-theFace->bbox.xMin))/((double)theFace->units_per_EM);
- }
- if ( FT_HAS_VERTICAL(theFace) ) {
- n_g.v_advance=((double)theFace->glyph->metrics.vertAdvance)/((double)theFace->units_per_EM);
- n_g.v_width=((double)theFace->glyph->metrics.height)/((double)theFace->units_per_EM);
- } else {
- n_g.v_width=n_g.v_advance=((double)theFace->height)/((double)theFace->units_per_EM);
- }
- if ( theFace->glyph->format == ft_glyph_format_outline ) {
- FT_Outline_Funcs ft2_outline_funcs = {
- ft2_move_to,
- ft2_line_to,
- ft2_conic_to,
- ft2_cubic_to,
- 0, 0
- };
- n_g.outline=new Path;
- ft2_to_liv tData;
- tData.theP=n_g.outline;
- tData.scale=1.0/((double)theFace->units_per_EM);
- tData.last=Geom::Point(0,0);
- FT_Outline_Decompose (&theFace->glyph->outline, &ft2_outline_funcs, &tData);
- }
- doAdd=true;
- }
+ if (FT_Load_Glyph (theFace, glyph_id, FT_LOAD_NO_SCALE | FT_LOAD_NO_HINTING | FT_LOAD_NO_BITMAP)) {
+ // shit happened
+ } else {
+ if ( FT_HAS_HORIZONTAL(theFace) ) {
+ n_g.h_advance=((double)theFace->glyph->metrics.horiAdvance)/((double)theFace->units_per_EM);
+ n_g.h_width=((double)theFace->glyph->metrics.width)/((double)theFace->units_per_EM);
+ } else {
+ n_g.h_width=n_g.h_advance=((double)(theFace->bbox.xMax-theFace->bbox.xMin))/((double)theFace->units_per_EM);
+ }
+ if ( FT_HAS_VERTICAL(theFace) ) {
+ n_g.v_advance=((double)theFace->glyph->metrics.vertAdvance)/((double)theFace->units_per_EM);
+ n_g.v_width=((double)theFace->glyph->metrics.height)/((double)theFace->units_per_EM);
+ } else {
+ n_g.v_width=n_g.v_advance=((double)theFace->height)/((double)theFace->units_per_EM);
+ }
+ if ( theFace->glyph->format == ft_glyph_format_outline ) {
+ FT_Outline_Funcs ft2_outline_funcs = {
+ ft2_move_to,
+ ft2_line_to,
+ ft2_conic_to,
+ ft2_cubic_to,
+ 0, 0
+ };
+ n_g.outline=new Path;
+ ft2_to_liv tData;
+ tData.theP=n_g.outline;
+ tData.scale=1.0/((double)theFace->units_per_EM);
+ tData.last=Geom::Point(0,0);
+ FT_Outline_Decompose (&theFace->glyph->outline, &ft2_outline_funcs, &tData);
+ }
+ doAdd=true;
+ }
#endif
- if ( doAdd ) {
- if ( n_g.outline ) {
- n_g.outline->FastBBox(n_g.bbox[0],n_g.bbox[1],n_g.bbox[2],n_g.bbox[3]);
+ if ( doAdd ) {
+ if ( n_g.outline ) {
+ n_g.outline->FastBBox(n_g.bbox[0],n_g.bbox[1],n_g.bbox[2],n_g.bbox[3]);
n_g.pathvector=n_g.outline->MakePathVector();
- }
- glyphs[nbGlyph]=n_g;
- id_to_no[glyph_id]=nbGlyph;
- nbGlyph++;
- }
+ }
+ glyphs[nbGlyph]=n_g;
+ id_to_no[glyph_id]=nbGlyph;
+ nbGlyph++;
+ }
} else {
}
}
bool font_instance::FontMetrics(double &ascent,double &descent,double &leading)
{
- if ( pFont == NULL ) return false;
+ if ( pFont == NULL ) {
+ return false;
+ }
InitTheFace();
- if ( theFace == NULL ) return false;
+ if ( theFace == NULL ) {
+ return false;
+ }
#ifdef USE_PANGO_WIN32
OUTLINETEXTMETRIC otm;
- if ( !GetOutlineTextMetrics(daddy->hScreenDC,sizeof(otm),&otm) ) return false;
+ if ( !GetOutlineTextMetrics(daddy->hScreenDC,sizeof(otm),&otm) ) {
+ return false;
+ }
double scale=1.0/daddy->fontSize;
ascent=fabs(otm.otmAscent*scale);
descent=fabs(otm.otmDescent*scale);
leading=fabs(otm.otmLineGap*scale);
#else
- if ( theFace->units_per_EM == 0 ) return false; // bitmap font
- ascent=fabs(((double)theFace->ascender)/((double)theFace->units_per_EM));
- descent=fabs(((double)theFace->descender)/((double)theFace->units_per_EM));
- leading=fabs(((double)theFace->height)/((double)theFace->units_per_EM));
- leading-=ascent+descent;
+ if ( theFace->units_per_EM == 0 ) {
+ return false; // bitmap font
+ }
+ ascent=fabs(((double)theFace->ascender)/((double)theFace->units_per_EM));
+ descent=fabs(((double)theFace->descender)/((double)theFace->units_per_EM));
+ leading=fabs(((double)theFace->height)/((double)theFace->units_per_EM));
+ leading-=ascent+descent;
#endif
- return true;
+ return true;
}
bool font_instance::FontSlope(double &run, double &rise)
run = 0.0;
rise = 1.0;
- if ( pFont == NULL ) return false;
+ if ( pFont == NULL ) {
+ return false;
+ }
InitTheFace();
- if ( theFace == NULL ) return false;
+ if ( theFace == NULL ) {
+ return false;
+ }
#ifdef USE_PANGO_WIN32
OUTLINETEXTMETRIC otm;
run=otm.otmsCharSlopeRun;
rise=otm.otmsCharSlopeRise;
#else
- if ( !FT_IS_SCALABLE(theFace) ) return false; // bitmap font
+ if ( !FT_IS_SCALABLE(theFace) ) {
+ return false; // bitmap font
+ }
TT_HoriHeader *hhea = (TT_HoriHeader*)FT_Get_Sfnt_Table(theFace, ft_sfnt_hhea);
- if (hhea == NULL) return false;
+ if (hhea == NULL) {
+ return false;
+ }
run = hhea->caret_Slope_Run;
rise = hhea->caret_Slope_Rise;
#endif
- return true;
+ return true;
}
Geom::OptRect font_instance::BBox(int glyph_id)
{
- int no=-1;
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- LoadGlyph(glyph_id);
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- // didn't load
- } else {
- no=id_to_no[glyph_id];
- }
- } else {
- no=id_to_no[glyph_id];
- }
- if ( no < 0 ) {
- return Geom::OptRect();
+ int no = -1;
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ LoadGlyph(glyph_id);
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ // didn't load
} else {
- Geom::Point rmin(glyphs[no].bbox[0],glyphs[no].bbox[1]);
- Geom::Point rmax(glyphs[no].bbox[2],glyphs[no].bbox[3]);
- return Geom::Rect(rmin, rmax);
+ no = id_to_no[glyph_id];
}
+ } else {
+ no = id_to_no[glyph_id];
+ }
+ if ( no < 0 ) {
+ return Geom::OptRect();
+ } else {
+ Geom::Point rmin(glyphs[no].bbox[0],glyphs[no].bbox[1]);
+ Geom::Point rmax(glyphs[no].bbox[2],glyphs[no].bbox[3]);
+ return Geom::Rect(rmin, rmax);
+ }
}
Path* font_instance::Outline(int glyph_id,Path* copyInto)
{
- int no=-1;
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- LoadGlyph(glyph_id);
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- // didn't load
- } else {
- no=id_to_no[glyph_id];
- }
- } else {
- no=id_to_no[glyph_id];
- }
- if ( no < 0 ) return NULL;
- Path* src_o=glyphs[no].outline;
- if ( copyInto ) {
- copyInto->Reset();
- copyInto->Copy(src_o);
- return copyInto;
- }
- return src_o;
+ int no = -1;
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ LoadGlyph(glyph_id);
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ // didn't load
+ } else {
+ no = id_to_no[glyph_id];
+ }
+ } else {
+ no = id_to_no[glyph_id];
+ }
+ if ( no < 0 ) return NULL;
+ Path *src_o = glyphs[no].outline;
+ if ( copyInto ) {
+ copyInto->Reset();
+ copyInto->Copy(src_o);
+ return copyInto;
+ }
+ return src_o;
}
Geom::PathVector* font_instance::PathVector(int glyph_id)
double font_instance::Advance(int glyph_id,bool vertical)
{
- int no=-1;
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- LoadGlyph(glyph_id);
- if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
- // didn't load
- } else {
- no=id_to_no[glyph_id];
- }
- } else {
- no=id_to_no[glyph_id];
- }
- if ( no >= 0 ) {
- if ( vertical ) {
- return glyphs[no].v_advance;
- } else {
- return glyphs[no].h_advance;
- }
- }
- return 0;
+ int no = -1;
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ LoadGlyph(glyph_id);
+ if ( id_to_no.find(glyph_id) == id_to_no.end() ) {
+ // didn't load
+ } else {
+ no=id_to_no[glyph_id];
+ }
+ } else {
+ no = id_to_no[glyph_id];
+ }
+ if ( no >= 0 ) {
+ if ( vertical ) {
+ return glyphs[no].v_advance;
+ } else {
+ return glyphs[no].h_advance;
+ }
+ }
+ return 0;
}
raster_font* font_instance::RasterFont(const Geom::Matrix &trs, double stroke_width, bool vertical, JoinType stroke_join, ButtType stroke_cap, float /*miter_limit*/)
{
- font_style nStyle;
- nStyle.transform=trs;
- nStyle.vertical=vertical;
- nStyle.stroke_width=stroke_width;
- nStyle.stroke_cap=stroke_cap;
- nStyle.stroke_join=stroke_join;
- nStyle.nbDash=0;
- nStyle.dash_offset=0;
- nStyle.dashes=NULL;
- return RasterFont(nStyle);
+ font_style nStyle;
+ nStyle.transform=trs;
+ nStyle.vertical=vertical;
+ nStyle.stroke_width=stroke_width;
+ nStyle.stroke_cap=stroke_cap;
+ nStyle.stroke_join=stroke_join;
+ nStyle.nbDash=0;
+ nStyle.dash_offset=0;
+ nStyle.dashes=NULL;
+ return RasterFont(nStyle);
}
raster_font* font_instance::RasterFont(const font_style &inStyle)
{
- raster_font *res=NULL;
- double *savDashes=NULL;
- font_style nStyle=inStyle;
+ raster_font *res=NULL;
+ double *savDashes=NULL;
+ font_style nStyle=inStyle;
// for some evil reason font_style doesn't have a copy ctor, so the
// stuff that should be done there is done here instead (because the
// raster_font ctor copies nStyle).
- if ( nStyle.stroke_width > 0 && nStyle.nbDash > 0 && nStyle.dashes ) {
- savDashes=nStyle.dashes;
- nStyle.dashes=(double*)malloc(nStyle.nbDash*sizeof(double));
- memcpy(nStyle.dashes,savDashes,nStyle.nbDash*sizeof(double));
- }
- StyleMap& loadedStyles = *static_cast<StyleMap*>(loadedPtr);
- if ( loadedStyles.find(nStyle) == loadedStyles.end() ) {
- raster_font *nR = new raster_font(nStyle);
- nR->Ref();
- nR->daddy=this;
- loadedStyles[nStyle]=nR;
- res=nR;
- if ( res ) Ref();
- } else {
- res=loadedStyles[nStyle];
- res->Ref();
- if ( nStyle.dashes ) free(nStyle.dashes); // since they're not taken by a new rasterfont
- }
- nStyle.dashes=savDashes;
- return res;
+ if ( (nStyle.stroke_width > 0) && (nStyle.nbDash > 0) && nStyle.dashes ) {
+ savDashes=nStyle.dashes;
+ nStyle.dashes=(double*)malloc(nStyle.nbDash*sizeof(double));
+ memcpy(nStyle.dashes,savDashes,nStyle.nbDash*sizeof(double));
+ }
+ StyleMap& loadedStyles = *static_cast<StyleMap*>(loadedPtr);
+ if ( loadedStyles.find(nStyle) == loadedStyles.end() ) {
+ raster_font *nR = new raster_font(nStyle);
+ nR->Ref();
+ nR->daddy=this;
+ loadedStyles[nStyle]=nR;
+ res=nR;
+ if ( res ) {
+ Ref();
+ }
+ } else {
+ res=loadedStyles[nStyle];
+ res->Ref();
+ if ( nStyle.dashes ) {
+ free(nStyle.dashes); // since they're not taken by a new rasterfont
+ }
+ }
+ nStyle.dashes=savDashes;
+ return res;
}
void font_instance::RemoveRasterFont(raster_font* who)
index d25eb039d34b4872a73370c606563ecd8bcad222..0d2b7647d9bb1c73b537287d59b82257a357af75 100644 (file)
@@ -117,10 +117,10 @@ void Layout::getBoundingBox(NRRect *bounding_box, Geom::Matrix const &transform,
Geom::Matrix total_transform = glyph_matrix;
total_transform *= transform;
if(_glyphs[glyph_index].span(this).font) {
- Geom::OptRect glyph_rect = _glyphs[glyph_index].span(this).font->BBox(_glyphs[glyph_index].glyph);
+ Geom::OptRect glyph_rect = _glyphs[glyph_index].span(this).font->BBox(_glyphs[glyph_index].glyph);
if (glyph_rect) {
- Geom::Point bmi = glyph_rect->min(), bma = glyph_rect->max();
- Geom::Point tlp(bmi[0],bmi[1]), trp(bma[0],bmi[1]), blp(bmi[0],bma[1]), brp(bma[0],bma[1]);
+ Geom::Point bmi = glyph_rect->min(), bma = glyph_rect->max();
+ Geom::Point tlp(bmi[0],bmi[1]), trp(bma[0],bmi[1]), blp(bmi[0],bma[1]), brp(bma[0],bma[1]);
tlp *= total_transform;
trp *= total_transform;
blp *= total_transform;
@@ -133,7 +133,7 @@ void Layout::getBoundingBox(NRRect *bounding_box, Geom::Matrix const &transform,
if ( (glyph_rect->min())[1] < bounding_box->y0 ) bounding_box->y0=(glyph_rect->min())[1];
if ( (glyph_rect->max())[1] > bounding_box->y1 ) bounding_box->y1=(glyph_rect->max())[1];
}
- }
+ }
}
}