]> git.lyx.org Git - features.git/blob - src/frontends/qt4/GuiFontMetrics.C
Document symbol fonts hack (text from Georg Baum).
[features.git] / src / frontends / qt4 / GuiFontMetrics.C
1 /**
2  * \file GuiFontMetrics.C
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author unknown
7  * \author John Levon
8  *
9  * Full author contact details are available in file CREDITS.
10  */
11
12 #include <config.h>
13
14 #include "GuiFontMetrics.h"
15
16 #include "qt_helpers.h"
17
18 #include "language.h"
19 #include "dimension.h"
20
21 #include "support/unicode.h"
22
23 using std::string;
24
25 namespace lyx {
26 namespace frontend {
27
28 GuiFontMetrics::GuiFontMetrics(QFont const & font)
29 : metrics_(font), smallcaps_metrics_(font), smallcaps_shape_(false)
30 {
31 }
32
33
34 GuiFontMetrics::GuiFontMetrics(QFont const & font, QFont const & smallcaps_font)
35 : metrics_(font), smallcaps_metrics_(smallcaps_font), smallcaps_shape_(true)
36 {
37 }
38
39
40 int GuiFontMetrics::maxAscent() const
41 {
42         return metrics_.ascent();
43 }
44
45
46 int GuiFontMetrics::maxDescent() const
47 {
48         // We add 1 as the value returned by QT is different than X
49         // See http://doc.trolltech.com/2.3/qfontmetrics.html#200b74
50         return metrics_.descent() + 1;
51 }
52
53
54 int GuiFontMetrics::lbearing(char_type c) const
55 {
56         return metrics_.leftBearing(ucs4_to_qchar(c));
57 }
58
59
60 int GuiFontMetrics::rbearing(char_type c) const
61 {
62         // Qt rbearing is from the right edge of the char's width().
63         QChar sc = ucs4_to_qchar(c);
64         return metrics_.width(sc) - metrics_.rightBearing(sc);
65 }
66
67
68 int GuiFontMetrics::smallcapsWidth(QString const & s) const
69 {
70         int w = 0;
71         int const ls = s.size();
72
73         for (int i = 0; i < ls; ++i) {
74                 QChar const & c = s[i];
75                 QChar const uc = c.toUpper();
76                 if (c != uc)
77                         w += smallcaps_metrics_.width(uc);
78                 else
79                         w += metrics_.width(c);
80         }
81         return w;
82 }
83
84
85 int GuiFontMetrics::width(docstring const & s) const
86 {
87         size_t ls = s.size();
88         if (ls == 0)
89                 return 0;
90
91         if (ls == 1 && !smallcaps_shape_) {
92                 return width(s[0]);
93         }
94
95         if (smallcaps_shape_)
96                 // Caution: The following ucs4 to QString conversions work
97                 // for symbol fonts only because they are no real conversions
98                 // but simple casts in reality. See comment in QLPainter::text()
99                 // for more explanation.
100                 return smallcapsWidth(toqstr(s));
101
102         int w = 0;
103         for (unsigned int i = 0; i < ls; ++i)
104                 w += width(s[i]);
105
106         return w;
107 }
108
109
110 int GuiFontMetrics::width(QString const & ucs2) const
111 {
112         int const ls = ucs2.size();
113         if (ls == 1 && !smallcaps_shape_) {
114                 return width(ucs2[0].unicode());
115         }
116
117         if (smallcaps_shape_)
118                 return smallcapsWidth(ucs2);
119
120         int w = 0;
121         for (int i = 0; i < ls; ++i)
122                 w += width(ucs2[i].unicode());
123
124         return w;
125 }
126
127
128 int GuiFontMetrics::signedWidth(docstring const & s) const
129 {
130         if (s.empty())
131                 return 0;
132
133         if (s[0] == '-')
134                 return -width(s.substr(1, s.size() - 1));
135         else
136                 return width(s);
137 }
138
139
140 void GuiFontMetrics::rectText(docstring const & str,
141         int & w, int & ascent, int & descent) const
142 {
143         static int const d = 2;
144         w = width(str) + d * 2 + 2;
145         ascent = metrics_.ascent() + d;
146         descent = metrics_.descent() + d;
147 }
148
149
150
151 void GuiFontMetrics::buttonText(docstring const & str,
152         int & w, int & ascent, int & descent) const
153 {
154         static int const d = 3;
155         w = width(str) + d * 2 + 2;
156         ascent = metrics_.ascent() + d;
157         descent = metrics_.descent() + d;
158 }
159
160
161 Dimension const GuiFontMetrics::defaultDimension() const
162 {
163         return Dimension(0, maxAscent(), maxDescent());
164 }
165
166
167 Dimension const GuiFontMetrics::dimension(char_type c) const
168 {
169         return Dimension(width(c), ascent(c), descent(c));
170 }
171
172
173 void GuiFontMetrics::fillMetricsCache(char_type c) const
174 {
175         QRect const & r = metrics_.boundingRect(ucs4_to_qchar(c));
176         AscendDescend ad = { -r.top(), r.bottom() + 1};
177         // We could as well compute the width but this is not really
178         // needed for now as it is done directly in width() below.
179         metrics_cache_.insert(c, ad);
180 }
181
182
183 int GuiFontMetrics::width(char_type c) const
184 {
185         if (smallcaps_shape_)
186                 return smallcapsWidth(ucs4_to_qchar(c));
187
188         if (!width_cache_.contains(c)) {
189                 width_cache_.insert(c, metrics_.width(ucs4_to_qchar(c)));
190         }
191
192         return width_cache_.value(c);
193 }
194
195
196 int GuiFontMetrics::ascent(char_type c) const
197 {
198         if (!metrics_cache_.contains(c))
199                 fillMetricsCache(c);
200
201         return metrics_cache_.value(c).ascent;
202 }
203
204
205 int GuiFontMetrics::descent(char_type c) const
206 {
207         if (!metrics_cache_.contains(c))
208                 fillMetricsCache(c);
209
210         return metrics_cache_.value(c).descent;
211 }
212
213 } // frontend
214 } // lyx