… | |
… | |
35 | |
35 | |
36 | struct _PangoOpenGLRenderer |
36 | struct _PangoOpenGLRenderer |
37 | { |
37 | { |
38 | PangoRenderer parent_instance; |
38 | PangoRenderer parent_instance; |
39 | float r, g, b, a; // modulate |
39 | float r, g, b, a; // modulate |
40 | GLuint curtex; // current texture |
40 | int flags; |
|
|
41 | rc_t *rc; // rendercache |
|
|
42 | rc_key_t key; // current render key |
|
|
43 | rc_array_t *arr; |
41 | }; |
44 | }; |
42 | |
45 | |
43 | G_DEFINE_TYPE (PangoOpenGLRenderer, pango_opengl_renderer, PANGO_TYPE_RENDERER) |
46 | G_DEFINE_TYPE (PangoOpenGLRenderer, pango_opengl_renderer, PANGO_TYPE_RENDERER) |
44 | |
47 | |
45 | typedef struct |
48 | typedef struct |
… | |
… | |
57 | if (size > alloc) |
60 | if (size > alloc) |
58 | { |
61 | { |
59 | size = (size + 4095) & ~4095; |
62 | size = (size + 4095) & ~4095; |
60 | free (buffer); |
63 | free (buffer); |
61 | alloc = size; |
64 | alloc = size; |
62 | buffer = malloc (size); |
65 | buffer = (char *)malloc (size); |
63 | } |
66 | } |
64 | |
67 | |
65 | return buffer; |
68 | return buffer; |
66 | } |
69 | } |
67 | |
70 | |
… | |
… | |
87 | glyph->width = width; |
90 | glyph->width = width; |
88 | glyph->height = height; |
91 | glyph->height = height; |
89 | glyph->top = top; |
92 | glyph->top = top; |
90 | glyph->left = left; |
93 | glyph->left = left; |
91 | |
94 | |
92 | glyph->bitmap = temp_buffer (width * height); |
95 | glyph->bitmap = (uint8_t *)temp_buffer (width * height); |
93 | memset (glyph->bitmap, 0, glyph->stride * height); |
96 | memset (glyph->bitmap, 0, glyph->stride * height); |
94 | |
97 | |
95 | for (i = width; i--; ) |
98 | for (i = width; i--; ) |
96 | glyph->bitmap [i] = glyph->bitmap [i + (height - 1) * glyph->stride] = 0xff; |
99 | glyph->bitmap [i] = glyph->bitmap [i + (height - 1) * glyph->stride] = 0xff; |
97 | |
100 | |
… | |
… | |
100 | } |
103 | } |
101 | |
104 | |
102 | static void |
105 | static void |
103 | font_render_glyph (Glyph *glyph, PangoFont *font, int glyph_index) |
106 | font_render_glyph (Glyph *glyph, PangoFont *font, int glyph_index) |
104 | { |
107 | { |
|
|
108 | FT_Face face; |
|
|
109 | |
105 | if (glyph_index & PANGO_GLYPH_UNKNOWN_FLAG) |
110 | if (glyph_index & PANGO_GLYPH_UNKNOWN_FLAG) |
106 | { |
111 | { |
107 | PangoFontMetrics *metrics; |
112 | PangoFontMetrics *metrics; |
108 | |
113 | |
109 | if (!font) |
114 | if (!font) |
… | |
… | |
120 | pango_font_metrics_unref (metrics); |
125 | pango_font_metrics_unref (metrics); |
121 | |
126 | |
122 | return; |
127 | return; |
123 | } |
128 | } |
124 | |
129 | |
125 | FT_Face face = pango_opengl_font_get_face (font); |
130 | face = pango_opengl_font_get_face (font); |
126 | |
131 | |
127 | if (face) |
132 | if (face) |
128 | { |
133 | { |
129 | PangoOpenGLFont *glfont = (PangoOpenGLFont *)font; |
134 | PangoOpenGLFont *glfont = (PangoOpenGLFont *)font; |
130 | |
135 | |
… | |
… | |
144 | } |
149 | } |
145 | |
150 | |
146 | typedef struct glyph_info { |
151 | typedef struct glyph_info { |
147 | tc_area tex; |
152 | tc_area tex; |
148 | int left, top; |
153 | int left, top; |
|
|
154 | int generation; |
149 | } glyph_info; |
155 | } glyph_info; |
150 | |
156 | |
151 | static void |
157 | static void |
152 | free_glyph_info (glyph_info *g) |
158 | free_glyph_info (glyph_info *g) |
153 | { |
159 | { |
154 | tc_put (&g->tex); |
160 | tc_put (&g->tex); |
155 | g_slice_free (glyph_info, g); |
161 | g_slice_free (glyph_info, g); |
156 | } |
162 | } |
157 | |
163 | |
|
|
164 | static int apple_nvidia_bug_workaround; |
|
|
165 | |
|
|
166 | static void |
|
|
167 | apple_nvidia_bug (int enable) |
|
|
168 | { |
|
|
169 | apple_nvidia_bug_workaround = enable; |
|
|
170 | } |
|
|
171 | |
|
|
172 | static void |
|
|
173 | tex_update (int name, int x, int y, int w, int stride, int h, void *bm) |
|
|
174 | { |
|
|
175 | glBindTexture (GL_TEXTURE_2D, name); |
|
|
176 | |
|
|
177 | if (!apple_nvidia_bug_workaround) |
|
|
178 | { |
|
|
179 | glPixelStorei (GL_UNPACK_ROW_LENGTH, stride); |
|
|
180 | /*glPixelStorei (GL_UNPACK_ALIGNMENT, 1); expected cfplus default */ |
|
|
181 | glTexSubImage2D (GL_TEXTURE_2D, 0, x, y, w, h, GL_ALPHA, GL_UNSIGNED_BYTE, bm); |
|
|
182 | /*glPixelStorei (GL_UNPACK_ALIGNMENT, 4);*/ |
|
|
183 | glPixelStorei (GL_UNPACK_ROW_LENGTH, 0); |
|
|
184 | } |
|
|
185 | else |
|
|
186 | { |
|
|
187 | /* starting with 10.5.5 (or 10.5.6), pple's nvidia driver corrupts textures */ |
|
|
188 | /* when glTexSubImage is used, so do it the horribly slow way, */ |
|
|
189 | /* reading/patching/uploading the full texture one each change */ |
|
|
190 | int r; |
|
|
191 | |
|
|
192 | glGetTexImage (GL_TEXTURE_2D, 0, GL_ALPHA, GL_UNSIGNED_BYTE, tc_temptile); |
|
|
193 | |
|
|
194 | for (r = 0; r < h; ++r) |
|
|
195 | memcpy (tc_temptile + (y + r) * TC_WIDTH + x, (char *)bm + r * stride, w); |
|
|
196 | |
|
|
197 | glTexImage2D (GL_TEXTURE_2D, 0, GL_ALPHA, TC_WIDTH, TC_HEIGHT, 0, GL_ALPHA, GL_UNSIGNED_BYTE, tc_temptile); |
|
|
198 | } |
|
|
199 | } |
|
|
200 | |
158 | static void |
201 | static void |
159 | draw_glyph (PangoRenderer *renderer_, PangoFont *font, PangoGlyph glyph, double x, double y) |
202 | draw_glyph (PangoRenderer *renderer_, PangoFont *font, PangoGlyph glyph, double x, double y) |
160 | { |
203 | { |
161 | PangoOpenGLRenderer *renderer = PANGO_OPENGL_RENDERER (renderer_); |
204 | PangoOpenGLRenderer *renderer = PANGO_OPENGL_RENDERER (renderer_); |
|
|
205 | glyph_info *g; |
162 | |
206 | |
163 | if (glyph & PANGO_GLYPH_UNKNOWN_FLAG) |
207 | if (glyph & PANGO_GLYPH_UNKNOWN_FLAG) |
164 | { |
208 | { |
165 | glyph = pango_opengl_get_unknown_glyph (font); |
209 | glyph = pango_opengl_get_unknown_glyph (font); |
166 | |
210 | |
167 | if (glyph == PANGO_GLYPH_EMPTY) |
211 | if (glyph == PANGO_GLYPH_EMPTY) |
168 | glyph = PANGO_GLYPH_UNKNOWN_FLAG; |
212 | glyph = PANGO_GLYPH_UNKNOWN_FLAG; |
169 | } |
213 | } |
170 | |
214 | |
171 | glyph_info *g = _pango_opengl_font_get_cache_glyph_data (font, glyph); |
215 | g = (glyph_info *)_pango_opengl_font_get_cache_glyph_data (font, glyph); |
172 | |
216 | |
173 | if (!g) |
217 | if (!g || g->generation != tc_generation) |
174 | { |
218 | { |
175 | Glyph bm; |
219 | Glyph bm; |
176 | font_render_glyph (&bm, font, glyph); |
220 | font_render_glyph (&bm, font, glyph); |
177 | |
221 | |
|
|
222 | if (!g) |
|
|
223 | { |
178 | g = g_slice_new (glyph_info); |
224 | g = g_slice_new (glyph_info); |
179 | |
225 | |
180 | tc_get (&g->tex, bm.width, bm.height); |
226 | _pango_opengl_font_set_glyph_cache_destroy (font, (GDestroyNotify)free_glyph_info); |
|
|
227 | _pango_opengl_font_set_cache_glyph_data (font, glyph, g); |
|
|
228 | } |
|
|
229 | |
|
|
230 | g->generation = tc_generation; |
181 | |
231 | |
182 | g->left = bm.left; |
232 | g->left = bm.left; |
183 | g->top = bm.top; |
233 | g->top = bm.top; |
184 | |
234 | |
185 | if (renderer->curtex) |
235 | tc_get (&g->tex, bm.width, bm.height); |
186 | { |
|
|
187 | glEnd (); |
|
|
188 | renderer->curtex = 0; |
|
|
189 | } |
|
|
190 | |
236 | |
191 | glBindTexture (GL_TEXTURE_2D, g->tex.name); |
237 | if (bm.width && bm.height) |
192 | glPixelStorei (GL_UNPACK_ROW_LENGTH, bm.stride); |
238 | tex_update (g->tex.name, g->tex.x, g->tex.y, bm.width, bm.stride, bm.height, bm.bitmap); |
193 | glPixelStorei (GL_UNPACK_ALIGNMENT, 1); |
|
|
194 | glTexSubImage2D (GL_TEXTURE_2D, 0, g->tex.x, g->tex.y, bm.width, bm.height, GL_ALPHA, GL_UNSIGNED_BYTE, bm.bitmap); |
|
|
195 | glPixelStorei (GL_UNPACK_ROW_LENGTH, 0); |
|
|
196 | glPixelStorei (GL_UNPACK_ALIGNMENT, 4); |
|
|
197 | |
|
|
198 | _pango_opengl_font_set_glyph_cache_destroy (font, (GDestroyNotify)free_glyph_info); |
|
|
199 | _pango_opengl_font_set_cache_glyph_data (font, glyph, g); |
|
|
200 | } |
239 | } |
201 | |
240 | |
202 | x += g->left; |
241 | x += g->left; |
203 | y -= g->top; |
242 | y -= g->top; |
204 | |
243 | |
205 | float x1 = g->tex.x * (1. / TC_WIDTH ); |
|
|
206 | float y1 = g->tex.y * (1. / TC_HEIGHT); |
|
|
207 | float x2 = g->tex.w * (1. / TC_WIDTH ) + x1; |
|
|
208 | float y2 = g->tex.h * (1. / TC_HEIGHT) + y1; |
|
|
209 | |
|
|
210 | if (g->tex.name != renderer->curtex) |
244 | if (g->tex.name != renderer->key.texname) |
211 | { |
245 | { |
212 | if (renderer->curtex) |
|
|
213 | glEnd (); |
|
|
214 | |
|
|
215 | glBindTexture (GL_TEXTURE_2D, g->tex.name); |
|
|
216 | renderer->curtex = g->tex.name; |
246 | renderer->key.texname = g->tex.name; |
217 | glBegin (GL_QUADS); |
247 | renderer->arr = rc_array (renderer->rc, &renderer->key); |
218 | } |
248 | } |
219 | |
249 | |
220 | glTexCoord2f (x1, y1); glVertex2i (x , y ); |
250 | rc_glyph (renderer->arr, g->tex.x, g->tex.y, g->tex.w, g->tex.h, x, y); |
221 | glTexCoord2f (x2, y1); glVertex2i (x + g->tex.w, y ); |
|
|
222 | glTexCoord2f (x2, y2); glVertex2i (x + g->tex.w, y + g->tex.h); |
|
|
223 | glTexCoord2f (x1, y2); glVertex2i (x , y + g->tex.h); |
|
|
224 | } |
251 | } |
225 | |
252 | |
226 | static void |
253 | static void |
227 | draw_trapezoid (PangoRenderer *renderer_, |
254 | draw_trapezoid (PangoRenderer *renderer_, |
228 | PangoRenderPart part, |
255 | PangoRenderPart part, |
… | |
… | |
232 | double y2, |
259 | double y2, |
233 | double x12, |
260 | double x12, |
234 | double x22) |
261 | double x22) |
235 | { |
262 | { |
236 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
263 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
|
|
264 | rc_key_t key = renderer->key; |
|
|
265 | rc_array_t *arr; |
237 | |
266 | |
238 | if (renderer->curtex) |
267 | key.mode = GL_QUADS; |
239 | { |
268 | key.format = GL_V2F; |
240 | glEnd (); |
269 | key.texname = 0; |
241 | renderer->curtex = 0; |
|
|
242 | } |
|
|
243 | |
270 | |
244 | glDisable (GL_TEXTURE_2D); |
271 | arr = rc_array (renderer->rc, &key); |
245 | |
272 | |
246 | glBegin (GL_QUADS); |
273 | rc_v2f (arr, x11, y1); |
247 | glVertex2d (x11, y1); |
274 | rc_v2f (arr, x21, y1); |
248 | glVertex2d (x12, y1); |
275 | rc_v2f (arr, x22, y2); |
249 | glVertex2d (x22, y2); |
276 | rc_v2f (arr, x12, y2); |
250 | glVertex2d (x21, y2); |
|
|
251 | glEnd (); |
|
|
252 | |
|
|
253 | glEnable (GL_TEXTURE_2D); |
|
|
254 | } |
277 | } |
255 | |
278 | |
256 | void |
279 | void |
257 | pango_opengl_render_layout_subpixel (PangoLayout *layout, |
280 | pango_opengl_render_layout_subpixel (PangoLayout *layout, |
|
|
281 | rc_t *rc, |
258 | int x, int y, |
282 | int x, int y, |
259 | float r, float g, float b, float a) |
283 | float r, float g, float b, float a, |
|
|
284 | int flags) |
260 | { |
285 | { |
261 | PangoContext *context; |
286 | PangoContext *context; |
262 | PangoFontMap *fontmap; |
287 | PangoFontMap *fontmap; |
263 | PangoRenderer *renderer; |
288 | PangoRenderer *renderer; |
|
|
289 | PangoOpenGLRenderer *gl; |
264 | |
290 | |
265 | context = pango_layout_get_context (layout); |
291 | context = pango_layout_get_context (layout); |
266 | fontmap = pango_context_get_font_map (context); |
292 | fontmap = pango_context_get_font_map (context); |
267 | renderer = _pango_opengl_font_map_get_renderer (PANGO_OPENGL_FONT_MAP (fontmap)); |
293 | renderer = _pango_opengl_font_map_get_renderer (PANGO_OPENGL_FONT_MAP (fontmap)); |
268 | |
|
|
269 | PANGO_OPENGL_RENDERER (renderer)->r = r; |
294 | gl = PANGO_OPENGL_RENDERER (renderer); |
270 | PANGO_OPENGL_RENDERER (renderer)->g = g; |
295 | |
271 | PANGO_OPENGL_RENDERER (renderer)->b = b; |
296 | gl->rc = rc; |
272 | PANGO_OPENGL_RENDERER (renderer)->a = a; |
297 | gl->r = r; |
|
|
298 | gl->g = g; |
|
|
299 | gl->b = b; |
|
|
300 | gl->a = a; |
|
|
301 | gl->flags = flags; |
273 | |
302 | |
274 | pango_renderer_draw_layout (renderer, layout, x, y); |
303 | pango_renderer_draw_layout (renderer, layout, x, y); |
275 | } |
304 | } |
276 | |
305 | |
277 | void |
306 | void |
278 | pango_opengl_render_layout (PangoLayout *layout, |
307 | pango_opengl_render_layout (PangoLayout *layout, |
|
|
308 | rc_t *rc, |
279 | int x, int y, |
309 | int x, int y, |
280 | float r, float g, float b, float a) |
310 | float r, float g, float b, float a, |
|
|
311 | int flags) |
281 | { |
312 | { |
282 | pango_opengl_render_layout_subpixel (layout, x * PANGO_SCALE, y * PANGO_SCALE, r, g, b, a); |
313 | pango_opengl_render_layout_subpixel ( |
|
|
314 | layout, rc, x * PANGO_SCALE, y * PANGO_SCALE, r, g, b, a, flags |
|
|
315 | ); |
283 | } |
316 | } |
284 | |
317 | |
285 | static void |
318 | static void |
286 | pango_opengl_renderer_init (PangoOpenGLRenderer *renderer) |
319 | pango_opengl_renderer_init (PangoOpenGLRenderer *renderer) |
287 | { |
320 | { |
|
|
321 | memset (&renderer->key, 0, sizeof (rc_key_t)); |
|
|
322 | |
288 | renderer->r = 1.; |
323 | renderer->r = 1.; |
289 | renderer->g = 1.; |
324 | renderer->g = 1.; |
290 | renderer->b = 1.; |
325 | renderer->b = 1.; |
291 | renderer->a = 1.; |
326 | renderer->a = 1.; |
292 | } |
327 | } |
293 | |
328 | |
294 | static void |
329 | static void |
295 | prepare_run (PangoRenderer *renderer, PangoLayoutRun *run) |
330 | prepare_run (PangoRenderer *renderer, PangoLayoutRun *run) |
296 | { |
331 | { |
297 | PangoOpenGLRenderer *glrenderer = (PangoOpenGLRenderer *)renderer; |
332 | PangoOpenGLRenderer *gl = (PangoOpenGLRenderer *)renderer; |
298 | PangoColor *fg = 0; |
333 | PangoColor *fg = 0; |
299 | GSList *l; |
334 | GSList *l; |
|
|
335 | unsigned char r, g, b, a; |
300 | |
336 | |
301 | renderer->underline = PANGO_UNDERLINE_NONE; |
337 | renderer->underline = PANGO_UNDERLINE_NONE; |
302 | renderer->strikethrough = FALSE; |
338 | renderer->strikethrough = FALSE; |
303 | |
339 | |
|
|
340 | gl->key.mode = GL_QUADS; |
|
|
341 | gl->key.format = 0; // glyphs |
|
|
342 | gl->key.texname = 0; |
|
|
343 | |
304 | for (l = run->item->analysis.extra_attrs; l; l = l->next) |
344 | for (l = run->item->analysis.extra_attrs; l; l = l->next) |
305 | { |
345 | { |
306 | PangoAttribute *attr = l->data; |
346 | PangoAttribute *attr = (PangoAttribute *)l->data; |
307 | |
347 | |
308 | switch (attr->klass->type) |
348 | switch (attr->klass->type) |
309 | { |
349 | { |
310 | case PANGO_ATTR_UNDERLINE: |
350 | case PANGO_ATTR_UNDERLINE: |
311 | renderer->underline = ((PangoAttrInt *)attr)->value; |
351 | renderer->underline = (PangoUnderline)((PangoAttrInt *)attr)->value; |
312 | break; |
352 | break; |
313 | |
353 | |
314 | case PANGO_ATTR_STRIKETHROUGH: |
354 | case PANGO_ATTR_STRIKETHROUGH: |
315 | renderer->strikethrough = ((PangoAttrInt *)attr)->value; |
355 | renderer->strikethrough = ((PangoAttrInt *)attr)->value; |
316 | break; |
356 | break; |
… | |
… | |
323 | break; |
363 | break; |
324 | } |
364 | } |
325 | } |
365 | } |
326 | |
366 | |
327 | if (fg) |
367 | if (fg) |
328 | glColor4f (fg->red / 65535., fg->green / 65535., fg->blue / 65535., glrenderer->a); |
368 | { |
|
|
369 | r = fg->red * (255.f / 65535.f); |
|
|
370 | g = fg->green * (255.f / 65535.f); |
|
|
371 | b = fg->blue * (255.f / 65535.f); |
|
|
372 | } |
329 | else |
373 | else |
330 | glColor4f (glrenderer->r, glrenderer->g, glrenderer->b, glrenderer->a); |
374 | { |
|
|
375 | r = gl->r * 255.f; |
|
|
376 | g = gl->g * 255.f; |
|
|
377 | b = gl->b * 255.f; |
|
|
378 | } |
|
|
379 | |
|
|
380 | a = gl->a * 255.f; |
|
|
381 | |
|
|
382 | if (gl->flags & FLAG_INVERSE) |
|
|
383 | { |
|
|
384 | r ^= 0xffU; |
|
|
385 | g ^= 0xffU; |
|
|
386 | b ^= 0xffU; |
|
|
387 | } |
|
|
388 | |
|
|
389 | gl->key.r = r; |
|
|
390 | gl->key.g = g; |
|
|
391 | gl->key.b = b; |
|
|
392 | gl->key.a = a; |
331 | } |
393 | } |
332 | |
394 | |
333 | static void |
395 | static void |
334 | draw_begin (PangoRenderer *renderer_) |
396 | draw_begin (PangoRenderer *renderer_) |
335 | { |
397 | { |
336 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
398 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
337 | |
|
|
338 | renderer->curtex = 0; |
|
|
339 | |
|
|
340 | glEnable (GL_TEXTURE_2D); |
|
|
341 | glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE); |
|
|
342 | glEnable (GL_BLEND); |
|
|
343 | gl_BlendFuncSeparate (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, |
|
|
344 | GL_ONE , GL_ONE_MINUS_SRC_ALPHA); |
|
|
345 | glEnable (GL_ALPHA_TEST); |
|
|
346 | glAlphaFunc (GL_GREATER, 0.01f); |
|
|
347 | } |
399 | } |
348 | |
400 | |
349 | static void |
401 | static void |
350 | draw_end (PangoRenderer *renderer_) |
402 | draw_end (PangoRenderer *renderer_) |
351 | { |
403 | { |
352 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
404 | PangoOpenGLRenderer *renderer = (PangoOpenGLRenderer *)renderer_; |
353 | |
|
|
354 | if (renderer->curtex) |
|
|
355 | glEnd (); |
|
|
356 | |
|
|
357 | glDisable (GL_ALPHA_TEST); |
|
|
358 | glDisable (GL_BLEND); |
|
|
359 | glDisable (GL_TEXTURE_2D); |
|
|
360 | } |
405 | } |
361 | |
406 | |
362 | static void |
407 | static void |
363 | pango_opengl_renderer_class_init (PangoOpenGLRendererClass *klass) |
408 | pango_opengl_renderer_class_init (PangoOpenGLRendererClass *klass) |
364 | { |
409 | { |