899184656a
This one was really tough to track, understand: In fact, the problem was comming from the misplacement of the menu window, which itself came from invalid measurement, which itself came from invalid fl_witdh() measurement, but only when fl_gc is not valid because fl_width() relies on Win32 on the call of GetTextExtentPoint32W which can't succeed if the HDC(here fl_gc) is not valid ! Now the fix: A best-effort algorithm has been furthered to supply a valid fltk hdc if we can have one or a screen hdc if no fltk window is found by fl::first_window(). Note that when fl_gc is NULL inside fl_width() call, it can happen that Fl_Window::current() is not null but invalid (already deleted). Finally, in the case of the buggy menu window observed here, this fl_gc was set to NULL just after an Fl_Menu_Window deletion and re-creation in Fl_Menu_Item::pulldown(). Also added a comment to describe the new fl_width() behavior. git-svn-id: file:///fltk/svn/fltk/branches/branch-1.3@6540 ea41ed52-d2ee-0310-a9c1-e6b18d33e121
358 lines
11 KiB
C++
358 lines
11 KiB
C++
//
|
|
// "$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
|
|
//
|
|
#ifndef FL_DOXYGEN
|
|
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;
|
|
// The following code is a best effort algorithm to further a valid fl_gc
|
|
// if no fl_gc is available at the time we call fl_width()
|
|
// We first choose a gc from the first fltk window,
|
|
// if it is null then the gc from the current screen (GetDC(NULL)).
|
|
// This should solve STR #2086
|
|
HDC gc = fl_gc;
|
|
HWND hWnd = 0;
|
|
if (!gc) {
|
|
gc = GetDC(hWnd);
|
|
}
|
|
if (!gc)
|
|
Fl::fatal("Invalid graphic context: fl_width() failed because no valid HDC was found!");
|
|
for (; i < 0x400; i++) {
|
|
GetTextExtentPoint32W(gc, (WCHAR*)&ii, 1, &s);
|
|
fl_fontsize->width[r][i] = s.cx;
|
|
ii++;
|
|
}
|
|
if (gc && gc!=fl_gc) ReleaseDC(hWnd, gc);
|
|
}
|
|
return (double) fl_fontsize->width[r][c & 0x03FF];
|
|
}
|
|
|
|
/* Add function pointer to allow us to access GetGlyphIndicesW on systems that have it,
|
|
* without crashing on systems that do not. */
|
|
/* DWORD WINAPI GetGlyphIndicesW(HDC,LPCWSTR,int,LPWORD,DWORD) */
|
|
typedef DWORD (WINAPI* fl_GetGlyphIndices_func)(HDC,LPCWSTR,int,LPWORD,DWORD);
|
|
|
|
static fl_GetGlyphIndices_func fl_GetGlyphIndices = NULL; // used to hold a proc pointer for GetGlyphIndicesW
|
|
static int have_loaded_GetGlyphIndices = 0; // Set this non-zero once we have tried to load GetGlyphIndices
|
|
|
|
// Function that tries to dynamically load GetGlyphIndicesW at runtime
|
|
static void GetGlyphIndices_init() {
|
|
// Since not all versions of Windows include GetGlyphIndicesW support,
|
|
// we do a run-time check for the required function.
|
|
HMODULE hMod = GetModuleHandle("GDI32.DLL");
|
|
if (hMod) {
|
|
// check that GetGlyphIndicesW is available
|
|
fl_GetGlyphIndices = (fl_GetGlyphIndices_func)GetProcAddress(hMod, "GetGlyphIndicesW");
|
|
}
|
|
have_loaded_GetGlyphIndices = -1; // set this non-zero when we have attempted to load GetGlyphIndicesW
|
|
} // GetGlyphIndices_init function
|
|
|
|
static unsigned short *ext_buff = NULL; // UTF-16 converted version of input UTF-8 string
|
|
static unsigned wc_len = 0; // current string buffer dimension
|
|
static WORD *gi = NULL; // glyph indices array
|
|
// Function to determine the extent of the "inked" area of the glyphs in a string
|
|
void fl_text_extents(const char *c, int n, int &dx, int &dy, int &w, int &h) {
|
|
if (!fl_fontsize) {
|
|
w = 0; h = 0;
|
|
dx = dy = 0;
|
|
return;
|
|
}
|
|
static const MAT2 matrix = { { 0, 1 }, { 0, 0 }, { 0, 0 }, { 0, 1 } };
|
|
GLYPHMETRICS metrics;
|
|
int maxw = 0, maxh = 0, dh;
|
|
int minx = 0, miny = -999999;
|
|
unsigned len = 0;
|
|
|
|
// Have we loaded the GetGlyphIndicesW function yet?
|
|
if (have_loaded_GetGlyphIndices == 0) {
|
|
GetGlyphIndices_init();
|
|
}
|
|
// Do we have a usable GetGlyphIndices function?
|
|
if(!fl_GetGlyphIndices) goto exit_error; // No GetGlyphIndices function, use fallback mechanism instead
|
|
|
|
// now convert the string to WCHAR and measure it
|
|
len = fl_utf8toUtf16(c, n, ext_buff, wc_len);
|
|
if(len >= wc_len) {
|
|
if(ext_buff) {delete [] ext_buff;}
|
|
if(gi) {delete [] gi;}
|
|
wc_len = len + 64;
|
|
ext_buff = new unsigned short[wc_len];
|
|
gi = new WORD[wc_len];
|
|
len = fl_utf8toUtf16(c, n, ext_buff, wc_len);
|
|
}
|
|
SelectObject(fl_gc, fl_fontsize->fid);
|
|
|
|
if (fl_GetGlyphIndices(fl_gc, (WCHAR*)ext_buff, len, gi, 0) == GDI_ERROR) {
|
|
// some error occured here - just return fl_measure values?
|
|
goto exit_error;
|
|
}
|
|
|
|
// now we have the glyph array we measure each glyph in turn...
|
|
for(unsigned idx = 0; idx < len; idx++){
|
|
if (GetGlyphOutlineW (fl_gc, gi[idx], GGO_METRICS | GGO_GLYPH_INDEX,
|
|
&metrics, 0, NULL, &matrix) == GDI_ERROR) {
|
|
goto exit_error;
|
|
}
|
|
maxw += metrics.gmCellIncX;
|
|
if(idx == 0) minx = metrics.gmptGlyphOrigin.x;
|
|
dh = metrics.gmBlackBoxY - metrics.gmptGlyphOrigin.y;
|
|
if(dh > maxh) maxh = dh;
|
|
if(miny < metrics.gmptGlyphOrigin.y) miny = metrics.gmptGlyphOrigin.y;
|
|
}
|
|
|
|
// for the last cell, we only want the bounding X-extent, not the glyphs increment step
|
|
maxw = maxw - metrics.gmCellIncX + metrics.gmBlackBoxX + metrics.gmptGlyphOrigin.x;
|
|
w = maxw - minx;
|
|
h = maxh + miny;
|
|
dx = minx;
|
|
dy = -miny;
|
|
return; // normal exit
|
|
|
|
exit_error:
|
|
// some error here - just return fl_measure values
|
|
w = (int)fl_width(c, n);
|
|
h = fl_height();
|
|
dx = 0;
|
|
dy = fl_descent() - h;
|
|
return;
|
|
} // fl_text_extents
|
|
|
|
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);
|
|
}
|
|
#endif
|
|
//
|
|
// End of "$Id$".
|
|
//
|