2000-10-12 07:05:40 +02:00
|
|
|
/***************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* ftcimage.c */
|
|
|
|
/* */
|
|
|
|
/* FreeType Image cache (body). */
|
|
|
|
/* */
|
|
|
|
/* Copyright 2000 by */
|
|
|
|
/* David Turner, Robert Wilhelm, and Werner Lemberg. */
|
|
|
|
/* */
|
|
|
|
/* This file is part of the FreeType project, and may only be used, */
|
|
|
|
/* modified, and distributed under the terms of the FreeType project */
|
|
|
|
/* license, LICENSE.TXT. By continuing to use, modify, or distribute */
|
|
|
|
/* this file you indicate that you have read the license and */
|
|
|
|
/* understand and accept it fully. */
|
|
|
|
/* */
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
|
2000-10-29 01:34:45 +02:00
|
|
|
#include <freetype/cache/ftcimage.h>
|
|
|
|
#include <freetype/internal/ftmemory.h>
|
2000-09-19 03:11:11 +02:00
|
|
|
|
2000-10-31 23:13:54 +01:00
|
|
|
#include <string.h> /* memcmp() */
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-10-29 01:34:45 +02:00
|
|
|
|
2000-10-31 23:13:54 +01:00
|
|
|
/* the FT_Glyph image `glyph node' type */
|
2000-10-29 01:34:45 +02:00
|
|
|
typedef struct FTC_GlyphImageRec_
|
|
|
|
{
|
2000-10-31 23:13:54 +01:00
|
|
|
FTC_GlyphNodeRec root;
|
|
|
|
FT_Glyph ft_glyph;
|
2000-10-29 01:34:45 +02:00
|
|
|
|
|
|
|
} FTC_GlyphImageRec, *FTC_GlyphImage;
|
|
|
|
|
|
|
|
|
|
|
|
/* the glyph image queue type */
|
|
|
|
typedef struct FTC_ImageSetRec_
|
|
|
|
{
|
|
|
|
FTC_GlyphSetRec root;
|
|
|
|
FTC_Image_Desc description;
|
2000-10-31 21:42:18 +01:00
|
|
|
|
2000-10-29 01:34:45 +02:00
|
|
|
} FTC_ImageSetRec, *FTC_ImageSet;
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct FTC_Image_CacheRec_
|
|
|
|
{
|
|
|
|
FTC_Glyph_CacheRec root;
|
2000-10-31 21:42:18 +01:00
|
|
|
|
2000-10-29 01:34:45 +02:00
|
|
|
} FTC_Image_CacheRec;
|
|
|
|
|
2000-10-26 12:04:16 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
|
|
|
/***** GLYPH IMAGE NODES *****/
|
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
LOCAL_FUNC_X
|
2000-10-28 15:17:11 +02:00
|
|
|
void ftc_glyph_image_node_destroy( FTC_GlyphImage node,
|
|
|
|
FTC_GlyphSet gset )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_Memory memory = gset->memory;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
2000-10-28 09:26:59 +02:00
|
|
|
FT_Done_Glyph( node->ft_glyph );
|
2000-09-19 03:11:11 +02:00
|
|
|
FREE( node );
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
LOCAL_FUNC_X
|
2000-10-31 23:13:54 +01:00
|
|
|
FT_Error ftc_glyph_image_node_new( FTC_GlyphSet gset,
|
|
|
|
FT_UInt glyph_index,
|
|
|
|
FTC_GlyphImage* anode )
|
2000-10-12 07:05:40 +02:00
|
|
|
{
|
2000-10-31 23:13:54 +01:00
|
|
|
FT_Memory memory = gset->memory;
|
|
|
|
FTC_ImageSet imageset = (FTC_ImageSet)gset;
|
|
|
|
FT_Error error;
|
|
|
|
FTC_GlyphImage node = 0;
|
|
|
|
FT_Face face;
|
|
|
|
FT_Size size;
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* allocate node */
|
2000-10-12 07:05:40 +02:00
|
|
|
if ( ALLOC( node, sizeof ( *node ) ) )
|
2000-09-19 03:11:11 +02:00
|
|
|
goto Exit;
|
|
|
|
|
2000-10-31 23:13:54 +01:00
|
|
|
/* initialize its inner fields */
|
|
|
|
FTC_GlyphNode_Init( FTC_GLYPHNODE( node ), gset, glyph_index );
|
2000-08-29 18:50:01 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* we will now load the glyph image */
|
2000-10-28 15:17:11 +02:00
|
|
|
error = FTC_Manager_Lookup_Size( gset->manager,
|
|
|
|
&imageset->description.font,
|
2000-08-23 13:22:30 +02:00
|
|
|
&face, &size );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( !error )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2000-10-28 09:26:59 +02:00
|
|
|
FT_UInt glyph_index = node->root.glyph_index;
|
2000-08-29 18:04:28 +02:00
|
|
|
FT_UInt load_flags = FT_LOAD_DEFAULT;
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_UInt image_type = imageset->description.image_type;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
if ( FTC_IMAGE_FORMAT( image_type ) == ftc_image_format_bitmap )
|
2000-10-12 07:05:40 +02:00
|
|
|
{
|
|
|
|
load_flags |= FT_LOAD_RENDER;
|
2000-08-29 18:04:28 +02:00
|
|
|
if ( image_type & ftc_image_flag_monochrome )
|
|
|
|
load_flags |= FT_LOAD_MONOCHROME;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-29 18:04:28 +02:00
|
|
|
/* disable embedded bitmaps loading if necessary */
|
2000-09-22 08:52:20 +02:00
|
|
|
if ( image_type & ftc_image_flag_no_sbits )
|
2000-08-29 18:04:28 +02:00
|
|
|
load_flags |= FT_LOAD_NO_BITMAP;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
2000-08-29 18:50:01 +02:00
|
|
|
else if ( FTC_IMAGE_FORMAT( image_type ) == ftc_image_format_outline )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2000-08-29 18:04:28 +02:00
|
|
|
/* disable embedded bitmaps loading */
|
|
|
|
load_flags |= FT_LOAD_NO_BITMAP;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
if ( image_type & ftc_image_flag_unscaled )
|
2000-08-29 18:04:28 +02:00
|
|
|
load_flags |= FT_LOAD_NO_SCALE;
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
if ( image_type & ftc_image_flag_unhinted )
|
2000-08-29 18:04:28 +02:00
|
|
|
load_flags |= FT_LOAD_NO_HINTING;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
if ( image_type & ftc_image_flag_autohinted )
|
2000-08-29 18:04:28 +02:00
|
|
|
load_flags |= FT_LOAD_FORCE_AUTOHINT;
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-08-29 18:04:28 +02:00
|
|
|
error = FT_Load_Glyph( face, glyph_index, load_flags );
|
2000-08-23 23:11:13 +02:00
|
|
|
if ( !error )
|
|
|
|
{
|
2000-08-29 18:04:28 +02:00
|
|
|
if ( face->glyph->format == ft_glyph_format_bitmap ||
|
|
|
|
face->glyph->format == ft_glyph_format_outline )
|
2000-10-12 07:05:40 +02:00
|
|
|
{
|
2000-08-23 23:11:13 +02:00
|
|
|
/* ok, copy it */
|
|
|
|
FT_Glyph glyph;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
2000-08-23 23:11:13 +02:00
|
|
|
error = FT_Get_Glyph( face->glyph, &glyph );
|
|
|
|
if ( !error )
|
2000-10-28 09:26:59 +02:00
|
|
|
node->ft_glyph = glyph;
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
2000-08-29 18:04:28 +02:00
|
|
|
else
|
|
|
|
error = FT_Err_Invalid_Argument;
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 23:11:13 +02:00
|
|
|
Exit:
|
2000-10-12 07:05:40 +02:00
|
|
|
if ( error && node )
|
|
|
|
FREE( node );
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
*anode = node;
|
2000-08-23 23:11:13 +02:00
|
|
|
return error;
|
2000-09-19 03:11:11 +02:00
|
|
|
}
|
2000-08-23 23:11:13 +02:00
|
|
|
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/* this function is important because it is both part of */
|
2000-10-31 23:13:54 +01:00
|
|
|
/* an FTC_GlyphSet_Class and an FTC_CacheNode_Class */
|
2000-10-12 07:05:40 +02:00
|
|
|
/* */
|
2000-09-19 03:11:11 +02:00
|
|
|
LOCAL_FUNC_X
|
2000-10-28 09:26:59 +02:00
|
|
|
FT_ULong ftc_glyph_image_node_size( FTC_GlyphImage node )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-12 07:05:40 +02:00
|
|
|
FT_ULong size = 0;
|
2000-10-28 09:26:59 +02:00
|
|
|
FT_Glyph glyph = node->ft_glyph;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
switch ( glyph->format )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-12 07:05:40 +02:00
|
|
|
case ft_glyph_format_bitmap:
|
|
|
|
{
|
|
|
|
FT_BitmapGlyph bitg;
|
|
|
|
|
|
|
|
|
|
|
|
bitg = (FT_BitmapGlyph)glyph;
|
|
|
|
size = bitg->bitmap.rows * labs( bitg->bitmap.pitch ) +
|
|
|
|
sizeof ( *bitg );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ft_glyph_format_outline:
|
|
|
|
{
|
|
|
|
FT_OutlineGlyph outg;
|
|
|
|
|
|
|
|
|
|
|
|
outg = (FT_OutlineGlyph)glyph;
|
|
|
|
size = outg->outline.n_points *
|
|
|
|
( sizeof( FT_Vector ) + sizeof ( FT_Byte ) ) +
|
|
|
|
outg->outline.n_contours * sizeof ( FT_Short ) +
|
|
|
|
sizeof ( *outg );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
;
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
size += sizeof ( *node );
|
2000-09-19 03:11:11 +02:00
|
|
|
return size;
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
2000-10-29 01:34:45 +02:00
|
|
|
/***** GLYPH IMAGE SETS *****/
|
2000-10-12 07:05:40 +02:00
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
LOCAL_FUNC_X
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_Error ftc_image_set_init( FTC_ImageSet iset,
|
|
|
|
FTC_Image_Desc* type )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
iset->description = *type;
|
2000-09-19 03:11:11 +02:00
|
|
|
return 0;
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
LOCAL_FUNC_X
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_Bool ftc_image_set_compare( FTC_ImageSet iset,
|
|
|
|
FTC_Image_Desc* type )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
return !memcmp( &iset->description, type, sizeof ( *type ) );
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_CPLUSPLUS( const FTC_GlyphSet_Class ) ftc_glyph_image_set_class =
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
sizeof( FTC_ImageSetRec ),
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
(FTC_GlyphSet_InitFunc) ftc_image_set_init,
|
|
|
|
(FTC_GlyphSet_DoneFunc) 0,
|
|
|
|
(FTC_GlyphSet_CompareFunc) ftc_image_set_compare,
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
(FTC_GlyphSet_NewNodeFunc) ftc_glyph_image_node_new,
|
|
|
|
(FTC_GlyphSet_SizeNodeFunc) ftc_glyph_image_node_size,
|
|
|
|
(FTC_GlyphSet_DestroyNodeFunc)ftc_glyph_image_node_destroy
|
2000-10-12 07:05:40 +02:00
|
|
|
};
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
|
|
|
/***** GLYPH IMAGE CACHE *****/
|
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
2000-08-29 18:04:28 +02:00
|
|
|
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
FT_CPLUSPLUS( const FTC_Glyph_Cache_Class ) ftc_glyph_image_cache_class =
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
sizeof( FTC_Image_CacheRec ),
|
|
|
|
(FTC_Cache_InitFunc) FTC_Glyph_Cache_Init,
|
|
|
|
(FTC_Cache_DoneFunc) FTC_Glyph_Cache_Done
|
2000-09-19 03:11:11 +02:00
|
|
|
},
|
2000-10-28 15:17:11 +02:00
|
|
|
(FTC_GlyphSet_Class*) &ftc_glyph_image_set_class
|
2000-09-19 03:11:11 +02:00
|
|
|
};
|
2000-08-23 23:11:13 +02:00
|
|
|
|
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
FT_EXPORT_FUNC( FT_Error ) FTC_Image_Cache_New( FTC_Manager manager,
|
|
|
|
FTC_Image_Cache* acache )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-09-19 03:11:11 +02:00
|
|
|
return FTC_Manager_Register_Cache(
|
2000-10-31 23:13:54 +01:00
|
|
|
manager,
|
|
|
|
(FTC_Cache_Class*)&ftc_glyph_image_cache_class,
|
|
|
|
(FTC_Cache*)acache );
|
2000-08-23 23:11:13 +02:00
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-10-31 23:13:54 +01:00
|
|
|
FT_EXPORT_DEF( FT_Error ) FTC_Image_Cache_Lookup( FTC_Image_Cache cache,
|
|
|
|
FTC_Image_Desc* desc,
|
|
|
|
FT_UInt gindex,
|
|
|
|
FT_Glyph* aglyph )
|
2000-08-23 23:11:13 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
FT_Error error;
|
|
|
|
FTC_GlyphSet gset;
|
|
|
|
FTC_GlyphNode node;
|
|
|
|
FTC_Manager manager;
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
FTC_ImageSet img_set;
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
/* check for valid `desc' delayed to FT_Lru_Lookup() */
|
|
|
|
|
|
|
|
if ( !cache || !aglyph )
|
|
|
|
return FT_Err_Invalid_Argument;
|
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
*aglyph = 0;
|
|
|
|
gset = cache->root.last_gset;
|
|
|
|
img_set = (FTC_ImageSet)gset;
|
|
|
|
if ( !gset || memcmp( &img_set->description, desc, sizeof ( *desc ) ) )
|
2000-08-24 14:39:40 +02:00
|
|
|
{
|
2000-10-28 15:17:11 +02:00
|
|
|
error = FT_Lru_Lookup( cache->root.gsets_lru,
|
2000-08-24 18:29:15 +02:00
|
|
|
(FT_LruKey)desc,
|
2000-10-28 15:17:11 +02:00
|
|
|
(FT_Pointer*)&gset );
|
|
|
|
cache->root.last_gset = gset;
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( error )
|
2000-08-24 14:39:40 +02:00
|
|
|
goto Exit;
|
|
|
|
}
|
2000-08-23 23:11:13 +02:00
|
|
|
|
2000-10-28 15:17:11 +02:00
|
|
|
error = FTC_GlyphSet_Lookup_Node( gset, gindex, &node );
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( error )
|
2000-08-23 23:11:13 +02:00
|
|
|
goto Exit;
|
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* now compress the manager's cache pool if needed */
|
|
|
|
manager = cache->root.root.manager;
|
2000-10-12 07:05:40 +02:00
|
|
|
if ( manager->num_bytes > manager->max_bytes )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2000-10-28 09:26:59 +02:00
|
|
|
FTC_GlyphNode_Ref ( node );
|
2000-09-19 03:11:11 +02:00
|
|
|
FTC_Manager_Compress( manager );
|
2000-10-28 09:26:59 +02:00
|
|
|
FTC_GlyphNode_Unref ( node );
|
2000-09-19 03:11:11 +02:00
|
|
|
}
|
2000-08-29 18:04:28 +02:00
|
|
|
|
2000-10-28 09:26:59 +02:00
|
|
|
*aglyph = ((FTC_GlyphImage)node)->ft_glyph;
|
2000-08-23 23:11:13 +02:00
|
|
|
|
|
|
|
Exit:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/* END */
|