2001-11-27 20:44:08 +03:00
|
|
|
//
|
2005-02-25 00:55:12 +03:00
|
|
|
// "$Id$"
|
2001-11-27 20:44:08 +03:00
|
|
|
//
|
|
|
|
// MacOS font selection routines for the Fast Light Tool Kit (FLTK).
|
|
|
|
//
|
2005-02-25 00:55:12 +03:00
|
|
|
// Copyright 1998-2005 by Bill Spitzak and others.
|
2001-11-27 20:44:08 +03:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
2005-04-16 04:13:17 +04:00
|
|
|
// Please report all bugs and problems on the following page:
|
|
|
|
//
|
|
|
|
// http://www.fltk.org/str.php
|
2001-11-27 20:44:08 +03:00
|
|
|
//
|
|
|
|
|
2004-08-25 04:20:27 +04:00
|
|
|
#include <config.h>
|
|
|
|
|
2001-11-27 20:44:08 +03:00
|
|
|
//: MeasureText, FontMetrics, WidthTabHandle, GetSysFont, SysFontSize
|
|
|
|
//: TextSize, TextFont
|
|
|
|
//: GetFNum (theName: Str255; VAR familyID: Integer);
|
|
|
|
//: FUNCTION FMSwapFont (inRec: FMInput): FMOutPtr;
|
2002-06-28 03:18:12 +04:00
|
|
|
//: SetFractEnable
|
2001-11-27 20:44:08 +03:00
|
|
|
|
|
|
|
Fl_FontSize::Fl_FontSize(const char* name, int Size) {
|
2004-09-10 01:34:48 +04:00
|
|
|
next = 0;
|
|
|
|
# if HAVE_GL
|
|
|
|
listbase = 0;
|
|
|
|
# endif
|
2004-08-26 04:18:43 +04:00
|
|
|
#ifdef __APPLE_QD__
|
2001-11-27 20:44:08 +03:00
|
|
|
knowMetrics = 0;
|
|
|
|
switch (*name++) {
|
|
|
|
case 'I': face = italic; break;
|
|
|
|
case 'P': face = italic | bold; break;
|
|
|
|
case 'B': face = bold; break;
|
|
|
|
default: face = 0; break;
|
|
|
|
}
|
|
|
|
unsigned char fn[80];
|
|
|
|
fn[0] = strlen(name); strcpy((char*)(fn+1), name);
|
|
|
|
GetFNum(fn, &font);
|
|
|
|
size = Size;
|
|
|
|
FMInput fIn = { font, size, face, 0, 0, { 1, 1}, { 1, 1} };
|
|
|
|
FMOutput *fOut = FMSwapFont(&fIn);
|
|
|
|
ascent = fOut->ascent; //: the following three lines give only temporary aproimations
|
|
|
|
descent = fOut->descent;
|
|
|
|
for (int i=0; i<256; i++) width[i] = fOut->widMax;
|
|
|
|
minsize = maxsize = size;
|
2004-08-26 04:18:43 +04:00
|
|
|
#elif defined(__APPLE_QUARTZ__)
|
2006-06-02 18:17:41 +04:00
|
|
|
// OpenGL needs those for its font handling
|
2004-09-09 04:55:41 +04:00
|
|
|
q_name = strdup(name);
|
2004-08-26 04:18:43 +04:00
|
|
|
size = Size;
|
2006-06-02 18:17:41 +04:00
|
|
|
OSStatus err;
|
|
|
|
// fill our structure with a few default values
|
2004-08-26 04:18:43 +04:00
|
|
|
ascent = Size*3/4;
|
|
|
|
descent = Size-ascent;
|
|
|
|
q_width = Size*2/3;
|
|
|
|
minsize = maxsize = Size;
|
2006-06-02 18:17:41 +04:00
|
|
|
// now use ATS to get the actual Glyph size information
|
|
|
|
CFStringRef cfname = CFStringCreateWithCString(0L, name, kCFStringEncodingASCII);
|
2004-09-09 04:55:41 +04:00
|
|
|
ATSFontRef font = ATSFontFindFromName(cfname, kATSOptionFlagsDefault);
|
|
|
|
if (font) {
|
|
|
|
ATSFontMetrics m = { 0 };
|
|
|
|
ATSFontGetHorizontalMetrics(font, kATSOptionFlagsDefault, &m);
|
2006-06-02 18:17:41 +04:00
|
|
|
if (m.avgAdvanceWidth) q_width = int(m.avgAdvanceWidth*Size);
|
|
|
|
// playing with the offsets a little to make standard sizes fit
|
|
|
|
if (m.ascent) ascent = int(m.ascent*Size-0.5f);
|
|
|
|
if (m.descent) descent = -int(m.descent*Size-1.5f);
|
2004-09-09 04:55:41 +04:00
|
|
|
}
|
|
|
|
CFRelease(cfname);
|
2006-06-02 18:17:41 +04:00
|
|
|
// now we allocate everything needed to render text in this font later
|
|
|
|
// get us the default layout and style
|
|
|
|
err = ATSUCreateTextLayout(&layout);
|
|
|
|
UniChar mTxt[2] = { 65, 0 };
|
|
|
|
err = ATSUSetTextPointerLocation(layout, mTxt, kATSUFromTextBeginning, 1, 1);
|
|
|
|
err = ATSUCreateStyle(&style);
|
|
|
|
err = ATSUSetRunStyle(layout, style, kATSUFromTextBeginning, kATSUToTextEnd);
|
|
|
|
// now set the actual font, size and attributes
|
|
|
|
Fixed fsize = IntToFixed(Size);
|
|
|
|
ATSUFontID fontID = FMGetFontFromATSFontRef(font);
|
|
|
|
ATSUAttributeTag sTag[] = { kATSUFontTag, kATSUSizeTag };
|
|
|
|
ByteCount sBytes[] = { sizeof(ATSUFontID), sizeof(Fixed) };
|
|
|
|
ATSUAttributeValuePtr sAttr[] = { &fontID, &fsize };
|
|
|
|
err = ATSUSetAttributes(style, 2, sTag, sBytes, sAttr);
|
|
|
|
// next, make sure that Quartz will only render at integer coordinates
|
|
|
|
ATSLineLayoutOptions llo = kATSLineUseDeviceMetrics|kATSLineDisableAllLayoutOperations;
|
|
|
|
ATSUAttributeTag aTag[] = { kATSULineLayoutOptionsTag };
|
|
|
|
ByteCount aBytes[] = { sizeof(ATSLineLayoutOptions) };
|
|
|
|
ATSUAttributeValuePtr aAttr[] = { &llo };
|
|
|
|
err = ATSUSetLineControls (layout, kATSUFromTextBeginning, 1, aTag, aBytes, aAttr);
|
|
|
|
// now we are finally ready to measure some letter to get the bounding box
|
|
|
|
Fixed bBefore, bAfter, bAscent, bDescent;
|
|
|
|
err = ATSUGetUnjustifiedBounds(layout, kATSUFromTextBeginning, 1, &bBefore, &bAfter, &bAscent, &bDescent);
|
|
|
|
ascent = FixedToInt(bAscent);
|
|
|
|
descent = FixedToInt(bDescent);
|
|
|
|
q_width = FixedToInt(bAfter);
|
2004-08-26 04:18:43 +04:00
|
|
|
#endif
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Fl_FontSize* fl_fontsize = 0L;
|
|
|
|
|
|
|
|
Fl_FontSize::~Fl_FontSize() {
|
|
|
|
/*
|
|
|
|
#if HAVE_GL
|
2002-06-27 08:29:39 +04:00
|
|
|
// ++ todo: remove OpenGL font alocations
|
2001-11-27 20:44:08 +03:00
|
|
|
// 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;
|
2004-08-26 04:18:43 +04:00
|
|
|
#ifdef __APPLE_QUARTZ__
|
2006-06-02 18:17:41 +04:00
|
|
|
ATSUDisposeTextLayout(layout);
|
|
|
|
ATSUDisposeStyle(style);
|
2004-08-26 04:18:43 +04:00
|
|
|
#endif
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
static Fl_Fontdesc built_in_table[] = {
|
2004-09-09 04:55:41 +04:00
|
|
|
#ifdef __APPLE_QD__
|
2001-11-27 20:44:08 +03:00
|
|
|
{" 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"},
|
|
|
|
{" Chicago"},
|
|
|
|
{"BChicago"},
|
|
|
|
{" Webdings"},
|
2004-09-09 04:55:41 +04:00
|
|
|
#elif defined(__APPLE_QUARTZ__)
|
|
|
|
{"Arial"},
|
|
|
|
{"Arial Bold"},
|
|
|
|
{"Arial Italic"},
|
|
|
|
{"Arial Bold Italic"},
|
|
|
|
{"Courier New"},
|
|
|
|
{"Courier New Bold"},
|
|
|
|
{"Courier New Italic"},
|
|
|
|
{"Courier New Bold Italic"},
|
|
|
|
{"Times New Roman"},
|
|
|
|
{"Times New Roman Bold"},
|
|
|
|
{"Times New Roman Italic"},
|
|
|
|
{"Times New Roman Bold Italic"},
|
|
|
|
{"Symbol"},
|
|
|
|
{"Monaco"},
|
|
|
|
{"Andale Mono"}, // there is no bold Monaco font on standard Mac
|
|
|
|
{"Webdings"},
|
|
|
|
#endif
|
2001-11-27 20:44:08 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
Fl_Fontdesc* fl_fonts = built_in_table;
|
|
|
|
|
|
|
|
void fl_font(Fl_FontSize* s) {
|
|
|
|
fl_fontsize = s;
|
2004-08-26 04:18:43 +04:00
|
|
|
#ifdef __APPLE_QD__
|
2001-12-20 08:27:14 +03:00
|
|
|
if (fl_window) SetPort( GetWindowPort(fl_window) );
|
2001-11-27 20:44:08 +03:00
|
|
|
TextFont(fl_fontsize->font); //: select font into current QuickDraw GC
|
|
|
|
TextFace(fl_fontsize->face);
|
|
|
|
TextSize(fl_fontsize->size);
|
2004-09-09 04:55:41 +04:00
|
|
|
if (!fl_fontsize->knowMetrics) { //: get the true metrics for the currnet GC
|
|
|
|
//: (fails on multiple monitors with different dpi's!)
|
2001-11-27 20:44:08 +03:00
|
|
|
FontInfo fi; GetFontInfo(&fi);
|
|
|
|
fl_fontsize->ascent = fi.ascent;
|
|
|
|
fl_fontsize->descent = fi.descent;
|
|
|
|
FMetricRec mr; FontMetrics(&mr);
|
|
|
|
short *f = (short*)*mr.wTabHandle; //: get the char size table
|
|
|
|
for (int i=0; i<256; i++) fl_fontsize->width[i] = f[2*i];
|
|
|
|
fl_fontsize->knowMetrics = 1;
|
|
|
|
}
|
2004-08-26 04:18:43 +04:00
|
|
|
#elif defined(__APPLE_QUARTZ__)
|
2006-06-02 18:17:41 +04:00
|
|
|
// we will use fl_fontsize later to access the required style and layout
|
2004-08-26 04:18:43 +04:00
|
|
|
#else
|
|
|
|
# error : need to defined either Quartz or Quickdraw
|
|
|
|
#endif
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static Fl_FontSize* find(int fnum, int size) {
|
|
|
|
Fl_Fontdesc* s = fl_fonts+fnum;
|
|
|
|
if (!s->name) s = fl_fonts; // use 0 if fnum undefined
|
|
|
|
Fl_FontSize* f;
|
|
|
|
for (f = s->first; f; f = f->next)
|
|
|
|
if (f->minsize <= size && f->maxsize >= size) return f;
|
|
|
|
f = new Fl_FontSize(s->name, size);
|
|
|
|
f->next = s->first;
|
|
|
|
s->first = f;
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// Public interface:
|
|
|
|
|
2003-04-06 16:54:57 +04:00
|
|
|
int fl_font_ = 0;
|
|
|
|
int fl_size_ = 0;
|
|
|
|
|
2001-11-27 20:44:08 +03:00
|
|
|
void fl_font(int fnum, int size) {
|
2003-04-06 16:54:57 +04:00
|
|
|
fl_font_ = fnum;
|
|
|
|
fl_size_ = size;
|
2001-11-27 20:44:08 +03:00
|
|
|
fl_font(find(fnum, size));
|
|
|
|
}
|
|
|
|
|
|
|
|
int fl_height() {
|
2004-09-08 00:59:18 +04:00
|
|
|
if (fl_fontsize) return fl_fontsize->ascent+fl_fontsize->descent;
|
|
|
|
else return -1;
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int fl_descent() {
|
2004-09-08 00:59:18 +04:00
|
|
|
if (fl_fontsize) return fl_fontsize->descent;
|
|
|
|
else return -1;
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
2006-04-18 17:07:42 +04:00
|
|
|
double fl_width(const char* txt, int n) {
|
2004-09-10 01:34:48 +04:00
|
|
|
#ifdef __APPLE_QD__
|
2004-09-13 04:30:08 +04:00
|
|
|
return (double)TextWidth( txt, 0, n );
|
2004-09-10 01:34:48 +04:00
|
|
|
#else
|
|
|
|
if (!fl_gc) {
|
|
|
|
Fl_Window *w = Fl::first_window();
|
|
|
|
if (w) w->make_current();
|
|
|
|
if (!fl_gc) return -1;
|
|
|
|
}
|
2006-06-02 18:17:41 +04:00
|
|
|
OSStatus err;
|
|
|
|
// convert to UTF-16 first
|
|
|
|
int i;
|
|
|
|
UniChar uniStr[n+1];
|
|
|
|
for (i=0; i<n; i++) {
|
|
|
|
uniStr[i] = txt[i];
|
|
|
|
}
|
|
|
|
uniStr[i] = 0;
|
|
|
|
// now collect our ATSU resources
|
|
|
|
ATSUTextLayout layout = fl_fontsize->layout;
|
|
|
|
err = ATSUSetTextPointerLocation(layout, uniStr, kATSUFromTextBeginning, n, n);
|
|
|
|
// now measure the bounding box
|
|
|
|
Fixed bBefore, bAfter, bAscent, bDescent;
|
|
|
|
err = ATSUGetUnjustifiedBounds(layout, kATSUFromTextBeginning, n, &bBefore, &bAfter, &bAscent, &bDescent);
|
|
|
|
return FixedToInt(bAfter);
|
2004-09-10 01:34:48 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
double fl_width(uchar c) {
|
|
|
|
return fl_width((const char*)(&c), 1);
|
|
|
|
}
|
2003-05-21 17:23:40 +04:00
|
|
|
|
2004-09-10 01:34:48 +04:00
|
|
|
void fl_draw(const char *str, int n, float x, float y);
|
2003-05-21 17:23:40 +04:00
|
|
|
|
2004-09-10 01:34:48 +04:00
|
|
|
void fl_draw(const char* str, int n, int x, int y) {
|
2004-08-26 04:18:43 +04:00
|
|
|
#ifdef __APPLE_QD__
|
2001-11-27 20:44:08 +03:00
|
|
|
MoveTo(x, y);
|
2006-04-18 17:07:42 +04:00
|
|
|
DrawText((const char *)str, 0, n);
|
2004-08-26 04:18:43 +04:00
|
|
|
#elif defined(__APPLE_QUARTZ__)
|
2004-09-10 01:34:48 +04:00
|
|
|
fl_draw(str, n, (float)x, (float)y);
|
2004-08-26 04:18:43 +04:00
|
|
|
#else
|
2004-09-13 04:30:08 +04:00
|
|
|
# error : neither Quartz no Quickdraw chosen
|
2004-08-26 04:18:43 +04:00
|
|
|
#endif
|
2001-11-27 20:44:08 +03:00
|
|
|
}
|
|
|
|
|
2004-09-10 01:34:48 +04:00
|
|
|
void fl_draw(const char *str, int n, float x, float y) {
|
|
|
|
#ifdef __APPLE_QD__
|
|
|
|
fl_draw(str, n, (int)x, (int)y);
|
|
|
|
#elif defined(__APPLE_QUARTZ__)
|
2006-06-02 18:17:41 +04:00
|
|
|
OSStatus err;
|
|
|
|
// convert to UTF-16 first
|
|
|
|
int i;
|
|
|
|
UniChar uniStr[n+1];
|
|
|
|
for (i=0; i<n; i++) {
|
|
|
|
uniStr[i] = str[i];
|
|
|
|
}
|
|
|
|
uniStr[i] = 0;
|
|
|
|
// now collect our ATSU resources
|
|
|
|
ATSUTextLayout layout = fl_fontsize->layout;
|
|
|
|
err = ATSUSetTextPointerLocation(layout, uniStr, kATSUFromTextBeginning, n, n);
|
|
|
|
err = ATSUDrawText(layout, kATSUFromTextBeginning, n, FloatToFixed(x), FloatToFixed(y));
|
2004-09-10 01:34:48 +04:00
|
|
|
#else
|
2004-09-13 04:30:08 +04:00
|
|
|
# error : neither Quartz no Quickdraw chosen
|
2004-09-10 01:34:48 +04:00
|
|
|
#endif
|
|
|
|
}
|
2003-05-21 17:23:40 +04:00
|
|
|
|
2001-11-27 20:44:08 +03:00
|
|
|
//
|
2005-02-25 00:55:12 +03:00
|
|
|
// End of "$Id$".
|
2001-11-27 20:44:08 +03:00
|
|
|
//
|