3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
6 * \author Asger Alstrup
8 * Full author contact details are available in file CREDITS
12 #include <cmath> // fabs()
15 #pragma implementation
18 #include "xfont_loader.h"
22 #include "lyxrc.h" // lyxrc.font_*
23 #include "BufferView.h"
24 #include "frontends/LyXView.h"
25 #include "support/systemcall.h"
26 #include "support/filetools.h"
28 #include FORMS_H_LOCATION
32 // The global fontloader
33 xfont_loader fontloader;
36 // Initialize font loader
37 xfont_loader::xfont_loader()
43 // Destroy font loader
44 xfont_loader::~xfont_loader()
50 // Update fonts after zoom, dpi, font names, or norm change
51 // For now, we just ditch all fonts we have. Later, we should
52 // reuse the ones that are already loaded.
53 void xfont_loader::update()
60 void xfont_loader::reset()
62 // Clear font infos, font structs and font metrics
63 for (int i1 = 0; i1 < LyXFont::NUM_FAMILIES; ++i1)
64 for (int i2 = 0; i2 < 2; ++i2)
65 for (int i3 = 0; i3 < 4; ++i3) {
66 fontinfo[i1][i2][i3] = 0;
67 for (int i4 = 0; i4<10; ++i4) {
68 fontstruct[i1][i2][i3][i4] = 0;
75 void xfont_loader::unload()
78 for (int i1 = 0; i1 < LyXFont::NUM_FAMILIES; ++i1)
79 for (int i2 = 0; i2 < 2; ++i2)
80 for (int i3 = 0; i3 < 4; ++i3) {
81 if (fontinfo[i1][i2][i3]) {
82 delete fontinfo[i1][i2][i3];
83 fontinfo[i1][i2][i3] = 0;
85 for (int i4 = 0; i4 < 10; ++i4) {
86 if (fontstruct[i1][i2][i3][i4]) {
87 XFreeFont(fl_get_display(), fontstruct[i1][i2][i3][i4]);
88 fontstruct[i1][i2][i3][i4] = 0;
96 string const symbolPattern(LyXFont::FONT_FAMILY family)
99 case LyXFont::SYMBOL_FAMILY:
100 return "-*-symbol-*-*-*-*-*-*-*-*-*-*-adobe-fontspecific";
102 case LyXFont::CMR_FAMILY:
103 return "-*-cmr10-medium-*-*-*-*-*-*-*-*-*-*-*";
105 case LyXFont::CMSY_FAMILY:
106 return "-*-cmsy10-*-*-*-*-*-*-*-*-*-*-*-*";
108 case LyXFont::CMM_FAMILY:
109 return "-*-cmmi10-medium-*-*-*-*-*-*-*-*-*-*-*";
111 case LyXFont::CMEX_FAMILY:
112 return "-*-cmex10-*-*-*-*-*-*-*-*-*-*-*-*";
114 case LyXFont::MSA_FAMILY:
115 return "-*-msam10-*-*-*-*-*-*-*-*-*-*-*-*";
117 case LyXFont::MSB_FAMILY:
118 return "-*-msbm10-*-*-*-*-*-*-*-*-*-*-*-*";
120 case LyXFont::EUFRAK_FAMILY:
121 return "-*-eufm10-medium-*-*-*-*-*-*-*-*-*-*-*";
123 case LyXFont::WASY_FAMILY:
124 return "-*-wasy10-medium-*-*-*-*-*-*-*-*-*-*-*";
133 string const dir = OnlyPath(LibFileSearch("xfonts", "fonts.dir"));
136 char ** p = XGetFontPath(fl_get_display(), &n);
137 if (std::find(p, p + n, dir) != p + n)
139 lyxerr << "Adding " << dir << " to the font path.\n";
140 string const command = "xset fp+ " + dir;
142 if (!s.startscript(Systemcall::Wait, command))
144 lyxerr << "Unable to add font path.\n";
152 /* Takes care of finding which font that can match the given request. Tries
153 different alternatives. */
154 void xfont_loader::getFontinfo(LyXFont::FONT_FAMILY family,
155 LyXFont::FONT_SERIES series,
156 LyXFont::FONT_SHAPE shape)
158 // Do we have the font info already?
159 if (fontinfo[family][series][shape] != 0)
163 string pat = symbolPattern(family);
165 static bool first_time = true;
166 fontinfo[family][series][shape] = new FontInfo(pat);
167 if (family != LyXFont::SYMBOL_FAMILY &&
168 !fontinfo[family][series][shape]->exist() &&
172 delete fontinfo[family][series][shape];
173 fontinfo[family][series][shape] = new FontInfo(pat);
180 // Normal font. Let's search for an existing name that matches.
184 string norm = lyxrc.font_norm;
187 FontInfo * fi = new FontInfo;
188 fontinfo[family][series][shape] = fi;
190 for (int cfam = 0; cfam < 2; ++cfam) {
191 // Determine family name
193 case LyXFont::ROMAN_FAMILY:
195 case 0: ffamily = lyxrc.roman_font_name; break;
196 case 1: ffamily = "-*-times";
200 case LyXFont::SANS_FAMILY:
202 case 0: ffamily = lyxrc.sans_font_name; break;
203 case 1: ffamily = "-*-helvetica";
207 case LyXFont::TYPEWRITER_FAMILY:
209 case 0: ffamily = lyxrc.typewriter_font_name; break;
210 case 1: ffamily = "-*-courier";
217 for (int cser = 0; cser < 4; ++cser) {
218 // Determine series name
220 case LyXFont::MEDIUM_SERIES:
222 case 0: fseries = "-medium"; break;
223 case 1: fseries = "-book"; break;
224 case 2: fseries = "-light";
228 case LyXFont::BOLD_SERIES:
230 case 0: fseries = "-bold"; break;
231 case 1: fseries = "-black"; break;
232 case 2: fseries = "-demi"; break;
233 case 3: fseries = "-demibold";
240 for (int csha = 0; csha < 2; ++csha) {
241 // Determine shape name
243 case LyXFont::UP_SHAPE:
244 case LyXFont::SMALLCAPS_SHAPE:
246 case 0: fshape = "-r";
250 case LyXFont::ITALIC_SHAPE:
252 case 0: fshape = "-i"; break;
253 case 1: fshape = "-o";
257 case LyXFont::SLANTED_SHAPE:
259 case 0: fshape = "-o"; break;
260 case 1: fshape = "-i";
267 fontname = ffamily + fseries + fshape +
268 "-normal-*-*-*-*-*-*-*-" + norm;
269 fi->setPattern(fontname);
279 // A dummy fontstruct used when there is no gui.
282 XFontStruct dummyXFontStruct;
283 bool dummyXFontStructisGood = false;
288 XFontStruct * xfont_loader::doLoad(LyXFont::FONT_FAMILY family,
289 LyXFont::FONT_SERIES series,
290 LyXFont::FONT_SHAPE shape,
291 LyXFont::FONT_SIZE size)
293 if (!lyxrc.use_gui) {
294 if (!dummyXFontStructisGood) {
295 // no character specific info
296 dummyXFontStruct.per_char = 0;
297 // unit ascent on character displays
298 dummyXFontStruct.ascent = 1;
299 // no descent on character displays
300 dummyXFontStruct.descent = 0;
301 dummyXFontStructisGood = true;
304 return &dummyXFontStruct;
307 getFontinfo(family, series, shape);
308 // FIXME! CHECK! Should we use 72.0 or 72.27? (Lgb)
309 int fsize = int((lyxrc.font_sizes[size] * lyxrc.dpi *
310 (lyxrc.zoom/100.0)) / 72.27 + 0.5);
312 string font = fontinfo[family][series][shape]->getFontname(fsize);
315 lyxerr << "No font matches request. Using 'fixed'." << endl;
316 lyxerr << "Start LyX as 'lyx -dbg 515' to get more information." << endl;
320 XFontStruct * fs = 0;
322 fs = XLoadQueryFont(fl_get_display(), font.c_str());
325 if (font == "fixed") {
326 lyxerr << "We're doomed. Can't get 'fixed' font." << endl;
328 lyxerr << "Could not get font '" << font
329 << "'. Using 'fixed'." << endl;
330 fs = XLoadQueryFont(fl_get_display(), "fixed");
332 } else if (lyxerr.debugging(Debug::FONT)) {
333 // Tell user the font matching
339 // The rest of the attributes are not interesting
340 f.setEmph(LyXFont::INHERIT);
341 f.setUnderbar(LyXFont::INHERIT);
342 f.setNoun(LyXFont::INHERIT);
343 f.setColor(LColor::inherit);
344 lyxerr << "Font '" << f.stateText(0)
345 << "' matched by\n" << font << endl;
348 fontstruct[family][series][shape][size] = fs;
353 bool xfont_loader::available(LyXFont const & f)
358 if (!fontinfo[f.family()][f.series()][f.realShape()])
359 getFontinfo(f.family(), f.series(), f.realShape());
360 return fontinfo[f.family()][f.series()][f.realShape()]->exist();