fluxbox/src/FbTk/XftFontImp.cc

254 lines
7.4 KiB
C++

// XftFontImp.cc Xft font implementation for FbTk
// Copyright (c) 2002 - 2006 Henrik Kinnunen (fluxgen at fluxbox dot org)
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
//$Id$
#include "XftFontImp.hh"
#include "App.hh"
#include "FbDrawable.hh"
#include <math.h>
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif //HAVE_CONFIG_H
namespace FbTk {
XftFontImp::XftFontImp(const char *name, bool utf8):
m_utf8mode(utf8), m_name("") {
for (int r = ROT0; r <= ROT270; r++) {
m_xftfonts[r] = 0;
m_xftfonts_loaded[r] = false;
}
if (name != 0)
load(name);
}
XftFontImp::~XftFontImp() {
for (int r = ROT0; r <= ROT270; r++)
if (m_xftfonts[r] != 0)
XftFontClose(App::instance()->display(), m_xftfonts[r]);
}
bool XftFontImp::load(const std::string &name) {
//Note: assumes screen 0 for now, changes on draw if needed
Display *disp = App::instance()->display();
XftFont *newxftfont = XftFontOpenXlfd(disp, 0, name.c_str());
if (newxftfont == 0) {
newxftfont = XftFontOpenName(disp, 0, name.c_str());
if (newxftfont == 0)
return false;
}
// destroy all old fonts and set new
for (int r = ROT0; r <= ROT270; r++) {
m_xftfonts_loaded[r] = false;
if (m_xftfonts[r] != 0) {
XftFontClose(App::instance()->display(), m_xftfonts[r]);
m_xftfonts[r] = 0;
}
}
m_xftfonts[ROT0] = newxftfont;
m_xftfonts_loaded[ROT0] = true;
m_name = name;
return true;
}
void XftFontImp::drawText(const FbDrawable &w, int screen, GC gc, const FbString &text, size_t len, int x, int y, FbTk::Orientation orient) {
if (!validOrientation(orient))
return;
// we adjust y slightly so that the baseline is in the right spot
// (it is offset one by rotation >=180 degrees)
switch (orient) {
case ROT0:
break;
case ROT90:
break;
case ROT180:
x+=1;
y+=1;
break;
case ROT270:
y+=1;
break;
}
XftFont *font = m_xftfonts[orient];
XftDraw *draw = XftDrawCreate(w.display(),
w.drawable(),
DefaultVisual(w.display(), screen),
DefaultColormap(w.display(), screen));
XGCValues gc_val;
// get foreground pixel value and convert it to XRenderColor value
// TODO: we should probably check return status
XGetGCValues(w.display(), gc, GCForeground, &gc_val);
// get red, green, blue values
XColor xcol;
xcol.pixel = gc_val.foreground;
XQueryColor(w.display(), DefaultColormap(w.display(), screen), &xcol);
// convert xcolor to XftColor
XRenderColor rendcol;
rendcol.red = xcol.red;
rendcol.green = xcol.green;
rendcol.blue = xcol.blue;
rendcol.alpha = 0xFFFF;
XftColor xftcolor;
XftColorAllocValue(w.display(),
DefaultVisual(w.display(), screen),
DefaultColormap(w.display(), screen),
&rendcol, &xftcolor);
// draw string
#ifdef HAVE_XFT_UTF8_STRING
if (m_utf8mode) {
// check the string size,
// if the size is zero we use the XftDrawString8 function instead.
XGlyphInfo ginfo;
XftTextExtentsUtf8(w.display(),
m_xftfonts[ROT0],
(XftChar8 *)text.data(), len,
&ginfo);
if (ginfo.xOff != 0) {
XftDrawStringUtf8(draw,
&xftcolor,
font,
x, y,
(XftChar8 *)(text.data()), len);
XftColorFree(w.display(),
DefaultVisual(w.display(), screen),
DefaultColormap(w.display(), screen), &xftcolor);
XftDrawDestroy(draw);
return;
}
}
#endif // HAVE_XFT_UTF8_STRING
XftDrawString8(draw,
&xftcolor,
font,
x, y,
(XftChar8 *)(text.data()), len);
XftColorFree(w.display(),
DefaultVisual(w.display(), screen),
DefaultColormap(w.display(), screen), &xftcolor);
XftDrawDestroy(draw);
}
unsigned int XftFontImp::textWidth(const FbString &text, unsigned int len) const {
if (m_xftfonts[ROT0] == 0)
return 0;
XGlyphInfo ginfo;
Display* disp = App::instance()->display();
XftFont *font = m_xftfonts[ROT0];
#ifdef HAVE_XFT_UTF8_STRING
if (m_utf8mode) {
XftTextExtentsUtf8(disp,
font,
(XftChar8 *)text.data(), len,
&ginfo);
if (ginfo.xOff != 0)
return ginfo.xOff;
// the utf8 failed, try normal extents
}
#endif //HAVE_XFT_UTF8_STRING
std::string localestr = text;
localestr.erase(len, std::string::npos);
localestr = FbStringUtil::FbStrToLocale(localestr);
XftTextExtents8(disp,
font,
(XftChar8 *)localestr.data(), localestr.size(),
&ginfo);
return ginfo.xOff;
}
unsigned int XftFontImp::height() const {
if (m_xftfonts[ROT0] == 0)
return 0;
else
return m_xftfonts[ROT0]->height;
//m_xftfont->ascent + m_xftfont->descent;
// curiously, fonts seem to have a smaller height, but the "height"
// is specified within the actual font, so it must be right, right?
}
bool XftFontImp::validOrientation(FbTk::Orientation orient) {
if (orient == ROT0 || m_xftfonts[orient])
return true;
if (m_xftfonts_loaded[orient])
return false; // m_xftfonts is zero here
if (m_xftfonts[ROT0] == 0)
return false;
m_xftfonts_loaded[orient] = true;
// otherwise, try to load that orientation
// radians is actually anti-clockwise, so we reverse it
double radians = -(orient) * 90 * M_PI / 180;
XftMatrix matrix;
XftMatrixInit(&matrix);
XftMatrixRotate(&matrix, cos(radians), sin(radians));
Display *disp = App::instance()->display();
XftPattern * pattern = XftNameParse(m_name.c_str());
XftResult result;
pattern = XftFontMatch(disp, 0, pattern, &result);
XftPatternAddMatrix(pattern, XFT_MATRIX, &matrix);
XftFont * new_font = XftFontOpenPattern(disp, pattern);
if (new_font == 0)
return false;
m_xftfonts[orient] = new_font;
return true;
}
}; // end namespace FbTk