aboutsummaryrefslogblamecommitdiffstats
path: root/libskrift_create_context.c
blob: c1732c0c54d81e4b2a525ee45e5e65955fbb76a3 (plain) (tree)
1
2
3
4
5
6
7
8
9


                                                         



                                                 
                                                    
                                                    
                                                                                                         
 




                                                       




                                                                                        










                                                                                               



                                                                         




                                                                                                                   
                                                       






                                           


                 
   
                                                                                                        
                                                                                    
 

                 

                       




                               
                        
                                                




                                       
                                                                                                 


                          
                                                     
                                             

                                        
                                             



                                             


                                                                           





                                                                                                      






                                                                                           
                                                              
                                                                                                            
                                                   
                                                   





                                                                                                           
                                                 

                                                                                   



                                                                                   










                                                                                                                         
                                                                               
         





















                                                                                 

                 
/* See LICENSE file for copyright and license details. */
#include "common.h"

#define FORCED_FLAGS (LIBSKRIFT_NO_LIGATURES   |\
                      LIBSKRIFT_NO_AUTOHINTING |\
                      LIBSKRIFT_NO_AUTOKERNING)

#define IMPLEMENTED_FLAGS (LIBSKRIFT_REMOVE_GAMMA |\
                           LIBSKRIFT_MIRROR_CHARS |\
                           FORCED_FLAGS) /* libschrift does not add gamma, so not handling is required */

#define TRANSFORMING_FLAGS (LIBSKRIFT_MIRROR_TEXT    |\
                            LIBSKRIFT_MIRROR_CHARS   |\
                            LIBSKRIFT_TRANSPOSE_TEXT |\
                            LIBSKRIFT_TRANSPOSE_CHARS)

#define COPY_ARRAY(DEST_STRUCT, SRC_STRUCT, FIELD)\
	memcpy((DEST_STRUCT).FIELD, (SRC_STRUCT).FIELD, sizeof((SRC_STRUCT).FIELD))

static const struct libskrift_rendering default_rendering = LIBSKRIFT_DEFAULT_RENDERING;

static void
multiply_matrices(const double a[restrict 6], const double b[restrict 6], double r[restrict 6])
{
	r[0] = a[0] * b[0] + a[1] * b[3];
	r[1] = a[0] * b[1] + a[1] * b[4];
	r[2] = a[0] * b[2] + a[1] * b[5] + a[2];
	r[3] = a[3] * b[0] + a[4] * b[3];
	r[4] = a[3] * b[1] + a[4] * b[4];
	r[5] = a[3] * b[2] + a[4] * b[5] + a[5];
}

static int
transformation_hook(void *hook_data, double advance, double transform[6])
{
	LIBSKRIFT_CONTEXT *ctx = hook_data;
	double schrift[6] = {transform[0], transform[2], transform[4],
	                     transform[1], transform[3], transform[5]};
	double m1[6] = {1, 0, 0, 0, 1, 0}, m2[6] = {1, 0, 0, 0, 1, 0};
	if (((ctx->rendering.flags / LIBSKRIFT_MIRROR_CHARS) ^ (ctx->rendering.flags / LIBSKRIFT_MIRROR_TEXT)) & 1)
		multiply_matrices((double []){-1, 0, advance / schrift[0], 0, 1, 0}, m2, m1);
	multiply_matrices(ctx->transformation, m1, m2);
	multiply_matrices(schrift, m2, m1);
	transform[0] = m1[0];
	transform[2] = m1[1];
	transform[4] = m1[2];
	transform[1] = m1[3];
	transform[3] = m1[4];
	transform[5] = m1[5];
	return 0;
}

