2000-05-29 22:55:13 +02:00
|
|
|
/***************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* t2gload.c */
|
|
|
|
/* */
|
|
|
|
/* OpenType Glyph Loader (body). */
|
|
|
|
/* */
|
2000-06-12 21:36:41 +02:00
|
|
|
/* Copyright 1996-2000 by */
|
2000-05-29 22:55:13 +02:00
|
|
|
/* David Turner, Robert Wilhelm, and Werner Lemberg. */
|
|
|
|
/* */
|
2000-06-12 21:36:41 +02:00
|
|
|
/* This file is part of the FreeType project, and may only be used, */
|
|
|
|
/* modified, and distributed under the terms of the FreeType project */
|
2000-05-29 22:55:13 +02:00
|
|
|
/* 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. */
|
|
|
|
/* */
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
#include <freetype/internal/ftdebug.h>
|
|
|
|
#include <freetype/internal/ftcalc.h>
|
|
|
|
#include <freetype/internal/ftstream.h>
|
|
|
|
#include <freetype/internal/sfnt.h>
|
2000-07-01 01:12:55 +02:00
|
|
|
#include <freetype/ftoutln.h>
|
2000-05-29 22:55:13 +02:00
|
|
|
#include <freetype/tttags.h>
|
|
|
|
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-08 02:41:13 +02:00
|
|
|
#ifdef FT_FLAT_COMPILE
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-08 02:41:13 +02:00
|
|
|
#include "t2load.h"
|
|
|
|
#include "t2gload.h"
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-08 02:41:13 +02:00
|
|
|
#else
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-08 02:41:13 +02:00
|
|
|
#include <cff/t2load.h>
|
|
|
|
#include <cff/t2gload.h>
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-08 02:41:13 +02:00
|
|
|
#endif
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
#include <freetype/internal/t2errors.h>
|
|
|
|
|
2000-06-12 21:36:41 +02:00
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* The macro FT_COMPONENT is used in trace mode. It is an implicit */
|
|
|
|
/* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */
|
|
|
|
/* messages during execution. */
|
|
|
|
/* */
|
|
|
|
#undef FT_COMPONENT
|
|
|
|
#define FT_COMPONENT trace_t2gload
|
|
|
|
|
|
|
|
|
|
|
|
typedef enum T2_Operator_
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
t2_op_unknown = 0,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_rmoveto,
|
|
|
|
t2_op_hmoveto,
|
|
|
|
t2_op_vmoveto,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_rlineto,
|
|
|
|
t2_op_hlineto,
|
|
|
|
t2_op_vlineto,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_rrcurveto,
|
|
|
|
t2_op_hhcurveto,
|
|
|
|
t2_op_hvcurveto,
|
|
|
|
t2_op_rcurveline,
|
|
|
|
t2_op_rlinecurve,
|
|
|
|
t2_op_vhcurveto,
|
|
|
|
t2_op_vvcurveto,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_flex,
|
|
|
|
t2_op_hflex,
|
|
|
|
t2_op_hflex1,
|
|
|
|
t2_op_flex1,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_endchar,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_hstem,
|
|
|
|
t2_op_vstem,
|
|
|
|
t2_op_hstemhm,
|
|
|
|
t2_op_vstemhm,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_hintmask,
|
|
|
|
t2_op_cntrmask,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_abs,
|
|
|
|
t2_op_add,
|
|
|
|
t2_op_sub,
|
|
|
|
t2_op_div,
|
|
|
|
t2_op_neg,
|
|
|
|
t2_op_random,
|
|
|
|
t2_op_mul,
|
|
|
|
t2_op_sqrt,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_blend,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_drop,
|
|
|
|
t2_op_exch,
|
|
|
|
t2_op_index,
|
|
|
|
t2_op_roll,
|
|
|
|
t2_op_dup,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_put,
|
|
|
|
t2_op_get,
|
|
|
|
t2_op_store,
|
|
|
|
t2_op_load,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_and,
|
|
|
|
t2_op_or,
|
|
|
|
t2_op_not,
|
|
|
|
t2_op_eq,
|
|
|
|
t2_op_ifelse,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
t2_op_callsubr,
|
|
|
|
t2_op_callgsubr,
|
|
|
|
t2_op_return,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* do not remove */
|
|
|
|
t2_op_max
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
} T2_Operator;
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
#define T2_COUNT_CHECK_WIDTH 0x80
|
|
|
|
#define T2_COUNT_EXACT 0x40
|
|
|
|
#define T2_COUNT_CLEAR_STACK 0x20
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
|
|
|
|
static const FT_Byte t2_argument_counts[] =
|
|
|
|
{
|
|
|
|
0, /* unknown */
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
2 | T2_COUNT_CHECK_WIDTH | T2_COUNT_EXACT, /* rmoveto */
|
|
|
|
1 | T2_COUNT_CHECK_WIDTH | T2_COUNT_EXACT,
|
|
|
|
1 | T2_COUNT_CHECK_WIDTH | T2_COUNT_EXACT,
|
|
|
|
|
|
|
|
0 | T2_COUNT_CLEAR_STACK, /* rlineto */
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
|
|
|
|
0 | T2_COUNT_CLEAR_STACK, /* rrcurveto */
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
0 | T2_COUNT_CLEAR_STACK,
|
|
|
|
|
|
|
|
13, /* flex */
|
|
|
|
7,
|
|
|
|
9,
|
|
|
|
11,
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
0, /* endchar */
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
2 | T2_COUNT_CHECK_WIDTH, /* hstem */
|
|
|
|
2 | T2_COUNT_CHECK_WIDTH,
|
|
|
|
2 | T2_COUNT_CHECK_WIDTH,
|
|
|
|
2 | T2_COUNT_CHECK_WIDTH,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
0, /* hintmask */
|
|
|
|
0, /* cntrmask */
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
1, /* abs */
|
|
|
|
2,
|
|
|
|
2,
|
|
|
|
2,
|
|
|
|
1,
|
|
|
|
0,
|
|
|
|
2,
|
|
|
|
1,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
1, /* blend */
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
1, /* drop */
|
|
|
|
2,
|
|
|
|
1,
|
|
|
|
2,
|
|
|
|
1,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
2, /* put */
|
|
|
|
1,
|
|
|
|
4,
|
|
|
|
3,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
2, /* and */
|
|
|
|
2,
|
|
|
|
1,
|
|
|
|
2,
|
|
|
|
4,
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
1, /* callsubr */
|
|
|
|
1,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/********** *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** GENERIC CHARSTRING PARSING *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** *********/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* <Function> */
|
|
|
|
/* T2_Init_Builder */
|
|
|
|
/* */
|
|
|
|
/* <Description> */
|
|
|
|
/* Initializes a given glyph builder. */
|
|
|
|
/* */
|
|
|
|
/* <InOut> */
|
|
|
|
/* builder :: A pointer to the glyph builder to initialize. */
|
|
|
|
/* */
|
|
|
|
/* <Input> */
|
|
|
|
/* face :: The current face object. */
|
|
|
|
/* */
|
|
|
|
/* size :: The current size object. */
|
|
|
|
/* */
|
|
|
|
/* glyph :: The current glyph object. */
|
|
|
|
/* */
|
2000-06-07 22:04:34 +02:00
|
|
|
static
|
|
|
|
void T2_Init_Builder( T2_Builder* builder,
|
|
|
|
TT_Face face,
|
|
|
|
T2_Size size,
|
|
|
|
T2_GlyphSlot glyph )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->path_begun = 0;
|
|
|
|
builder->load_points = 1;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->face = face;
|
|
|
|
builder->glyph = glyph;
|
|
|
|
builder->memory = face->root.memory;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( glyph )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-11-04 03:52:02 +01:00
|
|
|
FT_GlyphLoader* loader = glyph->root.internal->loader;
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
builder->loader = loader;
|
|
|
|
builder->base = &loader->base.outline;
|
|
|
|
builder->current = &loader->current.outline;
|
|
|
|
FT_GlyphLoader_Rewind( loader );
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( size )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->scale_x = size->metrics.x_scale;
|
|
|
|
builder->scale_y = size->metrics.y_scale;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->pos_x = 0;
|
|
|
|
builder->pos_y = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->left_bearing.x = 0;
|
|
|
|
builder->left_bearing.y = 0;
|
|
|
|
builder->advance.x = 0;
|
|
|
|
builder->advance.y = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* <Function> */
|
|
|
|
/* T2_Done_Builder */
|
|
|
|
/* */
|
|
|
|
/* <Description> */
|
|
|
|
/* Finalizes a given glyph builder. Its contents can still be used */
|
|
|
|
/* after the call, but the function saves important information */
|
|
|
|
/* within the corresponding glyph slot. */
|
|
|
|
/* */
|
|
|
|
/* <Input> */
|
|
|
|
/* builder :: A pointer to the glyph builder to finalize. */
|
|
|
|
/* */
|
2000-05-29 22:55:13 +02:00
|
|
|
static
|
2000-06-14 01:21:00 +02:00
|
|
|
void T2_Done_Builder( T2_Builder* builder )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_GlyphSlot glyph = builder->glyph;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( glyph )
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
glyph->root.outline = *builder->base;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* <Function> */
|
|
|
|
/* t2_compute_bias */
|
|
|
|
/* */
|
|
|
|
/* <Description> */
|
|
|
|
/* Computes the bias value in dependence of the number of glyph */
|
|
|
|
/* subroutines. */
|
|
|
|
/* */
|
|
|
|
/* <Input> */
|
|
|
|
/* num_subrs :: The number of glyph subroutines. */
|
|
|
|
/* */
|
|
|
|
/* <Return> */
|
|
|
|
/* The bias value. */
|
|
|
|
static
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int t2_compute_bias( FT_UInt num_subrs )
|
2000-06-14 01:21:00 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int result;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( num_subrs < 1240 )
|
2000-06-07 22:04:34 +02:00
|
|
|
result = 107;
|
2000-06-14 01:21:00 +02:00
|
|
|
else if ( num_subrs < 33900 )
|
2000-06-07 22:04:34 +02:00
|
|
|
result = 1131;
|
|
|
|
else
|
|
|
|
result = 32768;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
return result;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* <Function> */
|
|
|
|
/* T2_Init_Decoder */
|
|
|
|
/* */
|
|
|
|
/* <Description> */
|
|
|
|
/* Initializes a given glyph decoder. */
|
|
|
|
/* */
|
|
|
|
/* <InOut> */
|
|
|
|
/* decoder :: A pointer to the glyph builder to initialize. */
|
|
|
|
/* */
|
|
|
|
/* <Input> */
|
|
|
|
/* face :: The current face object. */
|
|
|
|
/* */
|
|
|
|
/* size :: The current size object. */
|
|
|
|
/* */
|
|
|
|
/* slot :: The current glyph object. */
|
|
|
|
/* */
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_LOCAL_DEF
|
2000-06-07 22:04:34 +02:00
|
|
|
void T2_Init_Decoder( T2_Decoder* decoder,
|
|
|
|
TT_Face face,
|
|
|
|
T2_Size size,
|
|
|
|
T2_GlyphSlot slot )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-25 06:49:19 +02:00
|
|
|
CFF_Font* cff = (CFF_Font*)face->extra.data;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* clear everything */
|
2000-06-14 01:21:00 +02:00
|
|
|
MEM_Set( decoder, 0, sizeof ( *decoder ) );
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/* initialize builder */
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Init_Builder( &decoder->builder, face, size, slot );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
/* initialize Type2 decoder */
|
2000-06-07 22:04:34 +02:00
|
|
|
decoder->num_globals = cff->num_global_subrs;
|
|
|
|
decoder->globals = cff->global_subrs;
|
|
|
|
decoder->globals_bias = t2_compute_bias( decoder->num_globals );
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
/* this function is used to select the locals subrs array */
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_LOCAL_DEF
|
2000-06-29 05:14:25 +02:00
|
|
|
void T2_Prepare_Decoder( T2_Decoder* decoder,
|
|
|
|
FT_UInt glyph_index )
|
2000-06-28 01:32:27 +02:00
|
|
|
{
|
|
|
|
CFF_Font* cff = (CFF_Font*)decoder->builder.face->extra.data;
|
|
|
|
CFF_SubFont* sub = &cff->top_font;
|
2000-06-29 05:14:25 +02:00
|
|
|
|
|
|
|
|
2000-06-28 01:32:27 +02:00
|
|
|
/* manage CID fonts */
|
2000-06-29 05:14:25 +02:00
|
|
|
if ( cff->num_subfonts >= 1 )
|
2000-06-28 01:32:27 +02:00
|
|
|
{
|
|
|
|
FT_Byte fd_index = CFF_Get_FD( &cff->fd_select, glyph_index );
|
2000-06-29 05:14:25 +02:00
|
|
|
|
|
|
|
|
2000-06-28 01:32:27 +02:00
|
|
|
sub = cff->subfonts[fd_index];
|
|
|
|
}
|
2000-06-29 05:14:25 +02:00
|
|
|
|
|
|
|
decoder->num_locals = sub->num_local_subrs;
|
|
|
|
decoder->locals = sub->local_subrs;
|
|
|
|
decoder->locals_bias = t2_compute_bias( decoder->num_locals );
|
2000-06-28 01:32:27 +02:00
|
|
|
|
|
|
|
decoder->glyph_width = sub->private_dict.default_width;
|
|
|
|
decoder->nominal_width = sub->private_dict.nominal_width;
|
2000-06-29 05:14:25 +02:00
|
|
|
}
|
2000-06-28 01:32:27 +02:00
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/* check that there is enough room for `count' more points */
|
2000-05-29 22:55:13 +02:00
|
|
|
static
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error check_points( T2_Builder* builder,
|
|
|
|
FT_Int count )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-25 08:47:11 +02:00
|
|
|
return FT_GlyphLoader_Check_Points( builder->loader, count, 0 );
|
2000-06-29 05:14:25 +02:00
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-16 08:49:56 +02:00
|
|
|
/* add a new point, do not check space */
|
2000-06-07 22:04:34 +02:00
|
|
|
static
|
|
|
|
void add_point( T2_Builder* builder,
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Pos x,
|
|
|
|
FT_Pos y,
|
|
|
|
FT_Byte flag )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
FT_Outline* outline = builder->current;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( builder->load_points )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Vector* point = outline->points + outline->n_points;
|
|
|
|
FT_Byte* control = (FT_Byte*)outline->tags + outline->n_points;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
point->x = x >> 16;
|
|
|
|
point->y = y >> 16;
|
2000-06-14 01:21:00 +02:00
|
|
|
*control = flag ? FT_Curve_Tag_On : FT_Curve_Tag_Cubic;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->last = *point;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
outline->n_points++;
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
/* check space for a new on-curve point, then add it */
|
2000-06-07 22:04:34 +02:00
|
|
|
static
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error add_point1( T2_Builder* builder,
|
|
|
|
FT_Pos x,
|
|
|
|
FT_Pos y )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error error;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
error = check_points( builder, 1 );
|
|
|
|
if ( !error )
|
2000-06-07 22:04:34 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* check room for a new contour, then add it */
|
2000-05-29 22:55:13 +02:00
|
|
|
static
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error add_contour( T2_Builder* builder )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
FT_Outline* outline = builder->current;
|
|
|
|
FT_Error error;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( !builder->load_points )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
outline->n_contours++;
|
2000-06-14 01:21:00 +02:00
|
|
|
return T2_Err_Ok;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
error = FT_GlyphLoader_Check_Points( builder->loader, 0, 1 );
|
2000-06-25 08:47:11 +02:00
|
|
|
if ( !error )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
if ( outline->n_contours > 0 )
|
|
|
|
outline->contours[outline->n_contours - 1] = outline->n_points - 1;
|
2000-06-29 05:14:25 +02:00
|
|
|
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
outline->n_contours++;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-25 08:47:11 +02:00
|
|
|
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
return error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* if a path was begun, add its first on-curve point */
|
|
|
|
static
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error start_point( T2_Builder* builder,
|
|
|
|
FT_Pos x,
|
|
|
|
FT_Pos y )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-07-07 22:14:22 +02:00
|
|
|
FT_Error error = 0;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/* test whether we are building a new contour */
|
|
|
|
if ( !builder->path_begun )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
builder->path_begun = 1;
|
|
|
|
error = add_contour( builder );
|
2000-07-13 02:21:51 +02:00
|
|
|
if ( !error )
|
|
|
|
error = add_point1( builder, x, y );
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-07-13 02:21:51 +02:00
|
|
|
return error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* close the current contour */
|
|
|
|
static
|
|
|
|
void close_contour( T2_Builder* builder )
|
|
|
|
{
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
FT_Outline* outline = builder->current;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
/* XXXX: We must not include the last point in the path if it */
|
|
|
|
/* is located on the first point. */
|
|
|
|
if ( outline->n_points > 1 )
|
2000-06-29 23:48:58 +02:00
|
|
|
{
|
2000-10-11 07:52:42 +02:00
|
|
|
FT_Int first = 0;
|
|
|
|
FT_Vector* p1 = outline->points + first;
|
|
|
|
FT_Vector* p2 = outline->points + outline->n_points - 1;
|
|
|
|
FT_Byte* control = (FT_Byte*)outline->tags + outline->n_points - 1;
|
|
|
|
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
if ( outline->n_contours > 1 )
|
2000-06-29 23:48:58 +02:00
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
first = outline->contours[outline->n_contours - 2] + 1;
|
2000-06-29 23:48:58 +02:00
|
|
|
p1 = outline->points + first;
|
|
|
|
}
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-10-11 07:52:42 +02:00
|
|
|
/* `delete' last point only if it coincides with the first */
|
|
|
|
/* point and it is not a control point (which can happen). */
|
2000-06-29 23:48:58 +02:00
|
|
|
if ( p1->x == p2->x && p1->y == p2->y )
|
2000-10-11 07:52:42 +02:00
|
|
|
if ( *control == FT_Curve_Tag_On )
|
|
|
|
outline->n_points--;
|
2000-06-29 23:48:58 +02:00
|
|
|
}
|
2000-06-25 08:47:11 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
if ( outline->n_contours > 0 )
|
2000-06-14 01:21:00 +02:00
|
|
|
outline->contours[outline->n_contours - 1] = outline->n_points - 1;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
#define USE_ARGS( n ) do \
|
|
|
|
{ \
|
|
|
|
top -= n; \
|
|
|
|
if ( top < decoder->stack ) \
|
|
|
|
goto Stack_Underflow; \
|
|
|
|
} while ( 0 )
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* <Function> */
|
|
|
|
/* T2_Parse_CharStrings */
|
|
|
|
/* */
|
|
|
|
/* <Description> */
|
|
|
|
/* Parses a given Type 2 charstrings program. */
|
|
|
|
/* */
|
|
|
|
/* <InOut> */
|
2000-06-30 08:21:26 +02:00
|
|
|
/* decoder :: The current Type 1 decoder. */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* */
|
|
|
|
/* <Input> */
|
2000-06-30 08:21:26 +02:00
|
|
|
/* charstring_base :: The base of the charstring stream. */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* */
|
2000-06-30 08:21:26 +02:00
|
|
|
/* charstring_len :: The length in bytes of the charstring stream. */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* */
|
|
|
|
/* <Return> */
|
2000-06-25 08:47:11 +02:00
|
|
|
/* FreeType error code. 0 means success. */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* */
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_LOCAL_DEF
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error T2_Parse_CharStrings( T2_Decoder* decoder,
|
|
|
|
FT_Byte* charstring_base,
|
|
|
|
FT_Int charstring_len )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_Error error;
|
|
|
|
T2_Decoder_Zone* zone;
|
|
|
|
FT_Byte* ip;
|
|
|
|
FT_Byte* limit;
|
|
|
|
T2_Builder* builder = &decoder->builder;
|
|
|
|
FT_Outline* outline;
|
|
|
|
FT_Pos x, y;
|
|
|
|
FT_Fixed seed;
|
|
|
|
FT_Fixed* stack;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* set default width */
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->num_hints = 0;
|
|
|
|
decoder->read_width = 1;
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* compute random seed from stack address of parameter */
|
2000-06-14 01:21:00 +02:00
|
|
|
seed = (FT_Fixed)(char*)&seed ^
|
|
|
|
(FT_Fixed)(char*)&decoder ^
|
|
|
|
(FT_Fixed)(char*)&charstring_base;
|
|
|
|
seed = ( seed ^ ( seed >> 10 ) ^ ( seed >> 20 ) ) & 0xFFFF;
|
|
|
|
if ( seed == 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
seed = 0x7384;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-16 08:49:56 +02:00
|
|
|
/* initialize the decoder */
|
2000-06-07 22:04:34 +02:00
|
|
|
decoder->top = decoder->stack;
|
|
|
|
decoder->zone = decoder->zones;
|
|
|
|
zone = decoder->zones;
|
|
|
|
stack = decoder->top;
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
builder->path_begun = 0;
|
2000-06-07 22:04:34 +02:00
|
|
|
|
|
|
|
zone->base = charstring_base;
|
|
|
|
limit = zone->limit = charstring_base + charstring_len;
|
|
|
|
ip = zone->cursor = zone->base;
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
error = T2_Err_Ok;
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
outline = builder->current;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
x = builder->pos_x;
|
|
|
|
y = builder->pos_y;
|
|
|
|
|
|
|
|
/* now, execute loop */
|
|
|
|
while ( ip < limit )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Operator op;
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Byte v;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/********************************************************************/
|
|
|
|
/* */
|
|
|
|
/* Decode operator or operand */
|
|
|
|
/* */
|
|
|
|
v = *ip++;
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( v >= 32 || v == 28 )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-07-08 02:41:13 +02:00
|
|
|
FT_Int shift = 16;
|
|
|
|
FT_Int32 val;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* this is an operand, push it on the stack */
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( v == 28 )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( ip + 1 >= limit )
|
|
|
|
goto Syntax_Error;
|
2000-07-08 03:18:38 +02:00
|
|
|
val = (FT_Short)( ( (FT_Short)ip[0] << 8 ) | ip[1] );
|
2000-06-07 22:04:34 +02:00
|
|
|
ip += 2;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
else if ( v < 247 )
|
2000-07-08 03:49:28 +02:00
|
|
|
val = (FT_Long)v - 139;
|
2000-06-07 22:04:34 +02:00
|
|
|
else if ( v < 251 )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( ip >= limit )
|
|
|
|
goto Syntax_Error;
|
2000-07-13 02:21:51 +02:00
|
|
|
val = ( (FT_Long)v - 247 ) * 256 + *ip++ + 108;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
else if ( v < 255 )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( ip >= limit )
|
|
|
|
goto Syntax_Error;
|
2000-07-13 02:21:51 +02:00
|
|
|
val = -( (FT_Long)v - 251 ) * 256 - *ip++ - 108;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
else
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( ip + 3 >= limit )
|
|
|
|
goto Syntax_Error;
|
2000-07-04 20:12:13 +02:00
|
|
|
val = ( (FT_Int32)ip[0] << 24 ) |
|
|
|
|
( (FT_Int32)ip[1] << 16 ) |
|
|
|
|
( (FT_Int32)ip[2] << 8 ) |
|
|
|
|
ip[3];
|
2000-06-07 22:04:34 +02:00
|
|
|
ip += 4;
|
2000-06-14 01:21:00 +02:00
|
|
|
shift = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( decoder->top - stack >= T2_MAX_OPERANDS )
|
2000-06-07 22:04:34 +02:00
|
|
|
goto Stack_Overflow;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
val <<= shift;
|
|
|
|
*decoder->top++ = val;
|
|
|
|
|
|
|
|
#ifdef FT_DEBUG_LEVEL_TRACE
|
|
|
|
if ( !( val & 0xFFFF ) )
|
2000-07-08 02:41:13 +02:00
|
|
|
FT_TRACE4(( " %d", (FT_Int32)( val >> 16 ) ));
|
2000-06-07 22:04:34 +02:00
|
|
|
else
|
|
|
|
FT_TRACE4(( " %.2f", val/65536.0 ));
|
2000-06-14 01:21:00 +02:00
|
|
|
#endif
|
|
|
|
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
else
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Fixed* args = decoder->top;
|
|
|
|
FT_Int num_args = args - decoder->stack;
|
|
|
|
FT_Int req_args;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* find operator */
|
|
|
|
op = t2_op_unknown;
|
2000-06-30 08:21:26 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
switch ( v )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
case 1:
|
|
|
|
op = t2_op_hstem;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
op = t2_op_vstem;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
op = t2_op_vmoveto;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
op = t2_op_rlineto;
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
op = t2_op_hlineto;
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
op = t2_op_vlineto;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
op = t2_op_rrcurveto;
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
op = t2_op_callsubr;
|
|
|
|
break;
|
|
|
|
case 11:
|
|
|
|
op = t2_op_return;
|
|
|
|
break;
|
|
|
|
case 12:
|
|
|
|
{
|
|
|
|
if ( ip >= limit )
|
|
|
|
goto Syntax_Error;
|
|
|
|
v = *ip++;
|
|
|
|
|
|
|
|
switch ( v )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
case 3:
|
|
|
|
op = t2_op_and;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
op = t2_op_or;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
op = t2_op_not;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
op = t2_op_store;
|
|
|
|
break;
|
|
|
|
case 9:
|
|
|
|
op = t2_op_abs;
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
op = t2_op_add;
|
|
|
|
break;
|
|
|
|
case 11:
|
|
|
|
op = t2_op_sub;
|
|
|
|
break;
|
|
|
|
case 12:
|
|
|
|
op = t2_op_div;
|
|
|
|
break;
|
|
|
|
case 13:
|
|
|
|
op = t2_op_load;
|
|
|
|
break;
|
|
|
|
case 14:
|
|
|
|
op = t2_op_neg;
|
|
|
|
break;
|
|
|
|
case 15:
|
|
|
|
op = t2_op_eq;
|
|
|
|
break;
|
|
|
|
case 18:
|
|
|
|
op = t2_op_drop;
|
|
|
|
break;
|
|
|
|
case 20:
|
|
|
|
op = t2_op_put;
|
|
|
|
break;
|
|
|
|
case 21:
|
|
|
|
op = t2_op_get;
|
|
|
|
break;
|
|
|
|
case 22:
|
|
|
|
op = t2_op_ifelse;
|
|
|
|
break;
|
|
|
|
case 23:
|
|
|
|
op = t2_op_random;
|
|
|
|
break;
|
|
|
|
case 24:
|
|
|
|
op = t2_op_mul;
|
|
|
|
break;
|
|
|
|
case 26:
|
|
|
|
op = t2_op_sqrt;
|
|
|
|
break;
|
|
|
|
case 27:
|
|
|
|
op = t2_op_dup;
|
|
|
|
break;
|
|
|
|
case 28:
|
|
|
|
op = t2_op_exch;
|
|
|
|
break;
|
|
|
|
case 29:
|
|
|
|
op = t2_op_index;
|
|
|
|
break;
|
|
|
|
case 30:
|
|
|
|
op = t2_op_roll;
|
|
|
|
break;
|
|
|
|
case 34:
|
|
|
|
op = t2_op_hflex;
|
|
|
|
break;
|
|
|
|
case 35:
|
|
|
|
op = t2_op_flex;
|
|
|
|
break;
|
|
|
|
case 36:
|
|
|
|
op = t2_op_hflex1;
|
|
|
|
break;
|
|
|
|
case 37:
|
|
|
|
op = t2_op_flex1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* decrement ip for syntax error message */
|
|
|
|
ip--;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 14:
|
|
|
|
op = t2_op_endchar;
|
|
|
|
break;
|
|
|
|
case 16:
|
|
|
|
op = t2_op_blend;
|
|
|
|
break;
|
|
|
|
case 18:
|
|
|
|
op = t2_op_hstemhm;
|
|
|
|
break;
|
|
|
|
case 19:
|
|
|
|
op = t2_op_hintmask;
|
|
|
|
break;
|
|
|
|
case 20:
|
|
|
|
op = t2_op_cntrmask;
|
|
|
|
break;
|
|
|
|
case 21:
|
|
|
|
op = t2_op_rmoveto;
|
|
|
|
break;
|
|
|
|
case 22:
|
|
|
|
op = t2_op_hmoveto;
|
|
|
|
break;
|
|
|
|
case 23:
|
|
|
|
op = t2_op_vstemhm;
|
|
|
|
break;
|
|
|
|
case 24:
|
|
|
|
op = t2_op_rcurveline;
|
|
|
|
break;
|
|
|
|
case 25:
|
|
|
|
op = t2_op_rlinecurve;
|
|
|
|
break;
|
|
|
|
case 26:
|
|
|
|
op = t2_op_vvcurveto;
|
|
|
|
break;
|
|
|
|
case 27:
|
|
|
|
op = t2_op_hhcurveto;
|
|
|
|
break;
|
|
|
|
case 29:
|
|
|
|
op = t2_op_callgsubr;
|
|
|
|
break;
|
|
|
|
case 30:
|
|
|
|
op = t2_op_vhcurveto;
|
|
|
|
break;
|
|
|
|
case 31:
|
|
|
|
op = t2_op_hvcurveto;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
|
|
|
if ( op == t2_op_unknown )
|
|
|
|
goto Syntax_Error;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* check arguments */
|
2000-08-01 00:51:00 +02:00
|
|
|
req_args = t2_argument_counts[op];
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( req_args & T2_COUNT_CHECK_WIDTH )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
args = stack;
|
2000-06-28 22:43:07 +02:00
|
|
|
if ( num_args & 1 && decoder->read_width )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->glyph_width = decoder->nominal_width +
|
|
|
|
( stack[0] >> 16 );
|
2000-06-07 22:04:34 +02:00
|
|
|
num_args--;
|
|
|
|
args++;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-28 22:43:07 +02:00
|
|
|
decoder->read_width = 0;
|
2000-06-07 22:04:34 +02:00
|
|
|
req_args = 0;
|
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
req_args &= 15;
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( num_args < req_args )
|
|
|
|
goto Stack_Underflow;
|
2000-06-07 22:04:34 +02:00
|
|
|
args -= req_args;
|
|
|
|
num_args -= req_args;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
switch ( op )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_hstem:
|
|
|
|
case t2_op_vstem:
|
|
|
|
case t2_op_hstemhm:
|
|
|
|
case t2_op_vstemhm:
|
2000-06-30 08:21:26 +02:00
|
|
|
/* if the number of arguments is not even, the first one */
|
|
|
|
/* is simply the glyph width, encoded as the difference */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* to nominalWidthX */
|
|
|
|
FT_TRACE4(( op == t2_op_hstem ? " hstem" :
|
|
|
|
op == t2_op_vstem ? " vstem" :
|
|
|
|
op == t2_op_hstemhm ? " hstemhm" :
|
|
|
|
" vstemhm" ));
|
|
|
|
decoder->num_hints += num_args / 2;
|
|
|
|
args = stack;
|
|
|
|
break;
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_hintmask:
|
|
|
|
case t2_op_cntrmask:
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_TRACE4(( op == t2_op_hintmask ? " hintmask"
|
|
|
|
: " cntrmask" ));
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->num_hints += num_args / 2;
|
|
|
|
ip += ( decoder->num_hints + 7 ) >> 3;
|
|
|
|
if ( ip >= limit )
|
|
|
|
goto Syntax_Error;
|
|
|
|
args = stack;
|
|
|
|
break;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_rmoveto:
|
|
|
|
FT_TRACE4(( " rmoveto" ));
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
close_contour( builder );
|
|
|
|
builder->path_begun = 0;
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
args = stack;
|
|
|
|
break;
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_vmoveto:
|
|
|
|
FT_TRACE4(( " vmoveto" ));
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
close_contour( builder );
|
|
|
|
builder->path_begun = 0;
|
|
|
|
y += args[0];
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_hmoveto:
|
2000-07-07 21:47:34 +02:00
|
|
|
FT_TRACE4(( " hmoveto" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
close_contour( builder );
|
|
|
|
builder->path_begun = 0;
|
|
|
|
x += args[0];
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_rlineto:
|
|
|
|
FT_TRACE4(( " rlineto" ));
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( start_point ( builder, x, y ) ||
|
|
|
|
check_points( builder, num_args / 2 ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
if ( num_args < 2 || num_args & 1 )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
while ( args < decoder->top )
|
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
args += 2;
|
|
|
|
}
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_hlineto:
|
|
|
|
case t2_op_vlineto:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int phase = ( op == t2_op_hlineto );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_TRACE4(( op == t2_op_hlineto ? " hlineto"
|
|
|
|
: " vlineto" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) ||
|
|
|
|
check_points( builder, num_args ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
while (args < decoder->top )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( phase )
|
2000-06-07 22:04:34 +02:00
|
|
|
x += args[0];
|
2000-06-14 01:21:00 +02:00
|
|
|
else
|
|
|
|
y += args[0];
|
|
|
|
|
|
|
|
if ( add_point1( builder, x, y ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args++;
|
|
|
|
phase ^= 1;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
args = stack;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_rrcurveto:
|
|
|
|
FT_TRACE4(( " rrcurveto" ));
|
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
/* check number of arguments; must be a multiple of 6 */
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( num_args % 6 != 0 )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) ||
|
|
|
|
check_points( builder, num_args / 2 ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
while ( args < decoder->top )
|
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[2];
|
|
|
|
y += args[3];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[4];
|
|
|
|
y += args[5];
|
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
args += 6;
|
|
|
|
}
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_vvcurveto:
|
|
|
|
FT_TRACE4(( " vvcurveto" ));
|
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
if ( num_args & 1 )
|
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
args++;
|
|
|
|
num_args--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( num_args % 4 != 0 )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( check_points( builder, 3 * ( num_args / 4 ) ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
while ( args < decoder->top )
|
|
|
|
{
|
|
|
|
y += args[0];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[1];
|
|
|
|
y += args[2];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
y += args[3];
|
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
args += 4;
|
|
|
|
}
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_hhcurveto:
|
|
|
|
FT_TRACE4(( " hhcurveto" ));
|
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
if ( num_args & 1 )
|
|
|
|
{
|
|
|
|
y += args[0];
|
|
|
|
args++;
|
|
|
|
num_args--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( num_args % 4 != 0 )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( check_points( builder, 3 * ( num_args / 4 ) ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
while ( args < decoder->top )
|
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[1];
|
|
|
|
y += args[2];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[3];
|
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
args += 4;
|
|
|
|
}
|
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_vhcurveto:
|
|
|
|
case t2_op_hvcurveto:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int phase;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_TRACE4(( op == t2_op_vhcurveto ? " vhcurveto"
|
|
|
|
: " hvcurveto" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
|
|
|
if (num_args < 4 || ( num_args % 4 ) > 1 )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( check_points( builder, ( num_args / 4 ) * 3 ) )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
phase = ( op == t2_op_hvcurveto );
|
2000-06-30 08:21:26 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
while ( num_args >= 4 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
num_args -= 4;
|
|
|
|
if ( phase )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[1];
|
|
|
|
y += args[2];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
y += args[3];
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( num_args == 1 )
|
|
|
|
x += args[4];
|
2000-06-07 22:04:34 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
else
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
y += args[0];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[1];
|
|
|
|
y += args[2];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[3];
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( num_args == 1 )
|
|
|
|
y += args[4];
|
2000-06-07 22:04:34 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
args += 4;
|
|
|
|
phase ^= 1;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
args = stack;
|
|
|
|
}
|
|
|
|
break;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_rlinecurve:
|
|
|
|
{
|
2000-06-29 05:14:25 +02:00
|
|
|
FT_Int num_lines = ( num_args - 6 ) / 2;
|
|
|
|
|
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
FT_TRACE4(( " rlinecurve" ));
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
if ( num_args < 8 || ( num_args - 6 ) & 1 )
|
2000-06-14 01:21:00 +02:00
|
|
|
goto Stack_Underflow;
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
if ( start_point( builder, x, y ) ||
|
|
|
|
check_points( builder, num_lines + 3 ) )
|
2000-06-14 01:21:00 +02:00
|
|
|
goto Memory_Error;
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
args = stack;
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
/* first, add the line segments */
|
2000-06-29 05:14:25 +02:00
|
|
|
while ( num_lines > 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-06-28 07:35:58 +02:00
|
|
|
args += 2;
|
|
|
|
num_lines--;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
/* then the curve */
|
2000-06-28 07:35:58 +02:00
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[2];
|
|
|
|
y += args[3];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[4];
|
|
|
|
y += args[5];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-07-02 02:27:53 +02:00
|
|
|
args = stack;
|
2000-06-28 07:35:58 +02:00
|
|
|
}
|
|
|
|
break;
|
2000-07-02 02:27:53 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
case t2_op_rcurveline:
|
|
|
|
{
|
2000-06-29 05:14:25 +02:00
|
|
|
FT_Int num_curves = ( num_args - 2 ) / 6;
|
2000-06-28 07:35:58 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
FT_TRACE4(( " rcurveline" ));
|
|
|
|
|
2000-06-29 05:14:25 +02:00
|
|
|
if ( num_args < 8 || ( num_args - 2 ) % 6 )
|
2000-06-28 07:35:58 +02:00
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( start_point ( builder, x, y ) ||
|
|
|
|
check_points( builder, num_curves*3 + 2 ) )
|
|
|
|
goto Memory_Error;
|
|
|
|
|
|
|
|
args = stack;
|
2000-06-29 05:14:25 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
/* first, add the curves */
|
2000-06-29 05:14:25 +02:00
|
|
|
while ( num_curves > 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[2];
|
|
|
|
y += args[3];
|
|
|
|
add_point( builder, x, y, 0 );
|
|
|
|
x += args[4];
|
|
|
|
y += args[5];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-06-28 07:35:58 +02:00
|
|
|
args += 6;
|
|
|
|
num_curves--;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-29 05:14:25 +02:00
|
|
|
|
2000-06-28 07:35:58 +02:00
|
|
|
/* then the final line */
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-06-14 01:21:00 +02:00
|
|
|
args = stack;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
case t2_op_hflex1:
|
|
|
|
{
|
|
|
|
FT_Pos start_y;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
FT_TRACE4(( " hflex1" ));
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
/* adding five more points; 4 control points, 1 on-curve point */
|
2000-07-13 02:21:51 +02:00
|
|
|
/* make sure we have enough space for the start point if it */
|
|
|
|
/* needs to be added.. */
|
2000-07-08 21:51:42 +02:00
|
|
|
if ( start_point( builder, x, y ) ||
|
2000-07-13 02:21:51 +02:00
|
|
|
check_points( builder, 6 ) )
|
2000-07-07 21:47:34 +02:00
|
|
|
goto Memory_Error;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* Record the starting point's y postion for later use */
|
|
|
|
start_y = y;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* first control point */
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* second control point */
|
|
|
|
x += args[2];
|
|
|
|
y += args[3];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* join point; on curve, with y-value the same as the last */
|
|
|
|
/* control point's y-value */
|
|
|
|
x += args[4];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* third control point, with y-value the same as the join */
|
|
|
|
/* point's y-value */
|
|
|
|
x += args[5];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* fourth control point */
|
|
|
|
x += args[6];
|
|
|
|
y += args[7];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
/* ending point, with y-value the same as the start */
|
2000-07-07 21:47:34 +02:00
|
|
|
x += args[8];
|
|
|
|
y = start_y;
|
2000-07-13 02:21:51 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case t2_op_hflex:
|
|
|
|
{
|
|
|
|
FT_Pos start_y;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
FT_TRACE4(( " hflex" ));
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-14 08:16:47 +02:00
|
|
|
/* adding six more points; 4 control points, 2 on-curve points */
|
2000-07-08 21:51:42 +02:00
|
|
|
if ( start_point( builder, x, y ) ||
|
2000-07-13 02:21:51 +02:00
|
|
|
check_points ( builder, 6 ) )
|
2000-07-07 21:47:34 +02:00
|
|
|
goto Memory_Error;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
/* record the starting point's y-position for later use */
|
2000-07-07 21:47:34 +02:00
|
|
|
start_y = y;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* first control point */
|
|
|
|
x += args[0];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* second control point */
|
|
|
|
x += args[1];
|
|
|
|
y += args[2];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* join point; on curve, with y-value the same as the last */
|
|
|
|
/* control point's y-value */
|
|
|
|
x += args[3];
|
|
|
|
add_point( builder, x, y, 1 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* third control point, with y-value the same as the join */
|
|
|
|
/* point's y-value */
|
|
|
|
x += args[4];
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* fourth control point */
|
|
|
|
x += args[5];
|
|
|
|
y = start_y;
|
|
|
|
add_point( builder, x, y, 0 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* ending point, with y-value the same as the start point's */
|
2000-07-08 21:51:42 +02:00
|
|
|
/* y-value -- we don't add this point, though */
|
2000-07-07 21:47:34 +02:00
|
|
|
x += args[6];
|
2000-07-13 02:21:51 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case t2_op_flex1:
|
|
|
|
{
|
2000-07-08 21:51:42 +02:00
|
|
|
FT_Pos start_x, start_y; /* record start x, y values for alter */
|
|
|
|
/* use */
|
|
|
|
FT_Int dx = 0, dy = 0; /* used in horizontal/vertical */
|
|
|
|
/* algorithm below */
|
|
|
|
FT_Int horizontal, count;
|
2000-07-09 21:15:30 +02:00
|
|
|
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
FT_TRACE4(( " flex1" ));
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-14 08:16:47 +02:00
|
|
|
/* adding six more points; 4 control points, 2 on-curve points */
|
2000-07-08 21:51:42 +02:00
|
|
|
if ( start_point( builder, x, y ) ||
|
2000-07-13 02:21:51 +02:00
|
|
|
check_points( builder, 6 ) )
|
2000-07-07 21:47:34 +02:00
|
|
|
goto Memory_Error;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
/* record the starting point's x, y postion for later use */
|
2000-07-07 21:47:34 +02:00
|
|
|
start_x = x;
|
|
|
|
start_y = y;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
/* XXX: figure out whether this is supposed to be a horizontal */
|
|
|
|
/* or vertical flex; the Type 2 specification is vague... */
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* grab up to the last argument */
|
2000-07-08 02:41:13 +02:00
|
|
|
for ( count = 5; count > 0; count-- )
|
2000-07-07 21:47:34 +02:00
|
|
|
{
|
|
|
|
dx += args[0];
|
|
|
|
dy += args[1];
|
|
|
|
args += 2;
|
|
|
|
}
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* rewind */
|
|
|
|
args = stack;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
if ( dx < 0 ) dx = -dx;
|
|
|
|
if ( dy < 0 ) dy = -dy;
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
/* strange test, but here it is... */
|
2000-07-08 21:51:42 +02:00
|
|
|
horizontal = ( dx > dy );
|
2000-07-09 21:15:30 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
for ( count = 5; count > 0; count-- )
|
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
2000-07-08 21:51:42 +02:00
|
|
|
add_point( builder, x, y, (FT_Bool)( count == 3 ) );
|
2000-07-07 21:47:34 +02:00
|
|
|
args += 2;
|
|
|
|
}
|
2000-07-09 21:15:30 +02:00
|
|
|
|
2000-07-14 08:16:47 +02:00
|
|
|
/* is last operand an x- or y-delta? */
|
2000-07-08 21:51:42 +02:00
|
|
|
if ( horizontal )
|
2000-07-07 21:47:34 +02:00
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
y = start_y;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
x = start_x;
|
|
|
|
y += args[0];
|
|
|
|
}
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-13 02:21:51 +02:00
|
|
|
add_point( builder, x, y, 1 );
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case t2_op_flex:
|
|
|
|
{
|
2000-07-08 21:51:42 +02:00
|
|
|
FT_UInt count;
|
|
|
|
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
FT_TRACE4(( " flex" ));
|
2000-07-08 21:51:42 +02:00
|
|
|
|
|
|
|
if ( start_point( builder, x, y ) ||
|
2000-07-13 02:21:51 +02:00
|
|
|
check_points( builder, 6 ) )
|
2000-07-08 21:51:42 +02:00
|
|
|
goto Memory_Error;
|
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
2000-07-13 02:21:51 +02:00
|
|
|
for ( count = 6; count > 0; count-- )
|
2000-07-07 21:47:34 +02:00
|
|
|
{
|
|
|
|
x += args[0];
|
|
|
|
y += args[1];
|
2000-07-14 08:16:47 +02:00
|
|
|
add_point( builder, x, y,
|
|
|
|
(FT_Bool)( count == 3 || count == 0 ) );
|
2000-07-07 21:47:34 +02:00
|
|
|
args += 2;
|
|
|
|
}
|
2000-07-08 21:51:42 +02:00
|
|
|
|
2000-07-07 21:47:34 +02:00
|
|
|
args = stack;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
case t2_op_endchar:
|
|
|
|
FT_TRACE4(( " endchar" ));
|
|
|
|
|
|
|
|
close_contour( builder );
|
|
|
|
|
|
|
|
/* add current outline to the glyph slot */
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
FT_GlyphLoader_Add( builder->loader );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
/* return now! */
|
|
|
|
FT_TRACE4(( "\n\n" ));
|
|
|
|
return T2_Err_Ok;
|
|
|
|
|
|
|
|
case t2_op_abs:
|
|
|
|
FT_TRACE4(( " abs" ));
|
|
|
|
|
|
|
|
if ( args[0] < 0 )
|
|
|
|
args[0] = -args[0];
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_add:
|
|
|
|
FT_TRACE4(( " add" ));
|
|
|
|
|
|
|
|
args[0] += args[1];
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_sub:
|
|
|
|
FT_TRACE4(( " sub" ));
|
|
|
|
|
|
|
|
args[0] -= args[1];
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_div:
|
|
|
|
FT_TRACE4(( " div" ));
|
|
|
|
|
|
|
|
args[0] = FT_DivFix( args[0], args[1] );
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_neg:
|
|
|
|
FT_TRACE4(( " neg" ));
|
|
|
|
|
|
|
|
args[0] = -args[0];
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_random:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Fixed rand;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " rand" ));
|
|
|
|
|
|
|
|
rand = seed;
|
|
|
|
if ( rand >= 0x8000 )
|
|
|
|
rand++;
|
|
|
|
|
|
|
|
args[0] = rand;
|
|
|
|
seed = FT_MulFix( seed, 0x10000L - seed );
|
|
|
|
if ( seed == 0 )
|
|
|
|
seed += 0x2873;
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_mul:
|
|
|
|
FT_TRACE4(( " mul" ));
|
|
|
|
|
|
|
|
args[0] = FT_MulFix( args[0], args[1] );
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_sqrt:
|
|
|
|
FT_TRACE4(( " sqrt" ));
|
|
|
|
|
|
|
|
if ( args[0] > 0 )
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int count = 9;
|
|
|
|
FT_Fixed root = args[0];
|
|
|
|
FT_Fixed new_root;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
for (;;)
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
new_root = ( root + FT_DivFix(args[0],root) + 1 ) >> 1;
|
|
|
|
if ( new_root == root || count <= 0 )
|
|
|
|
break;
|
|
|
|
root = new_root;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
args[0] = new_root;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
args[0] = 0;
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_drop:
|
|
|
|
/* nothing */
|
|
|
|
FT_TRACE4(( " drop" ));
|
2000-06-30 08:21:26 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_exch:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Fixed tmp;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " exch" ));
|
|
|
|
|
|
|
|
tmp = args[0];
|
|
|
|
args[0] = args[1];
|
|
|
|
args[1] = tmp;
|
|
|
|
args += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_index:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int index = args[0] >> 16;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " index" ));
|
|
|
|
|
|
|
|
if ( index < 0 )
|
|
|
|
index = 0;
|
|
|
|
else if ( index > num_args - 2 )
|
|
|
|
index = num_args - 2;
|
|
|
|
args[0] = args[-( index + 1 )];
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_roll:
|
|
|
|
{
|
2000-08-01 00:51:00 +02:00
|
|
|
FT_Int count = (FT_Int)( args[0] >> 16 );
|
|
|
|
FT_Int index = (FT_Int)( args[1] >> 16 );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " roll" ));
|
|
|
|
|
|
|
|
if ( count <= 0 )
|
|
|
|
count = 1;
|
|
|
|
|
|
|
|
args -= count;
|
|
|
|
if ( args < stack )
|
|
|
|
goto Stack_Underflow;
|
|
|
|
|
|
|
|
if ( index >= 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
while ( index > 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-08-01 00:51:00 +02:00
|
|
|
FT_Fixed tmp = args[count - 1];
|
|
|
|
FT_Int i;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
for ( i = count - 2; i >= 0; i-- )
|
|
|
|
args[i + 1] = args[i];
|
|
|
|
args[0] = tmp;
|
|
|
|
index--;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
else
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
while ( index < 0 )
|
2000-06-14 01:21:00 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Fixed tmp = args[0];
|
|
|
|
FT_Int i;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
for ( i = 0; i < count - 1; i++ )
|
|
|
|
args[i] = args[i + 1];
|
|
|
|
args[count - 1] = tmp;
|
|
|
|
index++;
|
|
|
|
}
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
args += count;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_dup:
|
|
|
|
FT_TRACE4(( " dup" ));
|
|
|
|
|
|
|
|
args[1] = args[0];
|
|
|
|
args++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_put:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Fixed val = args[0];
|
|
|
|
FT_Int index = (FT_Int)( args[1] >> 16 );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " put" ));
|
2000-06-30 08:21:26 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( index >= 0 && index < decoder->len_buildchar )
|
|
|
|
decoder->buildchar[index] = val;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_get:
|
|
|
|
{
|
|
|
|
FT_Int index = (FT_Int)( args[0] >> 16 );
|
|
|
|
FT_Fixed val = 0;
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " get" ));
|
|
|
|
|
|
|
|
if ( index >= 0 && index < decoder->len_buildchar )
|
|
|
|
val = decoder->buildchar[index];
|
|
|
|
|
|
|
|
args[0] = val;
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_store:
|
|
|
|
FT_TRACE4(( " store "));
|
|
|
|
|
|
|
|
goto Unimplemented;
|
|
|
|
|
|
|
|
case t2_op_load:
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_TRACE4(( " load" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
goto Unimplemented;
|
|
|
|
|
|
|
|
case t2_op_and:
|
|
|
|
{
|
|
|
|
FT_Fixed cond = args[0] && args[1];
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " and" ));
|
|
|
|
|
|
|
|
args[0] = cond ? 0x10000L : 0;
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_or:
|
|
|
|
{
|
|
|
|
FT_Fixed cond = args[0] || args[1];
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " or" ));
|
|
|
|
|
|
|
|
args[0] = cond ? 0x10000L : 0;
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_eq:
|
|
|
|
{
|
|
|
|
FT_Fixed cond = !args[0];
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " eq" ));
|
|
|
|
|
|
|
|
args[0] = cond ? 0x10000L : 0;
|
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_ifelse:
|
|
|
|
{
|
|
|
|
FT_Fixed cond = (args[2] <= args[3]);
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " ifelse" ));
|
|
|
|
|
|
|
|
if ( !cond )
|
2000-06-07 22:04:34 +02:00
|
|
|
args[0] = args[1];
|
2000-06-14 01:21:00 +02:00
|
|
|
args++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_callsubr:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_UInt index = (FT_UInt)( ( args[0] >> 16 ) +
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->locals_bias );
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " callsubr(%d)", index ));
|
|
|
|
|
|
|
|
if ( index >= decoder->num_locals )
|
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings:" ));
|
|
|
|
FT_ERROR(( " invalid local subr index\n" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( zone - decoder->zones >= T2_MAX_SUBRS_CALLS )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings: too many nested subrs\n" ));
|
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
zone->cursor = ip; /* save current instruction pointer */
|
|
|
|
|
|
|
|
zone++;
|
2000-06-30 08:21:26 +02:00
|
|
|
zone->base = decoder->locals[index];
|
|
|
|
zone->limit = decoder->locals[index+1];
|
|
|
|
zone->cursor = zone->base;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( !zone->base )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings: invoking empty subrs!\n" ));
|
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
decoder->zone = zone;
|
|
|
|
ip = zone->base;
|
|
|
|
limit = zone->limit;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_callgsubr:
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_UInt index = (FT_UInt)( ( args[0] >> 16 ) +
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->globals_bias );
|
|
|
|
|
|
|
|
|
|
|
|
FT_TRACE4(( " callgsubr(%d)", index ));
|
|
|
|
|
|
|
|
if ( index >= decoder->num_globals )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings:" ));
|
|
|
|
FT_ERROR(( " invalid global subr index\n" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( zone - decoder->zones >= T2_MAX_SUBRS_CALLS )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings: too many nested subrs\n" ));
|
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
zone->cursor = ip; /* save current instruction pointer */
|
|
|
|
|
|
|
|
zone++;
|
|
|
|
zone->base = decoder->globals[index];
|
|
|
|
zone->limit = decoder->globals[index+1];
|
|
|
|
zone->cursor = zone->base;
|
|
|
|
|
|
|
|
if ( !zone->base )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-14 01:21:00 +02:00
|
|
|
FT_ERROR(( "T2_Parse_CharStrings: invoking empty subrs!\n" ));
|
|
|
|
goto Syntax_Error;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder->zone = zone;
|
|
|
|
ip = zone->base;
|
|
|
|
limit = zone->limit;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case t2_op_return:
|
|
|
|
FT_TRACE4(( " return" ));
|
|
|
|
|
|
|
|
if ( decoder->zone <= decoder->zones )
|
|
|
|
{
|
|
|
|
FT_ERROR(( "T2_Parse_CharStrings: unexpected return\n" ));
|
|
|
|
goto Syntax_Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
decoder->zone--;
|
|
|
|
zone = decoder->zone;
|
|
|
|
ip = zone->cursor;
|
|
|
|
limit = zone->limit;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Unimplemented:
|
|
|
|
FT_ERROR(( "Unimplemented opcode: %d", ip[-1] ));
|
|
|
|
|
|
|
|
if ( ip[-1] == 12 )
|
|
|
|
FT_ERROR(( " %d", ip[0] ));
|
|
|
|
FT_ERROR(( "\n" ));
|
2000-06-30 08:21:26 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
return T2_Err_Unimplemented_Feature;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
decoder->top = args;
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
} /* general operator processing */
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
} /* while ip < limit */
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
FT_TRACE4(( "..end..\n\n" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
return error;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
Syntax_Error:
|
2000-06-29 05:14:25 +02:00
|
|
|
FT_TRACE4(( "T2_Parse_CharStrings: syntax error!" ));
|
2000-06-14 01:21:00 +02:00
|
|
|
return T2_Err_Invalid_File_Format;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
Stack_Underflow:
|
2000-06-29 05:14:25 +02:00
|
|
|
FT_TRACE4(( "T2_Parse_CharStrings: stack underflow!" ));
|
2000-06-07 22:04:34 +02:00
|
|
|
return T2_Err_Too_Few_Arguments;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
Stack_Overflow:
|
2000-06-29 05:14:25 +02:00
|
|
|
FT_TRACE4(( "T2_Parse_CharStrings: stack overflow!" ));
|
2000-06-07 22:04:34 +02:00
|
|
|
return T2_Err_Stack_Overflow;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
Memory_Error:
|
|
|
|
return builder->error;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/********** *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** COMPUTE THE MAXIMUM ADVANCE WIDTH *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** The following code is in charge of computing *********/
|
|
|
|
/********** the maximum advance width of the font. It *********/
|
|
|
|
/********** quickly processes each glyph charstring to *********/
|
|
|
|
/********** extract the value from either a `sbw' or `seac' *********/
|
|
|
|
/********** operator. *********/
|
|
|
|
/********** *********/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
|
|
|
|
#if 0 /* unused until we support pure CFF fonts */
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_LOCAL_DEF
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error T2_Compute_Max_Advance( TT_Face face,
|
|
|
|
FT_Int* max_advance )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error error = 0;
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Decoder decoder;
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int glyph_index;
|
2000-06-07 22:04:34 +02:00
|
|
|
CFF_Font* cff = (CFF_Font*)face->other;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
*max_advance = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
/* Initialize load decoder */
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Init_Decoder( &decoder, face, 0, 0 );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
decoder.builder.metrics_only = 1;
|
|
|
|
decoder.builder.load_points = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* For each glyph, parse the glyph charstring and extract */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* the advance width. */
|
|
|
|
for ( glyph_index = 0; glyph_index < face->root.num_glyphs;
|
|
|
|
glyph_index++ )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Byte* charstring;
|
|
|
|
FT_ULong charstring_len;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* now get load the unscaled outline */
|
|
|
|
error = T2_Access_Element( &cff->charstrings_index, glyph_index,
|
|
|
|
&charstring, &charstring_len );
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( !error )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-28 01:32:27 +02:00
|
|
|
T2_Prepare_Decoder( &decoder, glyph_index );
|
2000-06-07 22:04:34 +02:00
|
|
|
error = T2_Parse_CharStrings( &decoder, charstring, charstring_len );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Forget_Element( &cff->charstrings_index, &charstring );
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-07-01 00:24:36 +02:00
|
|
|
/* ignore the error if one has occurred -- skip to next glyph */
|
2000-06-07 22:04:34 +02:00
|
|
|
error = 0;
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
*max_advance = decoder.builder.advance.x;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
return T2_Err_Ok;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
#endif /* 0 */
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/********** *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** UNHINTED GLYPH LOADER *********/
|
|
|
|
/********** *********/
|
|
|
|
/********** The following code is in charge of loading a *********/
|
|
|
|
/********** single outline. It completely ignores hinting *********/
|
|
|
|
/********** and is used when FT_LOAD_NO_HINTING is set. *********/
|
|
|
|
/********** *********/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_LOCAL_DEF
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error T2_Load_Glyph( T2_GlyphSlot glyph,
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Size size,
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int glyph_index,
|
|
|
|
FT_Int load_flags )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Error error;
|
2000-06-14 01:21:00 +02:00
|
|
|
T2_Decoder decoder;
|
|
|
|
TT_Face face = (TT_Face)glyph->root.face;
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Bool hinting;
|
2000-06-25 06:49:19 +02:00
|
|
|
CFF_Font* cff = (CFF_Font*)face->extra.data;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
if ( load_flags & FT_LOAD_NO_RECURSE )
|
2000-06-07 22:04:34 +02:00
|
|
|
load_flags |= FT_LOAD_NO_SCALE | FT_LOAD_NO_HINTING;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
glyph->x_scale = 0x10000L;
|
|
|
|
glyph->y_scale = 0x10000L;
|
|
|
|
if ( size )
|
2000-06-29 23:48:58 +02:00
|
|
|
{
|
|
|
|
glyph->x_scale = size->metrics.x_scale;
|
|
|
|
glyph->y_scale = size->metrics.y_scale;
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
glyph->root.outline.n_points = 0;
|
|
|
|
glyph->root.outline.n_contours = 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
hinting = ( load_flags & FT_LOAD_NO_SCALE ) == 0 &&
|
|
|
|
( load_flags & FT_LOAD_NO_HINTING ) == 0;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-29 23:48:58 +02:00
|
|
|
glyph->root.format = ft_glyph_format_outline; /* by default */
|
2000-05-29 22:55:13 +02:00
|
|
|
|
|
|
|
{
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Byte* charstring;
|
|
|
|
FT_ULong charstring_len;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Init_Decoder( &decoder, face, size, glyph );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
decoder.builder.no_recurse =
|
2000-06-30 08:21:26 +02:00
|
|
|
(FT_Bool)( ( load_flags & FT_LOAD_NO_RECURSE ) != 0 );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* now load the unscaled outline */
|
|
|
|
error = T2_Access_Element( &cff->charstrings_index, glyph_index,
|
|
|
|
&charstring, &charstring_len );
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( !error )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-28 01:32:27 +02:00
|
|
|
T2_Prepare_Decoder( &decoder, glyph_index );
|
2000-06-07 22:04:34 +02:00
|
|
|
error = T2_Parse_CharStrings( &decoder, charstring, charstring_len );
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
T2_Forget_Element( &cff->charstrings_index, &charstring );
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* save new glyph tables */
|
|
|
|
T2_Done_Builder( &decoder.builder );
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-30 08:21:26 +02:00
|
|
|
/* Now, set the metrics -- this is rather simple, as */
|
2000-06-14 01:21:00 +02:00
|
|
|
/* the left side bearing is the xMin, and the top side */
|
|
|
|
/* bearing the yMax. */
|
|
|
|
if ( !error )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
/* for composite glyphs, return only left side bearing and */
|
|
|
|
/* advance width */
|
2000-06-29 23:48:58 +02:00
|
|
|
if ( glyph->root.format == ft_glyph_format_composite )
|
2000-05-29 22:55:13 +02:00
|
|
|
{
|
2000-06-07 22:04:34 +02:00
|
|
|
glyph->root.metrics.horiBearingX = decoder.builder.left_bearing.x;
|
|
|
|
glyph->root.metrics.horiAdvance = decoder.glyph_width;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-06-30 08:21:26 +02:00
|
|
|
FT_BBox cbox;
|
|
|
|
FT_Glyph_Metrics* metrics = &glyph->root.metrics;
|
2000-06-07 22:04:34 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* copy the _unscaled_ advance width */
|
2000-09-15 19:17:17 +02:00
|
|
|
metrics->horiAdvance = decoder.glyph_width;
|
|
|
|
glyph->root.linearHoriAdvance = decoder.glyph_width;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* make up vertical metrics */
|
|
|
|
metrics->vertBearingX = 0;
|
|
|
|
metrics->vertBearingY = 0;
|
|
|
|
metrics->vertAdvance = 0;
|
|
|
|
|
2000-09-15 19:17:17 +02:00
|
|
|
glyph->root.linearVertAdvance = 0;
|
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
glyph->root.format = ft_glyph_format_outline;
|
|
|
|
|
2000-06-29 23:48:58 +02:00
|
|
|
glyph->root.outline.flags = 0;
|
2000-06-07 22:04:34 +02:00
|
|
|
if ( size && size->metrics.y_ppem < 24 )
|
|
|
|
glyph->root.outline.flags |= ft_outline_high_precision;
|
|
|
|
|
|
|
|
glyph->root.outline.flags |= ft_outline_reverse_fill;
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( ( load_flags & FT_LOAD_NO_SCALE ) == 0 )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
/* scale the outline and the metrics */
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Int n;
|
- MAJOR INTERNAL REDESIGN:
A lot of internal modifications have been performed lately on the
source in order to provide the following enhancements:
- more generic module support:
The FT_Module type is now defined to represent a handle to a given
module. The file <freetype/ftmodule.h> contains the FT_Module_Class
definition, as well as the module-loading public API
The FT_Driver type is still defined, and still represents a pointer
to a font driver. Note that FT_Add_Driver is replaced by FT_Add_Module,
FT_Get_Driver by FT_Get_Module, etc..
- support for generic glyph image types:
The FT_Renderer type is a pointer to a module used to perform various
operations on glyph image.
Each renderer is capable of handling images in a single format
(e.g. ft_glyph_format_outline). Its functions are used to:
- transform an glyph image
- render a glyph image into a bitmap
- return the control box (dimensions) of a given glyph image
The scan converters "ftraster.c" and "ftgrays.c" have been moved
to the new directory "src/renderer", and are used to provide two
default renderer modules.
One corresponds to the "standard" scan-converter, the other to the
"smooth" one.
The current renderer can be set through the new function
FT_Set_Renderer.
The old raster-related function FT_Set_Raster, FT_Get_Raster and
FT_Set_Raster_Mode have now disappeared, in favor of the new:
FT_Get_Renderer
FT_Set_Renderer
see the file <freetype/ftrender.h> for more details..
These changes were necessary to properly support different scalable
formats in the future, like bi-color glyphs, etc..
- glyph loader object:
A new internal object, called a 'glyph loader' has been introduced
in the base layer. It is used by all scalable format font drivers
to load glyphs and composites.
This object has been created to reduce the code size of each driver,
as each one of them basically re-implemented its functionality.
See <freetype/internal/ftobjs.h> and the FT_GlyphLoader type for
more information..
- FT_GlyphSlot had new fields:
In order to support extended features (see below), the FT_GlyphSlot
structure has a few new fields:
linearHoriAdvance: this field gives the linearly scaled (i.e.
scaled but unhinted) advance width for the glyph,
expressed as a 16.16 fixed pixel value. This
is useful to perform WYSIWYG text.
linearVertAdvance: this field gives the linearly scaled advance
height for the glyph (relevant in vertical glyph
layouts only). This is useful to perform
WYSIWYG text.
Note that the two above field replace the removed "metrics2" field
in the glyph slot.
advance: this field is a vector that gives the transformed
advance for the glyph. By default, it corresponds
to the advance width, unless FT_LOAD_VERTICAL_LAYOUT
was specified when calling FT_Load_Glyph or FT_Load_Char
bitmap_left: this field gives the distance in integer pixels from
the current pen position to the left-most pixel of
a glyph image WHEN IT IS A BITMAP. It is only valid
when the "format" field is set to
"ft_glyph_format_bitmap", for example, after calling
the new function FT_Render_Glyph.
bitmap_top: this field gives the distance in integer pixels from
the current pen position (located on the baseline) to
the top-most pixel of the glyph image WHEN IT IS A
BITMAP. Positive values correspond to upwards Y.
loader: this is a new private field for the glyph slot. Client
applications should not touch it..
- support for transforms and direct rendering in FT_Load_Glyph:
Most of the functionality found in <freetype/ftglyph.h> has been
moved to the core library. Hence, the following:
- a transform can be specified for a face through FT_Set_Transform.
this transform is applied by FT_Load_Glyph to scalable glyph images
(i.e. NOT TO BITMAPS) before the function returns, unless the
bit flag FT_LOAD_IGNORE_TRANSFORM was set in the load flags..
- once a glyph image has been loaded, it can be directly converted to
a bitmap by using the new FT_Render_Glyph function. Note that this
function takes the glyph image from the glyph slot, and converts
it to a bitmap whose properties are returned in "face.glyph.bitmap",
"face.glyph.bitmap_left" and "face.glyph.bitmap_top". The original
native image might be lost after the conversion.
- when using the new bit flag FT_LOAD_RENDER, the FT_Load_Glyph
and FT_Load_Char functions will call FT_Render_Glyph automatically
when needed.
2000-06-22 02:17:42 +02:00
|
|
|
FT_Outline* cur = &glyph->root.outline;
|
2000-06-16 21:34:52 +02:00
|
|
|
FT_Vector* vec = cur->points;
|
|
|
|
FT_Fixed x_scale = glyph->x_scale;
|
|
|
|
FT_Fixed y_scale = glyph->y_scale;
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
|
|
|
|
/* First of all, scale the points */
|
|
|
|
for ( n = cur->n_points; n > 0; n--, vec++ )
|
|
|
|
{
|
|
|
|
vec->x = FT_MulFix( vec->x, x_scale );
|
|
|
|
vec->y = FT_MulFix( vec->y, y_scale );
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
FT_Outline_Get_CBox( &glyph->root.outline, &cbox );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* Then scale the metrics */
|
|
|
|
metrics->horiAdvance = FT_MulFix( metrics->horiAdvance, x_scale );
|
2000-07-01 00:24:36 +02:00
|
|
|
metrics->vertAdvance = FT_MulFix( metrics->vertAdvance, y_scale );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
metrics->vertBearingX = FT_MulFix( metrics->vertBearingX, x_scale );
|
|
|
|
metrics->vertBearingY = FT_MulFix( metrics->vertBearingY, y_scale );
|
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
#if 0
|
2000-06-30 08:21:26 +02:00
|
|
|
/* apply the font matrix */
|
2000-06-14 01:21:00 +02:00
|
|
|
FT_Outline_Transform( &glyph->root.outline, cff->font_matrix );
|
|
|
|
#endif
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* compute the other metrics */
|
|
|
|
FT_Outline_Get_CBox( &glyph->root.outline, &cbox );
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
/* grid fit the bounding box if necessary */
|
2000-06-14 01:21:00 +02:00
|
|
|
if ( hinting )
|
2000-06-07 22:04:34 +02:00
|
|
|
{
|
|
|
|
cbox.xMin &= -64;
|
|
|
|
cbox.yMin &= -64;
|
2000-06-30 08:21:26 +02:00
|
|
|
cbox.xMax = ( cbox.xMax + 63 ) & -64;
|
|
|
|
cbox.yMax = ( cbox.yMax + 63 ) & -64;
|
2000-06-07 22:04:34 +02:00
|
|
|
}
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
metrics->width = cbox.xMax - cbox.xMin;
|
|
|
|
metrics->height = cbox.yMax - cbox.yMin;
|
2000-05-29 22:55:13 +02:00
|
|
|
|
2000-06-07 22:04:34 +02:00
|
|
|
metrics->horiBearingX = cbox.xMin;
|
|
|
|
metrics->horiBearingY = cbox.yMax;
|
2000-05-29 22:55:13 +02:00
|
|
|
}
|
|
|
|
}
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-05-29 22:55:13 +02:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2000-06-14 01:21:00 +02:00
|
|
|
|
2000-05-29 22:55:13 +02:00
|
|
|
/* END */
|