]> git.lyx.org Git - features.git/blob - src/frontends/gtk/GPainter.C
the 'fitCursor fix'
[features.git] / src / frontends / gtk / GPainter.C
1 /**
2  * \file GPainter.C
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Huang Ying
7  *
8  * Full author contact details are available in file CREDITS.
9  */
10
11 #include <config.h>
12 #include <gtkmm.h>
13
14 #include "GPainter.h"
15 #include "debug.h"
16 #include "GWorkArea.h"
17 #include "lyxrc.h"
18 #include "encoding.h"
19 #include "language.h"
20 #include "LColor.h"
21 #include "xftFontLoader.h"
22 #include "xformsImage.h"
23 #include "frontends/font_metrics.h"
24 #include "codeConvert.h"
25
26 #include "support/lstrings.h"
27
28 #include <boost/scoped_array.hpp>
29
30 #include <X11/Xft/Xft.h>
31
32 #include <cmath>
33
34 using std::string;
35
36
37 GPainter::GPainter(GWorkArea & xwa)
38         : Painter(), owner_(xwa)
39 {
40 }
41
42
43 int GPainter::paperWidth() const
44 {
45         return owner_.workWidth();
46 }
47
48
49 int GPainter::paperHeight() const
50 {
51         return owner_.workHeight();
52 }
53
54
55 void GPainter::setForeground(Glib::RefPtr<Gdk::GC> gc, LColor_color clr)
56 {
57         Gdk::Color * gclr = owner_.getColorHandler().getGdkColor(clr);
58         gc->set_foreground(*gclr);
59 }
60
61
62 void GPainter::setLineParam(Glib::RefPtr<Gdk::GC> gc,
63                             line_style ls, line_width lw)
64 {
65         int width = 0;
66         switch (lw) {
67         case Painter::line_thin:
68                 width = 0;
69                 break;
70         case Painter::line_thick:
71                 width = 2;
72                 break;
73         }
74
75         Gdk::LineStyle style = Gdk::LINE_SOLID;
76         switch (ls) {
77         case Painter::line_solid:
78                 style = Gdk::LINE_SOLID;
79                 break;
80         case Painter::line_onoffdash:
81                 style = Gdk::LINE_ON_OFF_DASH;
82                 break;
83         }
84         gc->set_line_attributes(width, style,
85                                 Gdk::CAP_NOT_LAST, Gdk::JOIN_MITER);
86 }
87
88
89 void GPainter::point(int x, int y, LColor_color c)
90 {
91         setForeground(owner_.getGC(), c);
92         owner_.getPixmap()->draw_point(owner_.getGC(), x, y);
93         return *this;
94 }
95
96
97 void GPainter::line(int x1, int y1,
98         int x2, int y2,
99         LColor_color col,
100         line_style ls,
101         line_width lw)
102 {
103         setForeground(owner_.getGC(), col);
104         setLineParam(owner_.getGC(), ls, lw);
105         owner_.getPixmap()->draw_line(owner_.getGC(), x1, y1, x2, y2);
106         return *this;
107 }
108
109
110 void GPainter::lines(int const * xp, int const * yp, int np,
111         LColor_color col,
112         line_style ls,
113         line_width lw)
114 {
115         setForeground(owner_.getGC(), col);
116         setLineParam(owner_.getGC(), ls, lw);
117         std::vector<Gdk::Point> points(np);
118
119         for (int i = 0; i < np; ++i) {
120                 points[i].set_x(xp[i]);
121                 points[i].set_y(yp[i]);
122         }
123         owner_.getPixmap()->draw_lines(owner_.getGC(), points);
124         return *this;
125 }
126
127
128 void GPainter::rectangle(int x, int y, int w, int h,
129         LColor_color col,
130         line_style ls,
131         line_width lw)
132 {
133         setForeground(owner_.getGC(), col);
134         setLineParam(owner_.getGC(), ls, lw);
135         owner_.getPixmap()->draw_rectangle(owner_.getGC(), false, x, y, w, h);
136         return *this;
137 }
138
139
140 void GPainter::fillRectangle(int x, int y, int w, int h,
141         LColor_color col)
142 {
143         setForeground(owner_.getGC(), col);
144         owner_.getPixmap()->draw_rectangle(owner_.getGC(), true, x, y, w, h);
145         return *this;
146 }
147
148
149 void GPainter::fillPolygon(int const * xp, int const * yp,
150         int np, LColor_color col)
151 {
152         setForeground(owner_.getGC(), col);
153         std::vector<Gdk::Point> points(np);
154
155         for (int i = 0; i < np; ++i) {
156                 points[i].set_x(xp[i]);
157                 points[i].set_y(yp[i]);
158         }
159         owner_.getPixmap()->draw_polygon(owner_.getGC(), true, points);
160         return *this;
161 }
162
163
164 void GPainter::arc(int x, int y, unsigned int w, unsigned int h,
165         int a1, int a2, LColor_color col)
166 {
167         setForeground(owner_.getGC(), col);
168         owner_.getPixmap()->draw_arc(owner_.getGC(),
169                                      false, x, y, w, h, a1, a2);
170         return *this;
171 }
172
173
174 void GPainter::image(int x, int y, int w, int h,
175         lyx::graphics::Image const & i)
176 {
177         lyx::graphics::xformsImage const & image =
178                 static_cast<lyx::graphics::xformsImage const &>(i);
179         Pixmap pixmap = GDK_PIXMAP_XID(owner_.getPixmap()->gobj());
180         GC gc = GDK_GC_XGC(owner_.getGC()->gobj());
181         XCopyArea(owner_.getDisplay(), image.getPixmap(), pixmap,
182                   gc, 0, 0, w, h, x, y);
183         return *this;
184 }
185
186
187 void GPainter::text(int x, int y, std::string const & s, LyXFont const & f)
188 {
189         size_t size = s.length() + 1;
190         wchar_t * wcs = (wchar_t *) alloca(size * sizeof(wchar_t));
191         size = mbstowcs(wcs, s.c_str(), size);
192         return text(x, y, wcs, size, f);
193 }
194
195
196 void GPainter::text(int x, int y, char c, LyXFont const & f)
197 {
198         char s[2] = { c, '\0' };
199         return text(x, y, s, 1, f);
200 }
201
202
203 inline XftFont * getXftFont(LyXFont const & f)
204 {
205         return fontLoader.load(f.family(), f.series(),
206                                f.realShape(), f.size());
207 }
208
209
210 namespace font_metrics
211 {
212
213 int width(wchar_t const *s, size_t n, LyXFont const & f);
214
215 }
216
217
218 void GPainter::text(int x, int y, wchar_t const * s, int ls, LyXFont const & f)
219 {
220         XftFont * font = getXftFont(f);
221         XftColor * xftClr = owner_.getColorHandler().
222                 getXftColor(f.realColor());
223 //      getXftColor(f.realColor());
224         XftDraw * draw = owner_.getXftDraw();
225         if (f.realShape() != LyXFont::SMALLCAPS_SHAPE) {
226                 XftDrawString32(draw, xftClr, font, x, y,
227                                 wcsToXftChar32StrFast(s), ls);
228         } else {
229                 LyXFont smallfont(f);
230                 smallfont.decSize().decSize().setShape(LyXFont::UP_SHAPE);
231                 XftFont * fontS = getXftFont(smallfont);
232                 wchar_t c;
233                 int tmpx = x;
234                 for(int i = 0; i < ls; ++i) {
235                         c = lyx::support::uppercase(s[i]);
236                         if(c != s[i]) {
237                                 XftDrawString32(draw, xftClr, fontS, tmpx, y,
238                                                 wcsToXftChar32StrFast(&c), 1);
239                                 tmpx += font_metrics::width(c, smallfont);
240                         } else {
241                                 XftDrawString32(draw, xftClr, font, tmpx, y,
242                                                 wcsToXftChar32StrFast(&c), 1);
243                                 tmpx += font_metrics::width(c, f);
244                         }
245                 }
246         }
247         if (f.underbar() == LyXFont::ON)
248                 underline(f, x, y, font_metrics::width(s, ls, f));
249         return *this;
250 }
251
252
253 void GPainter::text(int x, int y, char const * s, size_t ls, LyXFont const & f)
254 {
255         boost::scoped_array<wchar_t> wcs(new wchar_t[ls + 1]);
256         size_t len;
257         if (fontLoader.isSpecial(f)) {
258                 unsigned char const * us =
259                         reinterpret_cast<unsigned char const *>(s);
260                 len = ls;
261                 std::copy(us, us + ls, wcs.get());
262         } else
263                 len = mbstowcs(wcs.get(), s, ls + 1);
264         return text(x, y, wcs.get(), len, f);
265 }