// // "$Id$" // // WIN32 font selection routines for the Fast Light Tool Kit (FLTK). // // Copyright 1998-2005 by Bill Spitzak and others. // // This library is free software; you can redistribute it and/or // modify it under the terms of the GNU Library General Public // License as published by the Free Software Foundation; either // version 2 of the License, or (at your option) any later version. // // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Library General Public License for more details. // // You should have received a copy of the GNU Library General Public // License along with this library; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 // USA. // // Please report all bugs and problems on the following page: // // http://www.fltk.org/str.php // Fl_Font_Descriptor::Fl_Font_Descriptor(const char* name, Fl_Fontsize size) { int weight = FW_NORMAL; int italic = 0; switch (*name++) { case 'I': italic = 1; break; case 'P': italic = 1; case 'B': weight = FW_BOLD; break; case ' ': break; default: name--; } fid = CreateFont( -size, // negative makes it use "char size" 0, // logical average character width 0, // angle of escapement 0, // base-line orientation angle weight, italic, FALSE, // underline attribute flag FALSE, // strikeout attribute flag DEFAULT_CHARSET, // character set identifier OUT_DEFAULT_PRECIS, // output precision CLIP_DEFAULT_PRECIS,// clipping precision DEFAULT_QUALITY, // output quality DEFAULT_PITCH, // pitch and family name // pointer to typeface name string ); if (!fl_gc) fl_GetDC(0); SelectObject(fl_gc, fid); GetTextMetrics(fl_gc, &metr); // BOOL ret = GetCharWidthFloat(fl_gc, metr.tmFirstChar, metr.tmLastChar, font->width+metr.tmFirstChar); // ...would be the right call, but is not implemented into Window95! (WinNT?) //GetCharWidth(fl_gc, 0, 255, width); int i; for (i = 0; i < 64; i++) width[i] = NULL; #if HAVE_GL listbase = 0; for (i = 0; i < 64; i++) glok[i] = 0; #endif minsize = maxsize = size; } Fl_Font_Descriptor* fl_fontsize; Fl_Font_Descriptor::~Fl_Font_Descriptor() { #if HAVE_GL // Delete list created by gl_draw(). This is not done by this code // as it will link in GL unnecessarily. There should be some kind // of "free" routine pointer, or a subclass? // if (listbase) { // int base = font->min_char_or_byte2; // int size = font->max_char_or_byte2-base+1; // int base = 0; int size = 256; // glDeleteLists(listbase+base,size); // } #endif if (this == fl_fontsize) fl_fontsize = 0; DeleteObject(fid); int i; for (i = 0; i < 64; i++) free(width[i]); } //////////////////////////////////////////////////////////////// // WARNING: if you add to this table, you must redefine FL_FREE_FONT // in Enumerations.H & recompile!! static Fl_Fontdesc built_in_table[] = { {" Arial"}, {"BArial"}, {"IArial"}, {"PArial"}, {" Courier New"}, {"BCourier New"}, {"ICourier New"}, {"PCourier New"}, {" Times New Roman"}, {"BTimes New Roman"}, {"ITimes New Roman"}, {"PTimes New Roman"}, {" Symbol"}, {" Terminal"}, {"BTerminal"}, {" Wingdings"}, }; Fl_Fontdesc* fl_fonts = built_in_table; static Fl_Font_Descriptor* find(Fl_Font fnum, Fl_Fontsize size) { Fl_Fontdesc* s = fl_fonts+fnum; if (!s->name) s = fl_fonts; // use 0 if fnum undefined Fl_Font_Descriptor* f; for (f = s->first; f; f = f->next) if (f->minsize <= size && f->maxsize >= size) return f; f = new Fl_Font_Descriptor(s->name, size); f->next = s->first; s->first = f; return f; } //////////////////////////////////////////////////////////////// // Public interface: Fl_Font fl_font_ = 0; Fl_Fontsize fl_size_ = 0; //static HDC font_gc; void fl_font(Fl_Font fnum, Fl_Fontsize size) { if (fnum==-1) { // just make sure that we will load a new font next time fl_font_ = 0; fl_size_ = 0; return; } if (fnum == fl_font_ && size == fl_size_) return; fl_font_ = fnum; fl_size_ = size; fl_fontsize = find(fnum, size); } int fl_height() { if (fl_fontsize) return (fl_fontsize->metr.tmAscent + fl_fontsize->metr.tmDescent); else return -1; } int fl_descent() { if (fl_fontsize) return fl_fontsize->metr.tmDescent; else return -1; } // Unicode string buffer static xchar *wstr = NULL; static int wstr_len = 0; double fl_width(const char* c, int n) { int i = 0; if (!fl_fontsize) return -1.0; double w = 0.0; char *end = (char *)&c[n]; while (i < n) { unsigned int ucs; // int l = fl_utf2ucs((const unsigned char*)c + i, n - i, &ucs); int l; ucs = fl_utf8decode((const char*)(c + i), end, &l); // if (l < 1) l = 1; i += l; if (!fl_nonspacing(ucs)) { w += fl_width(ucs); } } return w; } double fl_width(unsigned int c) { unsigned int r; r = (c & 0xFC00) >> 10; if (!fl_fontsize->width[r]) { SelectObject(fl_gc, fl_fontsize->fid); fl_fontsize->width[r] = (int*) malloc(sizeof(int) * 0x0400); SIZE s; unsigned short i = 0, ii = r * 0x400; for (; i < 0x400; i++) { GetTextExtentPoint32W(fl_gc, (WCHAR*)&ii, 1, &s); fl_fontsize->width[r][i] = s.cx; ii++; } } return (double) fl_fontsize->width[r][c & 0x03FF]; } void fl_draw(const char* str, int n, int x, int y) { int i = 0; int lx = 0; char *end = (char *)&str[n]; COLORREF oldColor = SetTextColor(fl_gc, fl_RGB()); SelectObject(fl_gc, fl_fontsize->fid); while (i < n) { unsigned int u; unsigned int u1; unsigned short ucs; // int l = fl_utf2ucs((const unsigned char*)str + i, n - i, &u); int l; u = fl_utf8decode((const char*)(str + i), end, &l); if (u1 = fl_nonspacing(u)) { x -= lx; u = u1; } else { lx = (int) fl_width(u); } ucs = u; if (l < 1) l = 1; i += l; TextOutW(fl_gc, x, y, (WCHAR*)&ucs, 1); x += lx; } SetTextColor(fl_gc, oldColor); } void fl_rtl_draw(const char* c, int n, int x, int y) { int wn; int i = 0; int lx = 0; // if (n > wstr_len) { // wstr = (xchar*) realloc(wstr, sizeof(xchar) * (n + 1)); // wstr_len = n; // } //wn = fl_utf2unicode((const unsigned char *)c, n, wstr); wn = fl_utf8toUtf16(c, n, (unsigned short*)wstr, wstr_len); if(wn >= wstr_len) { wstr = (xchar*) realloc(wstr, sizeof(xchar) * (wn + 1)); wstr_len = wn + 1; wn = fl_utf8toUtf16(c, n, (unsigned short*)wstr, wstr_len); } COLORREF oldColor = SetTextColor(fl_gc, fl_RGB()); SelectObject(fl_gc, fl_fontsize->fid); while (i < wn) { lx = (int) fl_width(wstr[i]); x -= lx; TextOutW(fl_gc, x, y, (WCHAR*)wstr + i, 1); if (fl_nonspacing(wstr[i])) { x += lx; } i++; } SetTextColor(fl_gc, oldColor); } // // End of "$Id$". //