mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
436 lines
12 KiB
C
436 lines
12 KiB
C
|
/*
|
||
|
* Unified font management for GTK.
|
||
|
*
|
||
|
* PuTTY is willing to use both old-style X server-side bitmap
|
||
|
* fonts _and_ GTK2/Pango client-side fonts. This requires us to
|
||
|
* do a bit of work to wrap the two wildly different APIs into
|
||
|
* forms the rest of the code can switch between seamlessly, and
|
||
|
* also requires a custom font selector capable of handling both
|
||
|
* types of font.
|
||
|
*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <gtk/gtk.h>
|
||
|
#include <gdk/gdkkeysyms.h>
|
||
|
#include <gdk/gdkx.h>
|
||
|
#include <X11/Xlib.h>
|
||
|
#include <X11/Xutil.h>
|
||
|
#include <X11/Xatom.h>
|
||
|
|
||
|
#include "putty.h"
|
||
|
#include "gtkfont.h"
|
||
|
|
||
|
/*
|
||
|
* To do:
|
||
|
*
|
||
|
* - import flags to do VT100 double-width, and import the icky
|
||
|
* pixmap stretch code for it.
|
||
|
*
|
||
|
* - add the Pango back end!
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Future work:
|
||
|
*
|
||
|
* - all the GDK font functions used in the x11font subclass are
|
||
|
* deprecated, so one day they may go away. When this happens -
|
||
|
* or before, if I'm feeling proactive - it oughtn't to be too
|
||
|
* difficult in principle to convert the whole thing to use
|
||
|
* actual Xlib font calls.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Ad-hoc vtable mechanism to allow font structures to be
|
||
|
* polymorphic.
|
||
|
*
|
||
|
* Any instance of `unifont' used in the vtable functions will
|
||
|
* actually be the first element of a larger structure containing
|
||
|
* data specific to the subtype. This is permitted by the ISO C
|
||
|
* provision that one may safely cast between a pointer to a
|
||
|
* structure and a pointer to its first element.
|
||
|
*/
|
||
|
|
||
|
struct unifont_vtable {
|
||
|
/*
|
||
|
* `Methods' of the `class'.
|
||
|
*/
|
||
|
unifont *(*create)(char *name, int wide, int bold,
|
||
|
int shadowoffset, int shadowalways);
|
||
|
void (*destroy)(unifont *font);
|
||
|
void (*draw_text)(GdkDrawable *target, GdkGC *gc, unifont *font,
|
||
|
int x, int y, const char *string, int len, int wide,
|
||
|
int bold);
|
||
|
/*
|
||
|
* `Static data members' of the `class'.
|
||
|
*/
|
||
|
const char *prefix;
|
||
|
};
|
||
|
|
||
|
/* ----------------------------------------------------------------------
|
||
|
* GDK-based X11 font implementation.
|
||
|
*/
|
||
|
|
||
|
static void x11font_draw_text(GdkDrawable *target, GdkGC *gc, unifont *font,
|
||
|
int x, int y, const char *string, int len,
|
||
|
int wide, int bold);
|
||
|
static unifont *x11font_create(char *name, int wide, int bold,
|
||
|
int shadowoffset, int shadowalways);
|
||
|
static void x11font_destroy(unifont *font);
|
||
|
|
||
|
struct x11font {
|
||
|
struct unifont u;
|
||
|
/*
|
||
|
* Actual font objects. We store a number of these, for
|
||
|
* automatically guessed bold and wide variants.
|
||
|
*
|
||
|
* The parallel array `allocated' indicates whether we've
|
||
|
* tried to fetch a subfont already (thus distinguishing NULL
|
||
|
* because we haven't tried yet from NULL because we tried and
|
||
|
* failed, so that we don't keep trying and failing
|
||
|
* subsequently).
|
||
|
*/
|
||
|
GdkFont *fonts[4];
|
||
|
int allocated[4];
|
||
|
/*
|
||
|
* `sixteen_bit' is true iff the font object is indexed by
|
||
|
* values larger than a byte. That is, this flag tells us
|
||
|
* whether we use gdk_draw_text_wc() or gdk_draw_text().
|
||
|
*/
|
||
|
int sixteen_bit;
|
||
|
/*
|
||
|
* Font charsets. public_charset and real_charset can differ
|
||
|
* for X11 fonts, because many X fonts use CS_ISO8859_1_X11.
|
||
|
*/
|
||
|
int public_charset, real_charset;
|
||
|
/*
|
||
|
* Data passed in to unifont_create().
|
||
|
*/
|
||
|
int wide, bold, shadowoffset, shadowalways;
|
||
|
};
|
||
|
|
||
|
static const struct unifont_vtable x11font_vtable = {
|
||
|
x11font_create,
|
||
|
x11font_destroy,
|
||
|
x11font_draw_text,
|
||
|
"x11"
|
||
|
};
|
||
|
|
||
|
char *x11_guess_derived_font_name(GdkFont *font, int bold, int wide)
|
||
|
{
|
||
|
XFontStruct *xfs = GDK_FONT_XFONT(font);
|
||
|
Display *disp = GDK_FONT_XDISPLAY(font);
|
||
|
Atom fontprop = XInternAtom(disp, "FONT", False);
|
||
|
unsigned long ret;
|
||
|
if (XGetFontProperty(xfs, fontprop, &ret)) {
|
||
|
char *name = XGetAtomName(disp, (Atom)ret);
|
||
|
if (name && name[0] == '-') {
|
||
|
char *strings[13];
|
||
|
char *dupname, *extrafree = NULL, *ret;
|
||
|
char *p, *q;
|
||
|
int nstr;
|
||
|
|
||
|
p = q = dupname = dupstr(name); /* skip initial minus */
|
||
|
nstr = 0;
|
||
|
|
||
|
while (*p && nstr < lenof(strings)) {
|
||
|
if (*p == '-') {
|
||
|
*p = '\0';
|
||
|
strings[nstr++] = p+1;
|
||
|
}
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
if (nstr < lenof(strings))
|
||
|
return NULL; /* XLFD was malformed */
|
||
|
|
||
|
if (bold)
|
||
|
strings[2] = "bold";
|
||
|
|
||
|
if (wide) {
|
||
|
/* 4 is `wideness', which obviously may have changed. */
|
||
|
/* 5 is additional style, which may be e.g. `ja' or `ko'. */
|
||
|
strings[4] = strings[5] = "*";
|
||
|
strings[11] = extrafree = dupprintf("%d", 2*atoi(strings[11]));
|
||
|
}
|
||
|
|
||
|
ret = dupcat("-", strings[ 0], "-", strings[ 1], "-", strings[ 2],
|
||
|
"-", strings[ 3], "-", strings[ 4], "-", strings[ 5],
|
||
|
"-", strings[ 6], "-", strings[ 7], "-", strings[ 8],
|
||
|
"-", strings[ 9], "-", strings[10], "-", strings[11],
|
||
|
"-", strings[12], NULL);
|
||
|
sfree(extrafree);
|
||
|
sfree(dupname);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
static int x11_font_width(GdkFont *font, int sixteen_bit)
|
||
|
{
|
||
|
if (sixteen_bit) {
|
||
|
XChar2b space;
|
||
|
space.byte1 = 0;
|
||
|
space.byte2 = ' ';
|
||
|
return gdk_text_width(font, (const gchar *)&space, 2);
|
||
|
} else {
|
||
|
return gdk_char_width(font, ' ');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static unifont *x11font_create(char *name, int wide, int bold,
|
||
|
int shadowoffset, int shadowalways)
|
||
|
{
|
||
|
struct x11font *xfont;
|
||
|
GdkFont *font;
|
||
|
XFontStruct *xfs;
|
||
|
Display *disp;
|
||
|
Atom charset_registry, charset_encoding;
|
||
|
unsigned long registry_ret, encoding_ret;
|
||
|
int pubcs, realcs, sixteen_bit;
|
||
|
int i;
|
||
|
|
||
|
font = gdk_font_load(name);
|
||
|
if (!font)
|
||
|
return NULL;
|
||
|
|
||
|
xfs = GDK_FONT_XFONT(font);
|
||
|
disp = GDK_FONT_XDISPLAY(font);
|
||
|
|
||
|
charset_registry = XInternAtom(disp, "CHARSET_REGISTRY", False);
|
||
|
charset_encoding = XInternAtom(disp, "CHARSET_ENCODING", False);
|
||
|
|
||
|
pubcs = realcs = CS_NONE;
|
||
|
sixteen_bit = FALSE;
|
||
|
|
||
|
if (XGetFontProperty(xfs, charset_registry, ®istry_ret) &&
|
||
|
XGetFontProperty(xfs, charset_encoding, &encoding_ret)) {
|
||
|
char *reg, *enc;
|
||
|
reg = XGetAtomName(disp, (Atom)registry_ret);
|
||
|
enc = XGetAtomName(disp, (Atom)encoding_ret);
|
||
|
if (reg && enc) {
|
||
|
char *encoding = dupcat(reg, "-", enc, NULL);
|
||
|
pubcs = realcs = charset_from_xenc(encoding);
|
||
|
|
||
|
/*
|
||
|
* iso10646-1 is the only wide font encoding we
|
||
|
* support. In this case, we expect clients to give us
|
||
|
* UTF-8, which this module must internally convert
|
||
|
* into 16-bit Unicode.
|
||
|
*/
|
||
|
if (!strcasecmp(encoding, "iso10646-1")) {
|
||
|
sixteen_bit = TRUE;
|
||
|
pubcs = realcs = CS_UTF8;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Hack for X line-drawing characters: if the primary
|
||
|
* font is encoded as ISO-8859-1, and has valid glyphs
|
||
|
* in the first 32 char positions, it is assumed that
|
||
|
* those glyphs are the VT100 line-drawing character
|
||
|
* set.
|
||
|
*
|
||
|
* Actually, we'll hack even harder by only checking
|
||
|
* position 0x19 (vertical line, VT100 linedrawing
|
||
|
* `x'). Then we can check it easily by seeing if the
|
||
|
* ascent and descent differ.
|
||
|
*/
|
||
|
if (pubcs == CS_ISO8859_1) {
|
||
|
int lb, rb, wid, asc, desc;
|
||
|
gchar text[2];
|
||
|
|
||
|
text[1] = '\0';
|
||
|
text[0] = '\x12';
|
||
|
gdk_string_extents(font, text, &lb, &rb, &wid, &asc, &desc);
|
||
|
if (asc != desc)
|
||
|
realcs = CS_ISO8859_1_X11;
|
||
|
}
|
||
|
|
||
|
sfree(encoding);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
xfont = snew(struct x11font);
|
||
|
xfont->u.vt = &x11font_vtable;
|
||
|
xfont->u.width = x11_font_width(font, sixteen_bit);
|
||
|
xfont->u.ascent = font->ascent;
|
||
|
xfont->u.descent = font->descent;
|
||
|
xfont->u.height = xfont->u.ascent + xfont->u.descent;
|
||
|
xfont->u.public_charset = pubcs;
|
||
|
xfont->u.real_charset = realcs;
|
||
|
xfont->fonts[0] = font;
|
||
|
xfont->allocated[0] = TRUE;
|
||
|
xfont->sixteen_bit = sixteen_bit;
|
||
|
xfont->wide = wide;
|
||
|
xfont->bold = bold;
|
||
|
xfont->shadowoffset = shadowoffset;
|
||
|
xfont->shadowalways = shadowalways;
|
||
|
|
||
|
for (i = 1; i < lenof(xfont->fonts); i++) {
|
||
|
xfont->fonts[i] = NULL;
|
||
|
xfont->allocated[i] = FALSE;
|
||
|
}
|
||
|
|
||
|
return (unifont *)xfont;
|
||
|
}
|
||
|
|
||
|
static void x11font_destroy(unifont *font)
|
||
|
{
|
||
|
struct x11font *xfont = (struct x11font *)font;
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < lenof(xfont->fonts); i++)
|
||
|
if (xfont->fonts[i])
|
||
|
gdk_font_unref(xfont->fonts[i]);
|
||
|
sfree(font);
|
||
|
}
|
||
|
|
||
|
static void x11_alloc_subfont(struct x11font *xfont, int sfid)
|
||
|
{
|
||
|
char *derived_name = x11_guess_derived_font_name
|
||
|
(xfont->fonts[0], sfid & 1, !!(sfid & 2));
|
||
|
xfont->fonts[sfid] = gdk_font_load(derived_name); /* may be NULL */
|
||
|
xfont->allocated[sfid] = TRUE;
|
||
|
sfree(derived_name);
|
||
|
}
|
||
|
|
||
|
static void x11font_draw_text(GdkDrawable *target, GdkGC *gc, unifont *font,
|
||
|
int x, int y, const char *string, int len,
|
||
|
int wide, int bold)
|
||
|
{
|
||
|
struct x11font *xfont = (struct x11font *)font;
|
||
|
int sfid;
|
||
|
int shadowbold = FALSE;
|
||
|
|
||
|
wide -= xfont->wide;
|
||
|
bold -= xfont->bold;
|
||
|
|
||
|
/*
|
||
|
* Decide which subfont we're using, and whether we have to
|
||
|
* use shadow bold.
|
||
|
*/
|
||
|
if (xfont->shadowalways && bold) {
|
||
|
shadowbold = TRUE;
|
||
|
bold = 0;
|
||
|
}
|
||
|
sfid = 2 * wide + bold;
|
||
|
if (!xfont->allocated[sfid])
|
||
|
x11_alloc_subfont(xfont, sfid);
|
||
|
if (bold && !xfont->fonts[sfid]) {
|
||
|
bold = 0;
|
||
|
shadowbold = TRUE;
|
||
|
sfid = 2 * wide + bold;
|
||
|
if (!xfont->allocated[sfid])
|
||
|
x11_alloc_subfont(xfont, sfid);
|
||
|
}
|
||
|
|
||
|
if (!xfont->fonts[sfid])
|
||
|
return; /* we've tried our best, but no luck */
|
||
|
|
||
|
if (xfont->sixteen_bit) {
|
||
|
/*
|
||
|
* This X font has 16-bit character indices, which means
|
||
|
* we expect our string to have been passed in UTF-8.
|
||
|
*/
|
||
|
XChar2b *xcs;
|
||
|
wchar_t *wcs;
|
||
|
int nchars, maxchars, i;
|
||
|
|
||
|
/*
|
||
|
* Convert the input string to wide-character Unicode.
|
||
|
*/
|
||
|
maxchars = 0;
|
||
|
for (i = 0; i < len; i++)
|
||
|
if ((unsigned char)string[i] <= 0x7F ||
|
||
|
(unsigned char)string[i] >= 0xC0)
|
||
|
maxchars++;
|
||
|
wcs = snewn(maxchars+1, wchar_t);
|
||
|
nchars = charset_to_unicode((char **)&string, &len, wcs, maxchars,
|
||
|
CS_UTF8, NULL, NULL, 0);
|
||
|
assert(nchars <= maxchars);
|
||
|
wcs[nchars] = L'\0';
|
||
|
|
||
|
xcs = snewn(nchars, XChar2b);
|
||
|
for (i = 0; i < nchars; i++) {
|
||
|
xcs[i].byte1 = wcs[i] >> 8;
|
||
|
xcs[i].byte2 = wcs[i];
|
||
|
}
|
||
|
|
||
|
gdk_draw_text(target, xfont->fonts[sfid], gc,
|
||
|
x, y, (gchar *)xcs, nchars*2);
|
||
|
if (shadowbold)
|
||
|
gdk_draw_text(target, xfont->fonts[sfid], gc,
|
||
|
x + xfont->shadowoffset, y, (gchar *)xcs, nchars*2);
|
||
|
sfree(xcs);
|
||
|
sfree(wcs);
|
||
|
} else {
|
||
|
gdk_draw_text(target, xfont->fonts[sfid], gc, x, y, string, len);
|
||
|
if (shadowbold)
|
||
|
gdk_draw_text(target, xfont->fonts[sfid], gc,
|
||
|
x + xfont->shadowoffset, y, string, len);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ----------------------------------------------------------------------
|
||
|
* Outermost functions which do the vtable dispatch.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* This function is the only one which needs to know the full set
|
||
|
* of font implementations available, because it has to try each
|
||
|
* in turn until one works, or condition on an override prefix in
|
||
|
* the font name.
|
||
|
*/
|
||
|
static const struct unifont_vtable *unifont_types[] = {
|
||
|
&x11font_vtable,
|
||
|
};
|
||
|
unifont *unifont_create(char *name, int wide, int bold,
|
||
|
int shadowoffset, int shadowalways)
|
||
|
{
|
||
|
int colonpos = strcspn(name, ":");
|
||
|
int i;
|
||
|
|
||
|
if (name[colonpos]) {
|
||
|
/*
|
||
|
* There's a colon prefix on the font name. Use it to work
|
||
|
* out which subclass to try to create.
|
||
|
*/
|
||
|
for (i = 0; i < lenof(unifont_types); i++) {
|
||
|
if (strlen(unifont_types[i]->prefix) == colonpos &&
|
||
|
!strncmp(unifont_types[i]->prefix, name, colonpos))
|
||
|
break;
|
||
|
}
|
||
|
if (i == lenof(unifont_types))
|
||
|
return NULL; /* prefix not recognised */
|
||
|
return unifont_types[i]->create(name+colonpos+1, wide, bold,
|
||
|
shadowoffset, shadowalways);
|
||
|
} else {
|
||
|
/*
|
||
|
* No colon prefix, so just go through all the subclasses.
|
||
|
*/
|
||
|
for (i = 0; i < lenof(unifont_types); i++) {
|
||
|
unifont *ret = unifont_types[i]->create(name, wide, bold,
|
||
|
shadowoffset,
|
||
|
shadowalways);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
}
|
||
|
return NULL; /* font not found in any scheme */
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void unifont_destroy(unifont *font)
|
||
|
{
|
||
|
font->vt->destroy(font);
|
||
|
}
|
||
|
|
||
|
void unifont_draw_text(GdkDrawable *target, GdkGC *gc, unifont *font,
|
||
|
int x, int y, const char *string, int len,
|
||
|
int wide, int bold)
|
||
|
{
|
||
|
font->vt->draw_text(target, gc, font, x, y, string, len, wide, bold);
|
||
|
}
|