parent
0b58b0a94c
commit
25f45461ed
13 changed files with 1970 additions and 379 deletions
@ -0,0 +1,231 @@ |
||||
#include "afglobal.h" |
||||
#include "aflatin.h" |
||||
|
||||
/* populate this list when you add new scripts
|
||||
*/ |
||||
static AF_ScriptClass const af_script_class_list[] = |
||||
{ |
||||
& af_latin_script_class, |
||||
|
||||
NULL /* do not remove */ |
||||
}; |
||||
|
||||
#define AF_SCRIPT_LIST_DEFAULT 0 /* index of default script in 'af_script_class_list' */ |
||||
#define AF_SCRIPT_LIST_NONE 255 /* indicates an uncovered glyph */ |
||||
|
||||
/*
|
||||
* note that glyph_scripts[] is used to map each glyph into |
||||
* an index into the 'af_script_class_list' array. |
||||
* |
||||
*/ |
||||
typedef struct AF_FaceGlobalsRec_ |
||||
{ |
||||
FT_Face face; |
||||
FT_UInt glyph_count; /* same as face->num_glyphs */ |
||||
FT_Byte* glyph_scripts; |
||||
|
||||
AF_ScriptMetrics metrics[ AF_SCRIPT_MAX ]; |
||||
|
||||
} AF_FaceGlobalsRec, *AF_FaceGlobals; |
||||
|
||||
|
||||
|
||||
|
||||
/* this function is used to compute the script index of each glyph
|
||||
* within a given face |
||||
*/ |
||||
static FT_Error |
||||
af_face_globals_compute_script_coverage( AF_FaceGlobals globals ) |
||||
{ |
||||
FT_Error error = 0; |
||||
FT_Face face = globals->face; |
||||
FT_CharMap old_charmap = face->charmap; |
||||
FT_Byte* gscripts = globals->glyph_scripts; |
||||
FT_UInt ss; |
||||
|
||||
/* the value 255 means "uncovered glyph"
|
||||
*/ |
||||
FT_MEM_SET( globals->glyph_scripts, |
||||
AF_SCRIPT_LIST_NONE, |
||||
globals->glyph_count ); |
||||
|
||||
error = FT_Select_Charmap( face, FT_ENCODING_UNICODE ); |
||||
if ( error ) |
||||
{ |
||||
/* ignore this error, we'll simply use Latin as the standard
|
||||
* script. XXX: Shouldn't we rather disable hinting ?? |
||||
*/ |
||||
error = 0; |
||||
goto Exit; |
||||
} |
||||
|
||||
/* scan each script in a Unicode charmap
|
||||
*/ |
||||
for ( ss = 0; af_script_classes[ss]; ss++ ) |
||||
{ |
||||
AF_ScriptClass clazz = af_script_classes[ss]; |
||||
AF_Script_UniRange range; |
||||
|
||||
/* scan all unicode points in the range, and set the corresponding
|
||||
* glyph script index |
||||
*/ |
||||
for ( range = clazz->script_uni_ranges; range->first != 0; range++ ) |
||||
{ |
||||
FT_ULong charcode = range->first; |
||||
FT_UInt gindex; |
||||
|
||||
gindex = FT_Get_Char_Index( face, charcode ); |
||||
|
||||
if ( gindex != 0 && |
||||
gindex < globals->glyph_count && |
||||
gscripts[ gindex ] == AF_SCRIPT_LIST_NONE ) |
||||
{ |
||||
gscripts[ gindex ] = (FT_Byte) ss; |
||||
} |
||||
for (;;) |
||||
{ |
||||
charcode = FT_Get_Next_Char( face, charcode, &gindex ); |
||||
|
||||
if ( gindex == 0 || charcode > range->last ) |
||||
break; |
||||
|
||||
if ( gindex < globals->glyph_count && |
||||
gscripts[ gindex ] == AF_SCRIPT_LIST_NONE ) |
||||
{ |
||||
gscripts[ gindex ] = (FT_Byte) ss; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
Exit: |
||||
/* by default, all uncovered glyphs are set to the latin script
|
||||
* XXX: shouldnt' we disable hinting or do something similar ? |
||||
*/ |
||||
{ |
||||
FT_UInt nn; |
||||
|
||||
for ( nn = 0; nn < globals->glyph_count; nn++ ) |
||||
{ |
||||
if ( gscripts[ nn ] == AF_SCRIPT_LIST_NONE ) |
||||
gscripts[ nn ] = AF_SCRIPT_LIST_DEFAULT; |
||||
} |
||||
} |
||||
|
||||
FT_Set_Charmap( face, old_charmap ); |
||||
return error; |
||||
} |
||||
|
||||
|
||||
|
||||
FT_LOCAL_DEF( FT_Error ) |
||||
af_face_globals_new( FT_Face face, |
||||
AF_FaceGlobals *aglobals ) |
||||
{ |
||||
FT_Error error; |
||||
FT_Memory memory; |
||||
AF_FaceGlobals globals; |
||||
|
||||
memory = face->memory; |
||||
|
||||
if ( !FT_ALLOC( globals, sizeof(*globals) + |
||||
face->num_glyphs*sizeof(FT_Byte) ) ) |
||||
{ |
||||
globals->face = face; |
||||
globals->glyph_count = face->num_glyphs; |
||||
globals->glyph_scripts = (FT_Byte*)( globals+1 ); |
||||
|
||||
error = af_face_globals_compute_script_coverage( globals ); |
||||
if ( error ) |
||||
{ |
||||
af_face_globals_free( globals ); |
||||
globals = NULL; |
||||
} |
||||
} |
||||
|
||||
Exit: |
||||
*aglobals = globals; |
||||
return error; |
||||
} |
||||
|
||||
|
||||
FT_LOCAL_DEF( void ) |
||||
af_face_globals_free( AF_FaceGlobals globals ) |
||||
{ |
||||
if ( globals ) |
||||
{ |
||||
FT_Memory memory = globals->face->memory; |
||||
FT_UInt nn; |
||||
|
||||
for ( nn = 0; nn < AF_SCRIPT_MAX; nn++ ) |
||||
{ |
||||
if ( globals->metrics[nn] ) |
||||
{ |
||||
AF_ScriptClass clazz = af_script_classes[nn]; |
||||
|
||||
FT_ASSERT( globals->metrics[nn].clazz == clazz ); |
||||
|
||||
if ( clazz->script_metrics_done ) |
||||
clazz->script_metrics_done( globals->metrics[nn] ); |
||||
|
||||
FT_FREE( globals->metrics[nn] ); |
||||
} |
||||
} |
||||
|
||||
globals->glyph_count = NULL; |
||||
globals->glyph_scripts = NULL; /* no need to free this one !! */ |
||||
globals->face = NULL; |
||||
FT_FREE( globals ); |
||||
} |
||||
} |
||||
|
||||
|
||||
FT_LOCAL_DEF( FT_Error ) |
||||
af_face_globals_get_metrics( AF_FaceGlobals globals, |
||||
FT_UInt gindex, |
||||
AF_ScriptMetrics *ametrics ) |
||||
{ |
||||
FT_Script script; |
||||
AF_ScriptMetrics metrics = NULL; |
||||
FT_UInt index; |
||||
AF_ScriptClass clazz; |
||||
FT_Error error; |
||||
|
||||
if ( gindex >= globals->glyph_count ) |
||||
{ |
||||
error = FT_Err_Invalid_Argument; |
||||
goto Exit; |
||||
} |
||||
|
||||
index = globals->glyph_scripts[ gindex ]; |
||||
clazz = af_script_class_list[ index ]; |
||||
metrics = globals->metrics[ clazz->script ]; |
||||
if ( metrics == NULL ) |
||||
{ |
||||
/* create the global metrics object when needed
|
||||
*/ |
||||
if ( FT_ALLOC( metrics, clazz->script_metrics_size ) ) |
||||
goto Exit; |
||||
|
||||
metrics->clazz = clazz; |
||||
|
||||
if ( clazz->script_metrics_init ) |
||||
{ |
||||
error = clazz->script_metrics_init( metrics, face ); |
||||
if ( error ) |
||||
{ |
||||
if ( clazz->script_metrics_done ) |
||||
clazz->script_metrics_done( metrics ); |
||||
|
||||
FT_FREE( metrics ); |
||||
goto Exit; |
||||
} |
||||
} |
||||
|
||||
globals->metrics[ script ] = metrics; |
||||
} |
||||
|
||||
Exit: |
||||
*ametrics = metrics; |
||||
return error; |
||||
} |
@ -0,0 +1,42 @@ |
||||
#ifndef __AF_GLOBAL_H__ |
||||
#define __AF_GLOBAL_H__ |
||||
|
||||
#include "aftypes.h" |
||||
|
||||
FT_BEGIN_HEADER |
||||
|
||||
/**************************************************************************/ |
||||
/**************************************************************************/ |
||||
/***** *****/ |
||||
/***** F A C E G L O B A L S *****/ |
||||
/***** *****/ |
||||
/**************************************************************************/ |
||||
/**************************************************************************/ |
||||
|
||||
|
||||
/*
|
||||
* models the global hints data for a given face, decomposed into |
||||
* script-specific items.. |
||||
* |
||||
*/ |
||||
typedef struct AF_FaceGlobalsRec_* AF_FaceGlobals; |
||||
|
||||
|
||||
FT_LOCAL( FT_Error ) |
||||
af_face_globals_new( FT_Face face, |
||||
AF_FaceGlobals *aglobals ); |
||||
|
||||
|
||||
FT_LOCAL( FT_Error ) |
||||
af_face_globals_get_metrics( AF_FaceGlobals globals, |
||||
FT_UInt gindex, |
||||
AF_ScriptMetrics *ametrics ); |
||||
|
||||
FT_LOCAL( void ) |
||||
af_face_globals_free( AF_FaceGlobals globals ); |
||||
|
||||
/* */ |
||||
|
||||
FT_END_HEADER |
||||
|
||||
#endif /* __AF_GLOBALS_H__ */ |
File diff suppressed because it is too large
Load Diff
Loading…
Reference in new issue