DRC | 2ff39b8 | 2011-07-28 08:38:59 +0000 | [diff] [blame] | 1 | // |
| 2 | // "$Id: fl_font_win32.cxx 8644 2011-05-10 15:37:05Z ianmacarthur $" |
| 3 | // |
| 4 | // WIN32 font selection routines for the Fast Light Tool Kit (FLTK). |
| 5 | // |
| 6 | // Copyright 1998-2011 by Bill Spitzak and others. |
| 7 | // |
| 8 | // This library is free software; you can redistribute it and/or |
| 9 | // modify it under the terms of the GNU Library General Public |
| 10 | // License as published by the Free Software Foundation; either |
| 11 | // version 2 of the License, or (at your option) any later version. |
| 12 | // |
| 13 | // This library is distributed in the hope that it will be useful, |
| 14 | // but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 15 | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 16 | // Library General Public License for more details. |
| 17 | // |
| 18 | // You should have received a copy of the GNU Library General Public |
| 19 | // License along with this library; if not, write to the Free Software |
| 20 | // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 |
| 21 | // USA. |
| 22 | // |
| 23 | // Please report all bugs and problems on the following page: |
| 24 | // |
| 25 | // http://www.fltk.org/str.php |
| 26 | // |
| 27 | |
| 28 | #include <FL/Fl_Printer.H> |
| 29 | |
| 30 | static int fl_angle_ = 0; |
| 31 | |
| 32 | #ifndef FL_DOXYGEN |
| 33 | Fl_Font_Descriptor::Fl_Font_Descriptor(const char* name, Fl_Fontsize fsize) { |
| 34 | int weight = FW_NORMAL; |
| 35 | int italic = 0; |
| 36 | switch (*name++) { |
| 37 | case 'I': italic = 1; break; |
| 38 | case 'P': italic = 1; |
| 39 | case 'B': weight = FW_BOLD; break; |
| 40 | case ' ': break; |
| 41 | default: name--; |
| 42 | } |
| 43 | fid = CreateFont( |
| 44 | -fsize, // negative makes it use "char size" |
| 45 | 0, // logical average character width |
| 46 | fl_angle_*10, // angle of escapement |
| 47 | fl_angle_*10, // base-line orientation angle |
| 48 | weight, |
| 49 | italic, |
| 50 | FALSE, // underline attribute flag |
| 51 | FALSE, // strikeout attribute flag |
| 52 | DEFAULT_CHARSET, // character set identifier |
| 53 | OUT_DEFAULT_PRECIS, // output precision |
| 54 | CLIP_DEFAULT_PRECIS,// clipping precision |
| 55 | DEFAULT_QUALITY, // output quality |
| 56 | DEFAULT_PITCH, // pitch and family |
| 57 | name // pointer to typeface name string |
| 58 | ); |
| 59 | angle = fl_angle_; |
| 60 | if (!fl_gc) fl_GetDC(0); |
| 61 | SelectObject(fl_gc, fid); |
| 62 | GetTextMetrics(fl_gc, &metr); |
| 63 | // BOOL ret = GetCharWidthFloat(fl_gc, metr.tmFirstChar, metr.tmLastChar, font->width+metr.tmFirstChar); |
| 64 | // ...would be the right call, but is not implemented into Window95! (WinNT?) |
| 65 | //GetCharWidth(fl_gc, 0, 255, width); |
| 66 | int i; |
| 67 | for (i = 0; i < 64; i++) width[i] = NULL; |
| 68 | #if HAVE_GL |
| 69 | listbase = 0; |
| 70 | for (i = 0; i < 64; i++) glok[i] = 0; |
| 71 | #endif |
| 72 | size = fsize; |
| 73 | } |
| 74 | |
| 75 | Fl_Font_Descriptor::~Fl_Font_Descriptor() { |
| 76 | #if HAVE_GL |
| 77 | // Delete list created by gl_draw(). This is not done by this code |
| 78 | // as it will link in GL unnecessarily. There should be some kind |
| 79 | // of "free" routine pointer, or a subclass? |
| 80 | // if (listbase) { |
| 81 | // int base = font->min_char_or_byte2; |
| 82 | // int size = font->max_char_or_byte2-base+1; |
| 83 | // int base = 0; int size = 256; |
| 84 | // glDeleteLists(listbase+base,size); |
| 85 | // } |
| 86 | #endif |
| 87 | if (this == fl_graphics_driver->font_descriptor()) fl_graphics_driver->font_descriptor(NULL); |
| 88 | DeleteObject(fid); |
| 89 | int i; |
| 90 | for (i = 0; i < 64; i++) free(width[i]); |
| 91 | } |
| 92 | |
| 93 | //////////////////////////////////////////////////////////////// |
| 94 | |
| 95 | // WARNING: if you add to this table, you must redefine FL_FREE_FONT |
| 96 | // in Enumerations.H & recompile!! |
| 97 | static Fl_Fontdesc built_in_table[] = { |
| 98 | {" Arial"}, |
| 99 | {"BArial"}, |
| 100 | {"IArial"}, |
| 101 | {"PArial"}, |
| 102 | {" Courier New"}, |
| 103 | {"BCourier New"}, |
| 104 | {"ICourier New"}, |
| 105 | {"PCourier New"}, |
| 106 | {" Times New Roman"}, |
| 107 | {"BTimes New Roman"}, |
| 108 | {"ITimes New Roman"}, |
| 109 | {"PTimes New Roman"}, |
| 110 | {" Symbol"}, |
| 111 | {" Terminal"}, |
| 112 | {"BTerminal"}, |
| 113 | {" Wingdings"}, |
| 114 | }; |
| 115 | |
| 116 | Fl_Fontdesc* fl_fonts = built_in_table; |
| 117 | |
| 118 | static Fl_Font_Descriptor* find(Fl_Font fnum, Fl_Fontsize size, int angle) { |
| 119 | Fl_Fontdesc* s = fl_fonts+fnum; |
| 120 | if (!s->name) s = fl_fonts; // use 0 if fnum undefined |
| 121 | Fl_Font_Descriptor* f; |
| 122 | for (f = s->first; f; f = f->next) |
| 123 | if (f->size == size && f->angle == angle) return f; |
| 124 | f = new Fl_Font_Descriptor(s->name, size); |
| 125 | f->next = s->first; |
| 126 | s->first = f; |
| 127 | return f; |
| 128 | } |
| 129 | |
| 130 | //////////////////////////////////////////////////////////////// |
| 131 | // Public interface: |
| 132 | |
| 133 | static void fl_font(Fl_Graphics_Driver *driver, Fl_Font fnum, Fl_Fontsize size, int angle) { |
| 134 | if (fnum==-1) { // just make sure that we will load a new font next time |
| 135 | fl_angle_ = 0; |
| 136 | driver->Fl_Graphics_Driver::font(0, 0); |
| 137 | return; |
| 138 | } |
| 139 | if (fnum == driver->Fl_Graphics_Driver::font() && size == driver->size() && angle == fl_angle_) return; |
| 140 | fl_angle_ = angle; |
| 141 | driver->Fl_Graphics_Driver::font(fnum, size); |
| 142 | driver->font_descriptor( find(fnum, size, angle) ); |
| 143 | } |
| 144 | |
| 145 | void Fl_GDI_Graphics_Driver::font(Fl_Font fnum, Fl_Fontsize size) { |
| 146 | fl_font(this, fnum, size, 0); |
| 147 | } |
| 148 | |
| 149 | int Fl_GDI_Graphics_Driver::height() { |
| 150 | Fl_Font_Descriptor *fl_fontsize = font_descriptor(); |
| 151 | if (fl_fontsize) return (fl_fontsize->metr.tmAscent + fl_fontsize->metr.tmDescent); |
| 152 | else return -1; |
| 153 | } |
| 154 | |
| 155 | int Fl_GDI_Graphics_Driver::descent() { |
| 156 | Fl_Font_Descriptor *fl_fontsize = font_descriptor(); |
| 157 | if (fl_fontsize) return fl_fontsize->metr.tmDescent; |
| 158 | else return -1; |
| 159 | } |
| 160 | |
| 161 | // Unicode string buffer |
| 162 | static unsigned short *wstr = NULL; |
| 163 | static int wstr_len = 0; |
| 164 | |
| 165 | |
| 166 | double Fl_GDI_Graphics_Driver::width(const char* c, int n) { |
| 167 | int i = 0; |
| 168 | if (!font_descriptor()) return -1.0; |
| 169 | double w = 0.0; |
| 170 | char *end = (char *)&c[n]; |
| 171 | while (i < n) { |
| 172 | unsigned int ucs; |
| 173 | int l; |
| 174 | ucs = fl_utf8decode((const char*)(c + i), end, &l); |
| 175 | // if (l < 1) l = 1; |
| 176 | i += l; |
| 177 | if (!fl_nonspacing(ucs)) { |
| 178 | w += width(ucs); |
| 179 | } |
| 180 | } |
| 181 | return w; |
| 182 | } |
| 183 | |
| 184 | double Fl_GDI_Graphics_Driver::width(unsigned int c) { |
| 185 | Fl_Font_Descriptor *fl_fontsize = font_descriptor(); |
| 186 | unsigned int r; |
| 187 | SIZE s; |
| 188 | // Special Case Handling of Unicode points over U+FFFF. |
| 189 | // The logic (below) computes a lookup table for char widths |
| 190 | // on-the-fly, but the table only covers codepoints up to |
| 191 | // U+FFFF, which covers the basic multilingual plane, but |
| 192 | // not any higher plane, or glyphs that require surrogate-pairs |
| 193 | // to encode them in WinXX, which is UTF16. |
| 194 | // This code assumes that these glyphs are rarely used and simply |
| 195 | // measures them explicitly if they occur - This will be slow... |
| 196 | if(c > 0x0000FFFF) { // UTF16 surrogate pair is needed |
| 197 | if (!fl_gc) { // We have no valid gc, so nothing to measure - bail out |
| 198 | return 0.0; |
| 199 | } |
| 200 | int cc; // cell count |
| 201 | unsigned short u16[4]; // Array for UTF16 representation of c |
| 202 | // Creates a UTF16 string from a UCS code point. |
| 203 | cc = fl_ucs_to_Utf16(c, u16, 4); |
| 204 | // Make sure the current font is selected before we make the measurement |
| 205 | SelectObject(fl_gc, fl_fontsize->fid); |
| 206 | // measure the glyph width |
| 207 | GetTextExtentPoint32W(fl_gc, (WCHAR*)u16, cc, &s); |
| 208 | return (double)s.cx; |
| 209 | } |
| 210 | // else - this falls through to the lookup-table for glyph widths |
| 211 | // in the basic multilingual plane |
| 212 | r = (c & 0xFC00) >> 10; |
| 213 | if (!fl_fontsize->width[r]) { |
| 214 | fl_fontsize->width[r] = (int*) malloc(sizeof(int) * 0x0400); |
| 215 | unsigned short i = 0, ii = r * 0x400; |
| 216 | // The following code makes a best effort attempt to obtain a valid fl_gc. |
| 217 | // If no fl_gc is available at the time we call fl_width(), then we first |
| 218 | // try to obtain a gc from the first fltk window. |
| 219 | // If that is null then we attempt to obtain the gc from the current screen |
| 220 | // using (GetDC(NULL)). |
| 221 | // This should resolve STR #2086 |
| 222 | HDC gc = fl_gc; |
| 223 | HWND hWnd = 0; |
| 224 | if (!gc) { // We have no valid gc, try and obtain one |
| 225 | // Use our first fltk window, or fallback to using the screen via GetDC(NULL) |
| 226 | hWnd = Fl::first_window() ? fl_xid(Fl::first_window()) : NULL; |
| 227 | gc = GetDC(hWnd); |
| 228 | } |
| 229 | if (!gc) |
| 230 | Fl::fatal("Invalid graphic context: fl_width() failed because no valid HDC was found!"); |
| 231 | SelectObject(gc, fl_fontsize->fid); |
| 232 | for (; i < 0x400; i++) { |
| 233 | GetTextExtentPoint32W(gc, (WCHAR*)&ii, 1, &s); |
| 234 | fl_fontsize->width[r][i] = s.cx; |
| 235 | ii++; |
| 236 | } |
| 237 | if (gc && gc!=fl_gc) ReleaseDC(hWnd, gc); |
| 238 | } |
| 239 | return (double) fl_fontsize->width[r][c & 0x03FF]; |
| 240 | } |
| 241 | |
| 242 | /* Add function pointer to allow us to access GetGlyphIndicesW on systems that have it, |
| 243 | * without crashing on systems that do not. */ |
| 244 | /* DWORD WINAPI GetGlyphIndicesW(HDC,LPCWSTR,int,LPWORD,DWORD) */ |
| 245 | typedef DWORD (WINAPI* fl_GetGlyphIndices_func)(HDC,LPCWSTR,int,LPWORD,DWORD); |
| 246 | |
| 247 | static fl_GetGlyphIndices_func fl_GetGlyphIndices = NULL; // used to hold a proc pointer for GetGlyphIndicesW |
| 248 | static int have_loaded_GetGlyphIndices = 0; // Set this non-zero once we have tried to load GetGlyphIndices |
| 249 | |
| 250 | // Function that tries to dynamically load GetGlyphIndicesW at runtime |
| 251 | static void GetGlyphIndices_init() { |
| 252 | // Since not all versions of Windows include GetGlyphIndicesW support, |
| 253 | // we do a run-time check for the required function. |
| 254 | HMODULE hMod = GetModuleHandle("GDI32.DLL"); |
| 255 | if (hMod) { |
| 256 | // check that GetGlyphIndicesW is available |
| 257 | fl_GetGlyphIndices = (fl_GetGlyphIndices_func)GetProcAddress(hMod, "GetGlyphIndicesW"); |
| 258 | } |
| 259 | have_loaded_GetGlyphIndices = -1; // set this non-zero when we have attempted to load GetGlyphIndicesW |
| 260 | } // GetGlyphIndices_init function |
| 261 | |
| 262 | static void on_printer_extents_update(int &dx, int &dy, int &w, int &h) |
| 263 | // converts text extents from device coords to logical coords |
| 264 | { |
| 265 | POINT pt[3] = { {0, 0}, {dx, dy}, {dx+w, dy+h} }; |
| 266 | DPtoLP(fl_gc, pt, 3); |
| 267 | w = pt[2].x - pt[1].x; |
| 268 | h = pt[2].y - pt[1].y; |
| 269 | dx = pt[1].x - pt[0].x; |
| 270 | dy = pt[1].y - pt[0].y; |
| 271 | } |
| 272 | |
| 273 | // if printer context, extents shd be converted to logical coords |
| 274 | #define EXTENTS_UPDATE(x,y,w,h) \ |
| 275 | if (Fl_Surface_Device::surface()->class_name() == Fl_Printer::class_id) { on_printer_extents_update(x,y,w,h); } |
| 276 | |
| 277 | // Function to determine the extent of the "inked" area of the glyphs in a string |
| 278 | void Fl_GDI_Graphics_Driver::text_extents(const char *c, int n, int &dx, int &dy, int &w, int &h) { |
| 279 | |
| 280 | Fl_Font_Descriptor *fl_fontsize = font_descriptor(); |
| 281 | if (!fl_fontsize) { // no valid font, nothing to measure |
| 282 | w = 0; h = 0; |
| 283 | dx = dy = 0; |
| 284 | return; |
| 285 | } |
| 286 | |
| 287 | static unsigned short *ext_buff = NULL; // UTF-16 converted version of input UTF-8 string |
| 288 | static WORD *w_buff = NULL; // glyph indices array |
| 289 | static unsigned wc_len = 0; // current string buffer dimensions |
| 290 | static const MAT2 matrix = { { 0, 1 }, { 0, 0 }, { 0, 0 }, { 0, 1 } }; // identity mat for GetGlyphOutlineW |
| 291 | GLYPHMETRICS metrics; |
| 292 | int maxw = 0, maxh = 0, dh; |
| 293 | int minx = 0, miny = -999999; |
| 294 | unsigned len = 0, idx = 0; |
| 295 | HWND hWnd = 0; |
| 296 | HDC gc = fl_gc; // local copy of current gc - make a copy in case we change it... |
| 297 | int has_surrogates; // will be set if the string contains surrogate pairs |
| 298 | |
| 299 | // Have we loaded the GetGlyphIndicesW function yet? |
| 300 | if (have_loaded_GetGlyphIndices == 0) { |
| 301 | GetGlyphIndices_init(); |
| 302 | } |
| 303 | // Do we have a usable GetGlyphIndices function? |
| 304 | if(!fl_GetGlyphIndices) goto exit_error; // No GetGlyphIndices function, use fallback mechanism instead |
| 305 | |
| 306 | // The following code makes a best effort attempt to obtain a valid fl_gc. |
| 307 | // See description in fl_width() above for an explanation. |
| 308 | if (!gc) { // We have no valid gc, try and obtain one |
| 309 | // Use our first fltk window, or fallback to using the screen via GetDC(NULL) |
| 310 | hWnd = Fl::first_window() ? fl_xid(Fl::first_window()) : NULL; |
| 311 | gc = GetDC(hWnd); |
| 312 | } |
| 313 | if (!gc) goto exit_error; // no valid gc, attempt to use fallback measure |
| 314 | |
| 315 | // now convert the string to WCHAR and measure it |
| 316 | len = fl_utf8toUtf16(c, n, ext_buff, wc_len); |
| 317 | if(len >= wc_len) { |
| 318 | if(ext_buff) {delete [] ext_buff;} |
| 319 | if(w_buff) {delete [] w_buff;} |
| 320 | wc_len = len + 64; |
| 321 | ext_buff = new unsigned short[wc_len]; |
| 322 | w_buff = new WORD[wc_len]; |
| 323 | len = fl_utf8toUtf16(c, n, ext_buff, wc_len); |
| 324 | } |
| 325 | SelectObject(gc, fl_fontsize->fid); |
| 326 | |
| 327 | // Are there surrogate-pairs in this string? If so GetGlyphIndicesW will fail |
| 328 | // since it can only handle the BMP range. |
| 329 | // We ideally want to use GetGlyphIndicesW, as it is the Right Thing, but it |
| 330 | // only works for the BMP, so we leverage GetCharacterPlacementW instead, which |
| 331 | // is not ideal, but works adequately well, and does handle surrogate pairs. |
| 332 | has_surrogates = 0; |
| 333 | for(unsigned ll = 0; ll < len; ll++) { |
| 334 | if((ext_buff[ll] >= 0xD800) && (ext_buff[ll] < 0xE000)) { |
| 335 | has_surrogates = -1; |
| 336 | break; |
| 337 | } |
| 338 | } |
| 339 | if (has_surrogates) { |
| 340 | // GetGlyphIndices will not work - use GetCharacterPlacementW() instead |
| 341 | GCP_RESULTSW gcp_res; |
| 342 | memset(w_buff, 0, (sizeof(WORD) * wc_len)); |
| 343 | memset(&gcp_res, 0, sizeof(GCP_RESULTSW)); |
| 344 | gcp_res.lpGlyphs = (LPWSTR)w_buff; |
| 345 | gcp_res.nGlyphs = wc_len; |
| 346 | gcp_res.lStructSize = sizeof(gcp_res); |
| 347 | |
| 348 | DWORD dr = GetCharacterPlacementW(gc, (WCHAR*)ext_buff, len, 0, &gcp_res, GCP_GLYPHSHAPE); |
| 349 | if(dr) { |
| 350 | len = gcp_res.nGlyphs; |
| 351 | } else goto exit_error; |
| 352 | } else { |
| 353 | if (fl_GetGlyphIndices(gc, (WCHAR*)ext_buff, len, w_buff, GGI_MARK_NONEXISTING_GLYPHS) == GDI_ERROR) { |
| 354 | // some error occured here - just return fl_measure values |
| 355 | goto exit_error; |
| 356 | } |
| 357 | } |
| 358 | |
| 359 | // now we have the glyph array we measure each glyph in turn... |
| 360 | for(idx = 0; idx < len; idx++){ |
| 361 | if (GetGlyphOutlineW (gc, w_buff[idx], GGO_METRICS | GGO_GLYPH_INDEX, |
| 362 | &metrics, 0, NULL, &matrix) == GDI_ERROR) { |
| 363 | goto exit_error; |
| 364 | } |
| 365 | maxw += metrics.gmCellIncX; |
| 366 | if(idx == 0) minx = metrics.gmptGlyphOrigin.x; |
| 367 | dh = metrics.gmBlackBoxY - metrics.gmptGlyphOrigin.y; |
| 368 | if(dh > maxh) maxh = dh; |
| 369 | if(miny < metrics.gmptGlyphOrigin.y) miny = metrics.gmptGlyphOrigin.y; |
| 370 | } |
| 371 | // for the last cell, we only want the bounding X-extent, not the glyphs increment step |
| 372 | maxw = maxw - metrics.gmCellIncX + metrics.gmBlackBoxX + metrics.gmptGlyphOrigin.x; |
| 373 | w = maxw - minx; |
| 374 | h = maxh + miny; |
| 375 | dx = minx; |
| 376 | dy = -miny; |
| 377 | EXTENTS_UPDATE(dx, dy, w, h); |
| 378 | return; // normal exit |
| 379 | |
| 380 | exit_error: |
| 381 | // some error here - just return fl_measure values |
| 382 | w = (int)width(c, n); |
| 383 | h = height(); |
| 384 | dx = 0; |
| 385 | dy = descent() - h; |
| 386 | EXTENTS_UPDATE(dx, dy, w, h); |
| 387 | return; |
| 388 | } // fl_text_extents |
| 389 | |
| 390 | void Fl_GDI_Graphics_Driver::draw(const char* str, int n, int x, int y) { |
| 391 | COLORREF oldColor = SetTextColor(fl_gc, fl_RGB()); |
| 392 | SelectObject(fl_gc, font_descriptor()->fid); |
| 393 | int wn = fl_utf8toUtf16(str, n, wstr, wstr_len); |
| 394 | if(wn >= wstr_len) { |
| 395 | wstr = (unsigned short*) realloc(wstr, sizeof(unsigned short) * (wn + 1)); |
| 396 | wstr_len = wn + 1; |
| 397 | wn = fl_utf8toUtf16(str, n, wstr, wstr_len); |
| 398 | } |
| 399 | TextOutW(fl_gc, x, y, (WCHAR*)wstr, wn); |
| 400 | SetTextColor(fl_gc, oldColor); // restore initial state |
| 401 | } |
| 402 | |
| 403 | void Fl_GDI_Graphics_Driver::draw(int angle, const char* str, int n, int x, int y) { |
| 404 | fl_font(this, Fl_Graphics_Driver::font(), size(), angle); |
| 405 | int wn = 0; // count of UTF16 cells to render full string |
| 406 | COLORREF oldColor = SetTextColor(fl_gc, fl_RGB()); |
| 407 | SelectObject(fl_gc, font_descriptor()->fid); |
| 408 | wn = fl_utf8toUtf16(str, n, wstr, wstr_len); |
| 409 | if(wn >= wstr_len) { // Array too small |
| 410 | wstr = (unsigned short*) realloc(wstr, sizeof(unsigned short) * (wn + 1)); |
| 411 | wstr_len = wn + 1; |
| 412 | wn = fl_utf8toUtf16(str, n, wstr, wstr_len); // respin the translation |
| 413 | } |
| 414 | TextOutW(fl_gc, x, y, (WCHAR*)wstr, wn); |
| 415 | SetTextColor(fl_gc, oldColor); |
| 416 | fl_font(this, Fl_Graphics_Driver::font(), size(), 0); |
| 417 | } |
| 418 | |
| 419 | void Fl_GDI_Graphics_Driver::rtl_draw(const char* c, int n, int x, int y) { |
| 420 | int wn; |
| 421 | wn = fl_utf8toUtf16(c, n, wstr, wstr_len); |
| 422 | if(wn >= wstr_len) { |
| 423 | wstr = (unsigned short*) realloc(wstr, sizeof(unsigned short) * (wn + 1)); |
| 424 | wstr_len = wn + 1; |
| 425 | wn = fl_utf8toUtf16(c, n, wstr, wstr_len); |
| 426 | } |
| 427 | |
| 428 | COLORREF oldColor = SetTextColor(fl_gc, fl_RGB()); |
| 429 | SelectObject(fl_gc, font_descriptor()->fid); |
| 430 | #ifdef RTL_CHAR_BY_CHAR |
| 431 | int i = 0; |
| 432 | int lx = 0; |
| 433 | while (i < wn) { // output char by char is very bad for Arabic but coherent with fl_width() |
| 434 | lx = (int) width(wstr[i]); |
| 435 | x -= lx; |
| 436 | TextOutW(fl_gc, x, y, (WCHAR*)wstr + i, 1); |
| 437 | if (fl_nonspacing(wstr[i])) { |
| 438 | x += lx; |
| 439 | } |
| 440 | i++; |
| 441 | } |
| 442 | #else |
| 443 | UINT old_align = SetTextAlign(fl_gc, TA_RIGHT | TA_RTLREADING); |
| 444 | TextOutW(fl_gc, x, y - height() + descent(), (WCHAR*)wstr, wn); |
| 445 | SetTextAlign(fl_gc, old_align); |
| 446 | #endif |
| 447 | SetTextColor(fl_gc, oldColor); |
| 448 | } |
| 449 | #endif |
| 450 | // |
| 451 | // End of "$Id: fl_font_win32.cxx 8644 2011-05-10 15:37:05Z ianmacarthur $". |
| 452 | // |