1/**************************************************************************
2 *
3 * Copyright 2010 LunarG, Inc.  All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the
7 * "Software"), to deal in the Software without restriction, including
8 * without limitation the rights to use, copy, modify, merge, publish,
9 * distribute, sub license, and/or sell copies of the Software, and to
10 * permit persons to whom the Software is furnished to do so, subject to
11 * the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the
14 * next paragraph) shall be included in all copies or substantial portions
15 * of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
20 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
21 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
22 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
23 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 *
25 **************************************************************************/
26
27#include "util/u_memory.h"
28#include "cso_cache/cso_hash.h"
29
30#include "text.h"
31#include "image.h"
32#include "path.h"
33
34#ifdef OPENVG_VERSION_1_1
35
36struct vg_font {
37   struct vg_object base;
38   struct cso_hash *glyphs;
39};
40
41struct vg_glyph {
42   struct vg_object *object; /* it could be NULL */
43   VGboolean is_hinted;
44   VGfloat glyph_origin[2];
45   VGfloat escapement[2];
46};
47
48static VGboolean del_glyph(struct vg_font *font,
49                           VGuint glyphIndex)
50{
51   struct vg_glyph *glyph;
52
53   glyph = (struct vg_glyph *)
54      cso_hash_take(font->glyphs, (unsigned) glyphIndex);
55   if (glyph)
56      FREE(glyph);
57
58   return (glyph != NULL);
59}
60
61static void add_glyph(struct vg_font *font,
62                      VGuint glyphIndex,
63                      struct vg_object *obj,
64                      VGboolean isHinted,
65                      const VGfloat glyphOrigin[2],
66                      const VGfloat escapement[2])
67{
68   struct vg_glyph *glyph;
69
70   /* remove the existing one */
71   del_glyph(font, glyphIndex);
72
73   glyph = CALLOC_STRUCT(vg_glyph);
74   glyph->object = obj;
75   glyph->is_hinted = isHinted;
76   memcpy(glyph->glyph_origin, glyphOrigin, sizeof(glyph->glyph_origin));
77   memcpy(glyph->escapement, escapement, sizeof(glyph->glyph_origin));
78
79   cso_hash_insert(font->glyphs, (unsigned) glyphIndex, glyph);
80}
81
82static struct vg_glyph *get_glyph(struct vg_font *font,
83                                  VGuint glyphIndex)
84{
85   struct cso_hash_iter iter;
86
87   iter = cso_hash_find(font->glyphs, (unsigned) glyphIndex);
88   return (struct vg_glyph *) cso_hash_iter_data(iter);
89}
90
91static void vg_render_glyph(struct vg_context *ctx,
92                            struct vg_glyph *glyph,
93                            VGbitfield paintModes,
94                            VGboolean allowAutoHinting)
95{
96   if (glyph->object && paintModes) {
97      struct vg_state *state = &ctx->state.vg;
98      struct matrix m;
99
100      m = state->glyph_user_to_surface_matrix;
101      matrix_translate(&m,
102            state->glyph_origin[0].f - glyph->glyph_origin[0],
103            state->glyph_origin[1].f - glyph->glyph_origin[1]);
104
105      if (glyph->object->type == VG_OBJECT_PATH) {
106         path_render((struct path *) glyph->object, paintModes, &m);
107      }
108      else {
109         assert(glyph->object->type == VG_OBJECT_IMAGE);
110         image_draw((struct vg_image *) glyph->object, &m);
111      }
112   }
113}
114
115static void vg_advance_glyph(struct vg_context *ctx,
116                             struct vg_glyph *glyph,
117                             VGfloat adjustment_x,
118                             VGfloat adjustment_y,
119                             VGboolean last)
120{
121   struct vg_value *glyph_origin = ctx->state.vg.glyph_origin;
122
123   glyph_origin[0].f += glyph->escapement[0] + adjustment_x;
124   glyph_origin[1].f += glyph->escapement[1] + adjustment_y;
125
126   if (last) {
127      glyph_origin[0].i = float_to_int_floor(glyph_origin[0].f);
128      glyph_origin[1].i = float_to_int_floor(glyph_origin[1].f);
129   }
130}
131
132struct vg_font *font_create(VGint glyphCapacityHint)
133{
134   struct vg_context *ctx = vg_current_context();
135   struct vg_font *font;
136
137   font = CALLOC_STRUCT(vg_font);
138   vg_init_object(&font->base, ctx, VG_OBJECT_FONT);
139   font->glyphs = cso_hash_create();
140
141   vg_context_add_object(ctx, &font->base);
142
143   return font;
144}
145
146void font_destroy(struct vg_font *font)
147{
148   struct vg_context *ctx = vg_current_context();
149   struct cso_hash_iter iter;
150
151   vg_context_remove_object(ctx, &font->base);
152
153   iter = cso_hash_first_node(font->glyphs);
154   while (!cso_hash_iter_is_null(iter)) {
155      struct vg_glyph *glyph = (struct vg_glyph *) cso_hash_iter_data(iter);
156      FREE(glyph);
157      iter = cso_hash_iter_next(iter);
158   }
159   cso_hash_delete(font->glyphs);
160
161   FREE(font);
162}
163
164void font_set_glyph_to_path(struct vg_font *font,
165                            VGuint glyphIndex,
166                            struct path *path,
167                            VGboolean isHinted,
168                            const VGfloat glyphOrigin[2],
169                            const VGfloat escapement[2])
170{
171   add_glyph(font, glyphIndex, (struct vg_object *) path,
172         isHinted, glyphOrigin, escapement);
173}
174
175void font_set_glyph_to_image(struct vg_font *font,
176                             VGuint glyphIndex,
177                             struct vg_image *image,
178                             const VGfloat glyphOrigin[2],
179                             const VGfloat escapement[2])
180{
181   add_glyph(font, glyphIndex, (struct vg_object *) image,
182         VG_TRUE, glyphOrigin, escapement);
183}
184
185void font_clear_glyph(struct vg_font *font,
186                      VGuint glyphIndex)
187{
188   if (!del_glyph(font, glyphIndex)) {
189      struct vg_context *ctx = vg_current_context();
190      vg_set_error(ctx, VG_ILLEGAL_ARGUMENT_ERROR);
191   }
192}
193
194void font_draw_glyph(struct vg_font *font,
195                     VGuint glyphIndex,
196                     VGbitfield paintModes,
197                     VGboolean allowAutoHinting)
198{
199   struct vg_context *ctx = vg_current_context();
200   struct vg_glyph *glyph;
201
202   glyph = get_glyph(font, glyphIndex);
203   if (!glyph) {
204      vg_set_error(ctx, VG_ILLEGAL_ARGUMENT_ERROR);
205      return;
206   }
207
208   vg_render_glyph(ctx, glyph, paintModes, allowAutoHinting);
209   vg_advance_glyph(ctx, glyph, 0.0f, 0.0f, VG_TRUE);
210}
211
212void font_draw_glyphs(struct vg_font *font,
213                      VGint glyphCount,
214                      const VGuint *glyphIndices,
215                      const VGfloat *adjustments_x,
216                      const VGfloat *adjustments_y,
217                      VGbitfield paintModes,
218                      VGboolean allowAutoHinting)
219{
220   struct vg_context *ctx = vg_current_context();
221   VGint i;
222
223   for (i = 0; i < glyphCount; ++i) {
224      if (!get_glyph(font, glyphIndices[i])) {
225         vg_set_error(ctx, VG_ILLEGAL_ARGUMENT_ERROR);
226         return;
227      }
228   }
229
230   for (i = 0; i < glyphCount; ++i) {
231      struct vg_glyph *glyph;
232      VGfloat adj_x, adj_y;
233
234      glyph = get_glyph(font, glyphIndices[i]);
235
236      vg_render_glyph(ctx, glyph, paintModes, allowAutoHinting);
237
238      adj_x = (adjustments_x) ? adjustments_x[i] : 0.0f;
239      adj_y = (adjustments_y) ? adjustments_y[i] : 0.0f;
240      vg_advance_glyph(ctx, glyph, adj_x, adj_y, (i == glyphCount - 1));
241   }
242}
243
244VGint font_num_glyphs(struct vg_font *font)
245{
246   return cso_hash_size(font->glyphs);
247}
248
249#endif /* OPENVG_VERSION_1_1 */
250