1 /* Window-specific OpenGL functions implementation. 2 * 3 * Copyright (c) 1999 Lionel Ulmer 4 * Copyright (c) 2005 Raphael Junqueira 5 * 6 * This library is free software; you can redistribute it and/or 7 * modify it under the terms of the GNU Lesser General Public 8 * License as published by the Free Software Foundation; either 9 * version 2.1 of the License, or (at your option) any later version. 10 * 11 * This library is distributed in the hope that it will be useful, 12 * but WITHOUT ANY WARRANTY; without even the implied warranty of 13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14 * Lesser General Public License for more details. 15 * 16 * You should have received a copy of the GNU Lesser General Public 17 * License along with this library; if not, write to the Free Software 18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 19 */ 20 21 #include "opengl32.h" 22 23 #include <math.h> 24 25 WINE_DEFAULT_DEBUG_CHANNEL(wgl); 26 27 /*********************************************************************** 28 * wglUseFontBitmaps_common 29 */ 30 static BOOL wglUseFontBitmaps_common( HDC hdc, DWORD first, DWORD count, DWORD listBase, BOOL unicode ) 31 { 32 const GLDISPATCHTABLE * funcs = IntGetCurrentDispatchTable(); 33 GLYPHMETRICS gm; 34 unsigned int glyph, size = 0; 35 void *bitmap = NULL, *gl_bitmap = NULL; 36 int org_alignment; 37 BOOL ret = TRUE; 38 39 funcs->GetIntegerv(GL_UNPACK_ALIGNMENT, &org_alignment); 40 funcs->PixelStorei(GL_UNPACK_ALIGNMENT, 4); 41 42 for (glyph = first; glyph < first + count; glyph++) { 43 static const MAT2 identity = { {0,1},{0,0},{0,0},{0,1} }; 44 unsigned int needed_size, height, width, width_int; 45 46 if (unicode) 47 needed_size = GetGlyphOutlineW(hdc, glyph, GGO_BITMAP, &gm, 0, NULL, &identity); 48 else 49 needed_size = GetGlyphOutlineA(hdc, glyph, GGO_BITMAP, &gm, 0, NULL, &identity); 50 51 TRACE("Glyph: %3d / List: %d size %d\n", glyph, listBase, needed_size); 52 if (needed_size == GDI_ERROR) { 53 ret = FALSE; 54 break; 55 } 56 57 if (needed_size > size) { 58 size = needed_size; 59 HeapFree(GetProcessHeap(), 0, bitmap); 60 HeapFree(GetProcessHeap(), 0, gl_bitmap); 61 bitmap = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, size); 62 gl_bitmap = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, size); 63 } 64 if (unicode) 65 ret = (GetGlyphOutlineW(hdc, glyph, GGO_BITMAP, &gm, size, bitmap, &identity) != GDI_ERROR); 66 else 67 ret = (GetGlyphOutlineA(hdc, glyph, GGO_BITMAP, &gm, size, bitmap, &identity) != GDI_ERROR); 68 if (!ret) break; 69 70 if (TRACE_ON(wgl)) { 71 unsigned int bitmask; 72 unsigned char *bitmap_ = bitmap; 73 74 TRACE(" - bbox: %d x %d\n", gm.gmBlackBoxX, gm.gmBlackBoxY); 75 TRACE(" - origin: (%d, %d)\n", gm.gmptGlyphOrigin.x, gm.gmptGlyphOrigin.y); 76 TRACE(" - increment: %d - %d\n", gm.gmCellIncX, gm.gmCellIncY); 77 if (needed_size != 0) { 78 TRACE(" - bitmap:\n"); 79 for (height = 0; height < gm.gmBlackBoxY; height++) { 80 TRACE(" "); 81 for (width = 0, bitmask = 0x80; width < gm.gmBlackBoxX; width++, bitmask >>= 1) { 82 if (bitmask == 0) { 83 bitmap_ += 1; 84 bitmask = 0x80; 85 } 86 if (*bitmap_ & bitmask) 87 TRACE("*"); 88 else 89 TRACE(" "); 90 } 91 bitmap_ += (4 - ((UINT_PTR)bitmap_ & 0x03)); 92 TRACE("\n"); 93 } 94 } 95 } 96 97 /* In OpenGL, the bitmap is drawn from the bottom to the top... So we need to invert the 98 * glyph for it to be drawn properly. 99 */ 100 if (needed_size != 0) { 101 width_int = (gm.gmBlackBoxX + 31) / 32; 102 for (height = 0; height < gm.gmBlackBoxY; height++) { 103 for (width = 0; width < width_int; width++) { 104 ((int *) gl_bitmap)[(gm.gmBlackBoxY - height - 1) * width_int + width] = 105 ((int *) bitmap)[height * width_int + width]; 106 } 107 } 108 } 109 110 funcs->NewList(listBase++, GL_COMPILE); 111 if (needed_size != 0) { 112 funcs->Bitmap(gm.gmBlackBoxX, gm.gmBlackBoxY, 113 0 - gm.gmptGlyphOrigin.x, (int) gm.gmBlackBoxY - gm.gmptGlyphOrigin.y, 114 gm.gmCellIncX, gm.gmCellIncY, 115 gl_bitmap); 116 } else { 117 /* This is the case of 'empty' glyphs like the space character */ 118 funcs->Bitmap(0, 0, 0, 0, gm.gmCellIncX, gm.gmCellIncY, NULL); 119 } 120 funcs->EndList(); 121 } 122 123 funcs->PixelStorei(GL_UNPACK_ALIGNMENT, org_alignment); 124 HeapFree(GetProcessHeap(), 0, bitmap); 125 HeapFree(GetProcessHeap(), 0, gl_bitmap); 126 return ret; 127 } 128 129 /*********************************************************************** 130 * wglUseFontBitmapsA (OPENGL32.@) 131 */ 132 BOOL WINAPI wglUseFontBitmapsA(HDC hdc, DWORD first, DWORD count, DWORD listBase) 133 { 134 return wglUseFontBitmaps_common( hdc, first, count, listBase, FALSE ); 135 } 136 137 /*********************************************************************** 138 * wglUseFontBitmapsW (OPENGL32.@) 139 */ 140 BOOL WINAPI wglUseFontBitmapsW(HDC hdc, DWORD first, DWORD count, DWORD listBase) 141 { 142 return wglUseFontBitmaps_common( hdc, first, count, listBase, TRUE ); 143 } 144 145 /* FIXME: should probably have a glu.h header */ 146 147 typedef struct GLUtesselator GLUtesselator; 148 typedef void (WINAPI *_GLUfuncptr)(void); 149 150 #define GLU_TESS_BEGIN 100100 151 #define GLU_TESS_VERTEX 100101 152 #define GLU_TESS_END 100102 153 154 static GLUtesselator * (WINAPI *pgluNewTess)(void); 155 static void (WINAPI *pgluDeleteTess)(GLUtesselator *tess); 156 static void (WINAPI *pgluTessNormal)(GLUtesselator *tess, GLdouble x, GLdouble y, GLdouble z); 157 static void (WINAPI *pgluTessBeginPolygon)(GLUtesselator *tess, void *polygon_data); 158 static void (WINAPI *pgluTessEndPolygon)(GLUtesselator *tess); 159 static void (WINAPI *pgluTessCallback)(GLUtesselator *tess, GLenum which, _GLUfuncptr fn); 160 static void (WINAPI *pgluTessBeginContour)(GLUtesselator *tess); 161 static void (WINAPI *pgluTessEndContour)(GLUtesselator *tess); 162 static void (WINAPI *pgluTessVertex)(GLUtesselator *tess, GLdouble *location, GLvoid* data); 163 164 static HMODULE load_libglu(void) 165 { 166 static const WCHAR glu32W[] = {'g','l','u','3','2','.','d','l','l',0}; 167 static int already_loaded; 168 static HMODULE module; 169 170 if (already_loaded) return module; 171 already_loaded = 1; 172 173 TRACE("Trying to load GLU library\n"); 174 module = LoadLibraryW( glu32W ); 175 if (!module) 176 { 177 WARN("Failed to load glu32\n"); 178 return NULL; 179 } 180 #define LOAD_FUNCPTR(f) p##f = (void *)GetProcAddress( module, #f ) 181 LOAD_FUNCPTR(gluNewTess); 182 LOAD_FUNCPTR(gluDeleteTess); 183 LOAD_FUNCPTR(gluTessBeginContour); 184 LOAD_FUNCPTR(gluTessNormal); 185 LOAD_FUNCPTR(gluTessBeginPolygon); 186 LOAD_FUNCPTR(gluTessCallback); 187 LOAD_FUNCPTR(gluTessEndContour); 188 LOAD_FUNCPTR(gluTessEndPolygon); 189 LOAD_FUNCPTR(gluTessVertex); 190 #undef LOAD_FUNCPTR 191 return module; 192 } 193 194 static void fixed_to_double(POINTFX fixed, UINT em_size, GLdouble vertex[3]) 195 { 196 vertex[0] = (fixed.x.value + (GLdouble)fixed.x.fract / (1 << 16)) / em_size; 197 vertex[1] = (fixed.y.value + (GLdouble)fixed.y.fract / (1 << 16)) / em_size; 198 vertex[2] = 0.0; 199 } 200 201 static void WINAPI tess_callback_vertex(GLvoid *vertex) 202 { 203 const GLDISPATCHTABLE * funcs = IntGetCurrentDispatchTable(); 204 GLdouble *dbl = vertex; 205 TRACE("%f, %f, %f\n", dbl[0], dbl[1], dbl[2]); 206 funcs->Vertex3dv(vertex); 207 } 208 209 static void WINAPI tess_callback_begin(GLenum which) 210 { 211 const GLDISPATCHTABLE * funcs = IntGetCurrentDispatchTable(); 212 TRACE("%d\n", which); 213 funcs->Begin(which); 214 } 215 216 static void WINAPI tess_callback_end(void) 217 { 218 const GLDISPATCHTABLE * funcs = IntGetCurrentDispatchTable(); 219 TRACE("\n"); 220 funcs->End(); 221 } 222 223 typedef struct _bezier_vector { 224 GLdouble x; 225 GLdouble y; 226 } bezier_vector; 227 228 static double bezier_deviation_squared(const bezier_vector *p) 229 { 230 bezier_vector deviation; 231 bezier_vector vertex; 232 bezier_vector base; 233 double base_length; 234 double dot; 235 236 vertex.x = (p[0].x + p[1].x*2 + p[2].x)/4 - p[0].x; 237 vertex.y = (p[0].y + p[1].y*2 + p[2].y)/4 - p[0].y; 238 239 base.x = p[2].x - p[0].x; 240 base.y = p[2].y - p[0].y; 241 242 base_length = sqrt(base.x*base.x + base.y*base.y); 243 base.x /= base_length; 244 base.y /= base_length; 245 246 dot = base.x*vertex.x + base.y*vertex.y; 247 dot = min(max(dot, 0.0), base_length); 248 base.x *= dot; 249 base.y *= dot; 250 251 deviation.x = vertex.x-base.x; 252 deviation.y = vertex.y-base.y; 253 254 return deviation.x*deviation.x + deviation.y*deviation.y; 255 } 256 257 static int bezier_approximate(const bezier_vector *p, bezier_vector *points, FLOAT deviation) 258 { 259 bezier_vector first_curve[3]; 260 bezier_vector second_curve[3]; 261 bezier_vector vertex; 262 int total_vertices; 263 264 if(bezier_deviation_squared(p) <= deviation*deviation) 265 { 266 if(points) 267 *points = p[2]; 268 return 1; 269 } 270 271 vertex.x = (p[0].x + p[1].x*2 + p[2].x)/4; 272 vertex.y = (p[0].y + p[1].y*2 + p[2].y)/4; 273 274 first_curve[0] = p[0]; 275 first_curve[1].x = (p[0].x + p[1].x)/2; 276 first_curve[1].y = (p[0].y + p[1].y)/2; 277 first_curve[2] = vertex; 278 279 second_curve[0] = vertex; 280 second_curve[1].x = (p[2].x + p[1].x)/2; 281 second_curve[1].y = (p[2].y + p[1].y)/2; 282 second_curve[2] = p[2]; 283 284 total_vertices = bezier_approximate(first_curve, points, deviation); 285 if(points) 286 points += total_vertices; 287 total_vertices += bezier_approximate(second_curve, points, deviation); 288 return total_vertices; 289 } 290 291 /*********************************************************************** 292 * wglUseFontOutlines_common 293 */ 294 static BOOL wglUseFontOutlines_common(HDC hdc, 295 DWORD first, 296 DWORD count, 297 DWORD listBase, 298 FLOAT deviation, 299 FLOAT extrusion, 300 int format, 301 LPGLYPHMETRICSFLOAT lpgmf, 302 BOOL unicode) 303 { 304 const GLDISPATCHTABLE * funcs = IntGetCurrentDispatchTable(); 305 UINT glyph; 306 const MAT2 identity = {{0,1},{0,0},{0,0},{0,1}}; 307 GLUtesselator *tess = NULL; 308 LOGFONTW lf; 309 HFONT old_font, unscaled_font; 310 UINT em_size = 1024; 311 RECT rc; 312 313 TRACE("(%p, %d, %d, %d, %f, %f, %d, %p, %s)\n", hdc, first, count, 314 listBase, deviation, extrusion, format, lpgmf, unicode ? "W" : "A"); 315 316 if(deviation <= 0.0) 317 deviation = 1.0/em_size; 318 319 if(format == WGL_FONT_POLYGONS) 320 { 321 if (!load_libglu()) 322 { 323 ERR("glu32 is required for this function but isn't available\n"); 324 return FALSE; 325 } 326 327 tess = pgluNewTess(); 328 if(!tess) return FALSE; 329 pgluTessCallback(tess, GLU_TESS_VERTEX, (_GLUfuncptr)tess_callback_vertex); 330 pgluTessCallback(tess, GLU_TESS_BEGIN, (_GLUfuncptr)tess_callback_begin); 331 pgluTessCallback(tess, GLU_TESS_END, tess_callback_end); 332 } 333 334 GetObjectW(GetCurrentObject(hdc, OBJ_FONT), sizeof(lf), &lf); 335 rc.left = rc.right = rc.bottom = 0; 336 rc.top = em_size; 337 DPtoLP(hdc, (POINT*)&rc, 2); 338 lf.lfHeight = -abs(rc.top - rc.bottom); 339 lf.lfOrientation = lf.lfEscapement = 0; 340 unscaled_font = CreateFontIndirectW(&lf); 341 old_font = SelectObject(hdc, unscaled_font); 342 343 for (glyph = first; glyph < first + count; glyph++) 344 { 345 DWORD needed; 346 GLYPHMETRICS gm; 347 BYTE *buf; 348 TTPOLYGONHEADER *pph; 349 TTPOLYCURVE *ppc; 350 GLdouble *vertices = NULL, *vertices_temp = NULL; 351 int vertex_total = -1; 352 353 if(unicode) 354 needed = GetGlyphOutlineW(hdc, glyph, GGO_NATIVE, &gm, 0, NULL, &identity); 355 else 356 needed = GetGlyphOutlineA(hdc, glyph, GGO_NATIVE, &gm, 0, NULL, &identity); 357 358 if(needed == GDI_ERROR) 359 goto error; 360 361 buf = HeapAlloc(GetProcessHeap(), 0, needed); 362 363 if(!buf) 364 goto error; 365 366 if(unicode) 367 GetGlyphOutlineW(hdc, glyph, GGO_NATIVE, &gm, needed, buf, &identity); 368 else 369 GetGlyphOutlineA(hdc, glyph, GGO_NATIVE, &gm, needed, buf, &identity); 370 371 TRACE("glyph %d\n", glyph); 372 373 if(lpgmf) 374 { 375 lpgmf->gmfBlackBoxX = (float)gm.gmBlackBoxX / em_size; 376 lpgmf->gmfBlackBoxY = (float)gm.gmBlackBoxY / em_size; 377 lpgmf->gmfptGlyphOrigin.x = (float)gm.gmptGlyphOrigin.x / em_size; 378 lpgmf->gmfptGlyphOrigin.y = (float)gm.gmptGlyphOrigin.y / em_size; 379 lpgmf->gmfCellIncX = (float)gm.gmCellIncX / em_size; 380 lpgmf->gmfCellIncY = (float)gm.gmCellIncY / em_size; 381 382 TRACE("%fx%f at %f,%f inc %f,%f\n", lpgmf->gmfBlackBoxX, lpgmf->gmfBlackBoxY, 383 lpgmf->gmfptGlyphOrigin.x, lpgmf->gmfptGlyphOrigin.y, lpgmf->gmfCellIncX, lpgmf->gmfCellIncY); 384 lpgmf++; 385 } 386 387 funcs->NewList(listBase++, GL_COMPILE); 388 funcs->FrontFace(GL_CCW); 389 if(format == WGL_FONT_POLYGONS) 390 { 391 funcs->Normal3d(0.0, 0.0, 1.0); 392 pgluTessNormal(tess, 0, 0, 1); 393 pgluTessBeginPolygon(tess, NULL); 394 } 395 396 while(!vertices) 397 { 398 if(vertex_total != -1) 399 vertices_temp = vertices = HeapAlloc(GetProcessHeap(), 0, vertex_total * 3 * sizeof(GLdouble)); 400 vertex_total = 0; 401 402 pph = (TTPOLYGONHEADER*)buf; 403 while((BYTE*)pph < buf + needed) 404 { 405 GLdouble previous[3]; 406 fixed_to_double(pph->pfxStart, em_size, previous); 407 408 if(vertices) 409 TRACE("\tstart %d, %d\n", pph->pfxStart.x.value, pph->pfxStart.y.value); 410 411 if(format == WGL_FONT_POLYGONS) 412 pgluTessBeginContour(tess); 413 else 414 funcs->Begin(GL_LINE_LOOP); 415 416 if(vertices) 417 { 418 fixed_to_double(pph->pfxStart, em_size, vertices); 419 if(format == WGL_FONT_POLYGONS) 420 pgluTessVertex(tess, vertices, vertices); 421 else 422 funcs->Vertex3d(vertices[0], vertices[1], vertices[2]); 423 vertices += 3; 424 } 425 vertex_total++; 426 427 ppc = (TTPOLYCURVE*)((char*)pph + sizeof(*pph)); 428 while((char*)ppc < (char*)pph + pph->cb) 429 { 430 int i, j; 431 int num; 432 433 switch(ppc->wType) { 434 case TT_PRIM_LINE: 435 for(i = 0; i < ppc->cpfx; i++) 436 { 437 if(vertices) 438 { 439 TRACE("\t\tline to %d, %d\n", 440 ppc->apfx[i].x.value, ppc->apfx[i].y.value); 441 fixed_to_double(ppc->apfx[i], em_size, vertices); 442 if(format == WGL_FONT_POLYGONS) 443 pgluTessVertex(tess, vertices, vertices); 444 else 445 funcs->Vertex3d(vertices[0], vertices[1], vertices[2]); 446 vertices += 3; 447 } 448 fixed_to_double(ppc->apfx[i], em_size, previous); 449 vertex_total++; 450 } 451 break; 452 453 case TT_PRIM_QSPLINE: 454 for(i = 0; i < ppc->cpfx-1; i++) 455 { 456 bezier_vector curve[3]; 457 bezier_vector *points; 458 GLdouble curve_vertex[3]; 459 460 if(vertices) 461 TRACE("\t\tcurve %d,%d %d,%d\n", 462 ppc->apfx[i].x.value, ppc->apfx[i].y.value, 463 ppc->apfx[i + 1].x.value, ppc->apfx[i + 1].y.value); 464 465 curve[0].x = previous[0]; 466 curve[0].y = previous[1]; 467 fixed_to_double(ppc->apfx[i], em_size, curve_vertex); 468 curve[1].x = curve_vertex[0]; 469 curve[1].y = curve_vertex[1]; 470 fixed_to_double(ppc->apfx[i + 1], em_size, curve_vertex); 471 curve[2].x = curve_vertex[0]; 472 curve[2].y = curve_vertex[1]; 473 if(i < ppc->cpfx-2) 474 { 475 curve[2].x = (curve[1].x + curve[2].x)/2; 476 curve[2].y = (curve[1].y + curve[2].y)/2; 477 } 478 num = bezier_approximate(curve, NULL, deviation); 479 points = HeapAlloc(GetProcessHeap(), 0, num*sizeof(bezier_vector)); 480 num = bezier_approximate(curve, points, deviation); 481 vertex_total += num; 482 if(vertices) 483 { 484 for(j=0; j<num; j++) 485 { 486 TRACE("\t\t\tvertex at %f,%f\n", points[j].x, points[j].y); 487 vertices[0] = points[j].x; 488 vertices[1] = points[j].y; 489 vertices[2] = 0.0; 490 if(format == WGL_FONT_POLYGONS) 491 pgluTessVertex(tess, vertices, vertices); 492 else 493 funcs->Vertex3d(vertices[0], vertices[1], vertices[2]); 494 vertices += 3; 495 } 496 } 497 HeapFree(GetProcessHeap(), 0, points); 498 previous[0] = curve[2].x; 499 previous[1] = curve[2].y; 500 } 501 break; 502 default: 503 ERR("\t\tcurve type = %d\n", ppc->wType); 504 if(format == WGL_FONT_POLYGONS) 505 pgluTessEndContour(tess); 506 else 507 funcs->End(); 508 goto error_in_list; 509 } 510 511 ppc = (TTPOLYCURVE*)((char*)ppc + sizeof(*ppc) + 512 (ppc->cpfx - 1) * sizeof(POINTFX)); 513 } 514 if(format == WGL_FONT_POLYGONS) 515 pgluTessEndContour(tess); 516 else 517 funcs->End(); 518 pph = (TTPOLYGONHEADER*)((char*)pph + pph->cb); 519 } 520 } 521 522 error_in_list: 523 if(format == WGL_FONT_POLYGONS) 524 pgluTessEndPolygon(tess); 525 funcs->Translated((GLdouble)gm.gmCellIncX / em_size, (GLdouble)gm.gmCellIncY / em_size, 0.0); 526 funcs->EndList(); 527 528 HeapFree(GetProcessHeap(), 0, buf); 529 530 if(vertices_temp) 531 HeapFree(GetProcessHeap(), 0, vertices_temp); 532 } 533 534 error: 535 DeleteObject(SelectObject(hdc, old_font)); 536 if(format == WGL_FONT_POLYGONS) 537 pgluDeleteTess(tess); 538 return TRUE; 539 540 } 541 542 /*********************************************************************** 543 * wglUseFontOutlinesA (OPENGL32.@) 544 */ 545 BOOL WINAPI wglUseFontOutlinesA(HDC hdc, 546 DWORD first, 547 DWORD count, 548 DWORD listBase, 549 FLOAT deviation, 550 FLOAT extrusion, 551 int format, 552 LPGLYPHMETRICSFLOAT lpgmf) 553 { 554 return wglUseFontOutlines_common(hdc, first, count, listBase, deviation, extrusion, format, lpgmf, FALSE); 555 } 556 557 /*********************************************************************** 558 * wglUseFontOutlinesW (OPENGL32.@) 559 */ 560 BOOL WINAPI wglUseFontOutlinesW(HDC hdc, 561 DWORD first, 562 DWORD count, 563 DWORD listBase, 564 FLOAT deviation, 565 FLOAT extrusion, 566 int format, 567 LPGLYPHMETRICSFLOAT lpgmf) 568 { 569 return wglUseFontOutlines_common(hdc, first, count, listBase, deviation, extrusion, format, lpgmf, TRUE); 570 } 571