int
libskrift_create_context(LIBSKRIFT_CONTEXT **ctxp, LIBSKRIFT_FONT **fonts, size_t nfonts, double height,
                         const struct libskrift_rendering *rendering, void *caching)
{
	size_t i;

	(void) caching;

	if (!nfonts) {
		errno = EINVAL;
		return -1;
	}

	if (rendering) {
		if (!rendering->grid_fineness) {
			errno = EINVAL;
			return -1;
		}
	}

	*ctxp = calloc(1, offsetof(LIBSKRIFT_CONTEXT, fonts) + nfonts * sizeof(*(*ctxp)->fonts));
	if (!*ctxp)
		return -1;

	(*ctxp)->schrift_ctx.font   = fonts[0]->font;
	(*ctxp)->schrift_ctx.yScale = height;
	(*ctxp)->x_advancement      = 1;
	(*ctxp)->y_advancement      = 0;
	(*ctxp)->nfonts             = nfonts;
	for (i = 0; i < nfonts; i++) {
		(*ctxp)->fonts[i] = fonts[i];
		fonts[i]->refcount += 1;
	}

	if (rendering) {
		memcpy(&(*ctxp)->rendering, rendering, sizeof(*rendering));
		COPY_ARRAY((*ctxp)->rendering, default_rendering, prestroke_transformation_rotation);
		COPY_ARRAY((*ctxp)->rendering, default_rendering, left_transformation);
		COPY_ARRAY((*ctxp)->rendering, default_rendering, right_transformation);
		COPY_ARRAY((*ctxp)->rendering, default_rendering, top_transformation);
		COPY_ARRAY((*ctxp)->rendering, default_rendering, bottom_transformation);
		COPY_ARRAY((*ctxp)->rendering, default_rendering, poststroke_transformation_rotation);
	} else {
		memcpy(&(*ctxp)->rendering, &default_rendering, sizeof(default_rendering));
	}

	(*ctxp)->rendering.struct_version      = LIBSKRIFT_RENDERING_STRUCT_VERSION;
	(*ctxp)->rendering.hinting             = LIBSKRIFT_NONE;
	(*ctxp)->rendering.flags              &= IMPLEMENTED_FLAGS;
	(*ctxp)->rendering.flags              |= FORCED_FLAGS;
	(*ctxp)->rendering.flags              |= LIBSKRIFT_REMOVE_GAMMA; /* libschrift does not add gamma */
	(*ctxp)->rendering.grid_fineness       = 1;
	(*ctxp)->rendering.kerning             = 0;

	if (!(*ctxp)->rendering.smoothing)
		(*ctxp)->rendering.smoothing = LIBSKRIFT_GREYSCALE;

	(*ctxp)->schrift_ctx.xScale = (*ctxp)->schrift_ctx.yScale;
	(*ctxp)->schrift_ctx.xScale *= (*ctxp)->rendering.horizontal_dpi / (*ctxp)->rendering.vertical_dpi;
	(*ctxp)->schrift_ctx.hook_data = (*ctxp);
	if (((*ctxp)->rendering.flags & TRANSFORMING_FLAGS) ||
	    fpclassify((*ctxp)->rendering.char_transformation[0] - 1) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.char_transformation[1]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.char_transformation[2]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.char_transformation[3]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.char_transformation[4] - 1) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.char_transformation[5]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[0] - 1) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[1]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[2]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[3]) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[4] - 1) != FP_ZERO ||
	    fpclassify((*ctxp)->rendering.text_transformation[5]) != FP_ZERO) {
		memcpy((*ctxp)->transformation, (*ctxp)->rendering.char_transformation, sizeof((*ctxp)->transformation));
		libskrift_add_transformation((*ctxp)->transformation, (*ctxp)->rendering.text_transformation);
		(*ctxp)->x_advancement = (*ctxp)->rendering.text_transformation[0];
		(*ctxp)->y_advancement = (*ctxp)->rendering.text_transformation[3];
		(*ctxp)->schrift_ctx.transformation_hook = transformation_hook;
	}

	if ((*ctxp)->rendering.smoothing == LIBSKRIFT_SUBPIXEL) {
		if ((*ctxp)->rendering.subpixel_order == LIBSKRIFT_RGB) {
			(*ctxp)->schrift_ctx.xScale   *= 3;
			(*ctxp)->subpixel_horizontally = 1;
		} else if ((*ctxp)->rendering.subpixel_order == LIBSKRIFT_BGR) {
			(*ctxp)->schrift_ctx.xScale   *= 3;
			(*ctxp)->subpixel_horizontally = 1;
			(*ctxp)->subpixel_bgr          = 1;
		} else if ((*ctxp)->rendering.subpixel_order == LIBSKRIFT_VRGB) {
			(*ctxp)->schrift_ctx.yScale   *= 3;
			(*ctxp)->subpixel_vertically   = 1;
		} else if ((*ctxp)->rendering.subpixel_order == LIBSKRIFT_VBGR) {
			(*ctxp)->schrift_ctx.yScale   *= 3;
			(*ctxp)->subpixel_vertically   = 1;
			(*ctxp)->subpixel_bgr          = 1;
		} else {
			(*ctxp)->rendering.subpixel_order = LIBSKRIFT_NONE;
			(*ctxp)->rendering.smoothing = LIBSKRIFT_GREYSCALE;
		}
	}

	return 0;
}