]>
Commit | Line | Data |
---|---|---|
b3c86150 VS |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: src/dfb/dc.cpp | |
4a624f6e | 3 | // Purpose: wxDFBDCImpl class |
b3c86150 VS |
4 | // Author: Vaclav Slavik |
5 | // Created: 2006-08-07 | |
b3c86150 VS |
6 | // Copyright: (c) 2006 REA Elektronik GmbH |
7 | // Licence: wxWindows licence | |
8 | ///////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | // =========================================================================== | |
11 | // declarations | |
12 | // =========================================================================== | |
13 | ||
14 | // --------------------------------------------------------------------------- | |
15 | // headers | |
16 | // --------------------------------------------------------------------------- | |
17 | ||
18 | // For compilers that support precompilation, includes "wx.h". | |
19 | #include "wx/wxprec.h" | |
20 | ||
21 | #ifdef __BORLANDC__ | |
22 | #pragma hdrstop | |
23 | #endif | |
24 | ||
25 | #ifndef WX_PRECOMP | |
23205be8 | 26 | #include "wx/dcmemory.h" |
b3c86150 VS |
27 | #include "wx/log.h" |
28 | #endif | |
29 | ||
4a624f6e | 30 | #include "wx/dfb/dc.h" |
b3c86150 VS |
31 | #include "wx/dfb/private.h" |
32 | ||
071f5576 VS |
33 | // these values are used to initialize newly created DC |
34 | #define DEFAULT_FONT (*wxNORMAL_FONT) | |
35 | #define DEFAULT_PEN (*wxBLACK_PEN) | |
36 | #define DEFAULT_BRUSH (*wxWHITE_BRUSH) | |
37 | ||
b3c86150 VS |
38 | // =========================================================================== |
39 | // implementation | |
40 | // =========================================================================== | |
41 | ||
42 | //----------------------------------------------------------------------------- | |
4a624f6e | 43 | // wxDFBDCImpl |
b3c86150 VS |
44 | //----------------------------------------------------------------------------- |
45 | ||
4a624f6e | 46 | IMPLEMENT_ABSTRACT_CLASS(wxDFBDCImpl, wxDCImpl) |
b3c86150 | 47 | |
4a624f6e | 48 | void wxDFBDCImpl::DFBInit(const wxIDirectFBSurfacePtr& surface) |
b3c86150 | 49 | { |
4a624f6e | 50 | m_surface = surface; |
b3c86150 | 51 | |
a5001e93 | 52 | wxCHECK_RET( surface != NULL, "invalid surface" ); |
b3c86150 | 53 | |
b3c86150 VS |
54 | m_mm_to_pix_x = (double)wxGetDisplaySize().GetWidth() / |
55 | (double)wxGetDisplaySizeMM().GetWidth(); | |
56 | m_mm_to_pix_y = (double)wxGetDisplaySize().GetHeight() / | |
57 | (double)wxGetDisplaySizeMM().GetHeight(); | |
58 | ||
071f5576 VS |
59 | SetFont(DEFAULT_FONT); |
60 | SetPen(DEFAULT_PEN); | |
61 | SetBrush(DEFAULT_BRUSH); | |
b3c86150 VS |
62 | } |
63 | ||
64 | ||
65 | // --------------------------------------------------------------------------- | |
66 | // clipping | |
67 | // --------------------------------------------------------------------------- | |
68 | ||
4a624f6e | 69 | void wxDFBDCImpl::DoSetClippingRegion(wxCoord cx, wxCoord cy, wxCoord cw, wxCoord ch) |
b3c86150 | 70 | { |
4a624f6e | 71 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 72 | |
564b429f VS |
73 | wxSize size(GetSize()); |
74 | ||
f87d0500 | 75 | wxASSERT_MSG( !m_clipping, |
a5001e93 | 76 | "narrowing clipping region not implemented yet" ); |
f87d0500 | 77 | |
564b429f VS |
78 | // NB: We intersect the clipping rectangle with surface's area here because |
79 | // DirectFB will return an error if you try to set clipping rectangle | |
80 | // that is partially outside of the surface. | |
b3c86150 | 81 | DFBRegion r; |
564b429f VS |
82 | r.x1 = wxMax(0, XLOG2DEV(cx)); |
83 | r.y1 = wxMax(0, YLOG2DEV(cy)); | |
84 | r.x2 = wxMin(r.x1 + XLOG2DEVREL(cw), size.x) - 1; | |
85 | r.y2 = wxMin(r.y1 + YLOG2DEVREL(ch), size.y) - 1; | |
b3c86150 | 86 | |
52c8d32a | 87 | if ( !m_surface->SetClip(&r) ) |
b3c86150 VS |
88 | return; |
89 | ||
90 | m_clipX1 = cx; | |
91 | m_clipY1 = cy; | |
92 | m_clipX2 = cx + cw - 1; | |
93 | m_clipY2 = cy + ch -1; | |
94 | m_clipping = true; | |
95 | } | |
96 | ||
fdaad94e | 97 | void wxDFBDCImpl::DoSetDeviceClippingRegion(const wxRegion& region) |
b3c86150 | 98 | { |
f87d0500 | 99 | // NB: this can be done because wxDFB only supports rectangular regions |
fdaad94e VZ |
100 | wxRect rect = region.AsRect(); |
101 | ||
102 | // our parameter is in physical coordinates while DoSetClippingRegion() | |
103 | // takes logical ones | |
104 | rect.x = XDEV2LOG(rect.x); | |
105 | rect.y = YDEV2LOG(rect.y); | |
106 | rect.width = XDEV2LOG(rect.width); | |
107 | rect.height = YDEV2LOG(rect.height); | |
108 | ||
6a8dba37 | 109 | DoSetClippingRegion(rect.x, rect.y, rect.width, rect.height); |
b3c86150 VS |
110 | } |
111 | ||
4a624f6e | 112 | void wxDFBDCImpl::DestroyClippingRegion() |
b3c86150 | 113 | { |
4a624f6e | 114 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 115 | |
52c8d32a | 116 | m_surface->SetClip(NULL); |
b3c86150 VS |
117 | |
118 | ResetClipping(); | |
119 | } | |
120 | ||
121 | // --------------------------------------------------------------------------- | |
122 | // query capabilities | |
123 | // --------------------------------------------------------------------------- | |
124 | ||
4a624f6e | 125 | int wxDFBDCImpl::GetDepth() const |
b3c86150 | 126 | { |
a5b31f4e | 127 | return m_surface->GetDepth(); |
b3c86150 VS |
128 | } |
129 | ||
130 | // --------------------------------------------------------------------------- | |
131 | // drawing | |
132 | // --------------------------------------------------------------------------- | |
133 | ||
4a624f6e | 134 | void wxDFBDCImpl::Clear() |
b3c86150 | 135 | { |
4a624f6e | 136 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 VS |
137 | |
138 | if ( m_backgroundBrush.GetStyle() == wxTRANSPARENT ) | |
139 | return; | |
140 | ||
141 | wxColour clr = m_backgroundBrush.GetColour(); | |
52c8d32a | 142 | m_surface->Clear(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha()); |
20671963 VS |
143 | |
144 | wxSize size(GetSize()); | |
145 | CalcBoundingBox(XDEV2LOG(0), YDEV2LOG(0)); | |
146 | CalcBoundingBox(XDEV2LOG(size.x), YDEV2LOG(size.y)); | |
b3c86150 VS |
147 | } |
148 | ||
149 | extern bool wxDoFloodFill(wxDC *dc, wxCoord x, wxCoord y, | |
89efaf2b | 150 | const wxColour & col, wxFloodFillStyle style); |
b3c86150 | 151 | |
4a624f6e | 152 | bool wxDFBDCImpl::DoFloodFill(wxCoord x, wxCoord y, |
89efaf2b | 153 | const wxColour& col, wxFloodFillStyle style) |
b3c86150 | 154 | { |
4a624f6e | 155 | return wxDoFloodFill(GetOwner(), x, y, col, style); |
b3c86150 VS |
156 | } |
157 | ||
4a624f6e | 158 | bool wxDFBDCImpl::DoGetPixel(wxCoord x, wxCoord y, wxColour *col) const |
b3c86150 | 159 | { |
4a624f6e | 160 | wxCHECK_MSG( col, false, "NULL colour parameter in wxDFBDCImpl::GetPixel"); |
b3c86150 | 161 | |
a5001e93 | 162 | wxFAIL_MSG( "GetPixel not implemented" ); |
4a624f6e VZ |
163 | |
164 | wxUnusedVar(x); | |
165 | wxUnusedVar(y); | |
166 | ||
b3c86150 VS |
167 | return false; |
168 | } | |
169 | ||
4a624f6e | 170 | void wxDFBDCImpl::DoCrossHair(wxCoord x, wxCoord y) |
b3c86150 | 171 | { |
4a624f6e | 172 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 173 | |
a5001e93 | 174 | wxFAIL_MSG( "CrossHair not implemented" ); |
4a624f6e VZ |
175 | |
176 | wxUnusedVar(x); | |
177 | wxUnusedVar(y); | |
b3c86150 VS |
178 | } |
179 | ||
4a624f6e | 180 | void wxDFBDCImpl::DoDrawLine(wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2) |
b3c86150 | 181 | { |
4a624f6e | 182 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 VS |
183 | |
184 | if ( m_pen.GetStyle() == wxTRANSPARENT ) | |
185 | return; | |
186 | ||
ea280776 | 187 | wxCoord xx1 = XLOG2DEV(x1); |
59eb2aca | 188 | wxCoord yy1 = YLOG2DEV(y1); |
ea280776 | 189 | wxCoord xx2 = XLOG2DEV(x2); |
59eb2aca | 190 | wxCoord yy2 = YLOG2DEV(y2); |
ea280776 VS |
191 | |
192 | // FIXME: DrawLine() shouldn't draw the last pixel, but DFB's DrawLine() | |
193 | // does draw it. We should undo any change to the last pixel by | |
194 | // using GetPixel() and PutPixel(), but until they are implemented, | |
195 | // handle at least the special case of vertical and horizontal | |
196 | // lines correctly: | |
197 | if ( xx1 == xx2 ) | |
198 | { | |
199 | if ( yy1 < yy2 ) | |
200 | yy2--; | |
201 | else if ( yy1 > yy2 ) | |
202 | yy2++; | |
203 | } | |
204 | if ( yy1 == yy2 ) | |
205 | { | |
206 | if ( xx1 < xx2 ) | |
207 | xx2--; | |
208 | else if ( xx1 > xx2 ) | |
209 | xx2++; | |
210 | } | |
211 | ||
212 | m_surface->DrawLine(xx1, yy1, xx2, yy2); | |
b3c86150 VS |
213 | |
214 | CalcBoundingBox(x1, y1); | |
215 | CalcBoundingBox(x2, y2); | |
216 | } | |
217 | ||
218 | // Draws an arc of a circle, centred on (xc, yc), with starting point (x1, y1) | |
219 | // and ending at (x2, y2) | |
4a624f6e VZ |
220 | void wxDFBDCImpl::DoDrawArc(wxCoord WXUNUSED(x1), wxCoord WXUNUSED(y1), |
221 | wxCoord WXUNUSED(x2), wxCoord WXUNUSED(y2), | |
222 | wxCoord WXUNUSED(xc), wxCoord WXUNUSED(yc)) | |
b3c86150 | 223 | { |
4a624f6e | 224 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 225 | |
a5001e93 | 226 | wxFAIL_MSG( "DrawArc not implemented" ); |
b3c86150 VS |
227 | } |
228 | ||
4a624f6e | 229 | void wxDFBDCImpl::DoDrawPoint(wxCoord x, wxCoord y) |
b3c86150 | 230 | { |
4a624f6e | 231 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 232 | |
3af4da32 VS |
233 | // NB: DirectFB API doesn't provide a function for drawing points, so |
234 | // implement it as 1px long line. This is inefficient, but then, so is | |
235 | // using DrawPoint() for drawing more than a few points. | |
236 | DoDrawLine(x, y, x, y); | |
237 | ||
238 | // FIXME_DFB: implement special cases for common formats (RGB24,RGBA/RGB32) | |
b3c86150 VS |
239 | } |
240 | ||
4787c92d | 241 | void wxDFBDCImpl::DoDrawPolygon(int WXUNUSED(n), const wxPoint WXUNUSED(points)[], |
4a624f6e | 242 | wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset), |
89efaf2b | 243 | wxPolygonFillMode WXUNUSED(fillStyle)) |
b3c86150 | 244 | { |
4a624f6e | 245 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 246 | |
a5001e93 | 247 | wxFAIL_MSG( "DrawPolygon not implemented" ); |
b3c86150 VS |
248 | } |
249 | ||
4787c92d | 250 | void wxDFBDCImpl::DoDrawLines(int WXUNUSED(n), const wxPoint WXUNUSED(points)[], |
4a624f6e | 251 | wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset)) |
b3c86150 | 252 | { |
4a624f6e | 253 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 VS |
254 | |
255 | // TODO: impl. using DirectDB's DrawLines | |
a5001e93 | 256 | wxFAIL_MSG( "DrawLines not implemented" ); |
b3c86150 VS |
257 | } |
258 | ||
4a624f6e | 259 | void wxDFBDCImpl::DoDrawRectangle(wxCoord x, wxCoord y, wxCoord width, wxCoord height) |
b3c86150 | 260 | { |
4a624f6e | 261 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 VS |
262 | |
263 | wxCoord xx = XLOG2DEV(x); | |
264 | wxCoord yy = YLOG2DEV(y); | |
265 | wxCoord ww = m_signX * XLOG2DEVREL(width); | |
266 | wxCoord hh = m_signY * YLOG2DEVREL(height); | |
267 | ||
268 | if ( ww == 0 || hh == 0 ) return; | |
269 | ||
270 | if ( ww < 0 ) | |
271 | { | |
272 | ww = -ww; | |
273 | xx = xx - ww; | |
274 | } | |
275 | if ( hh < 0 ) | |
276 | { | |
277 | hh = -hh; | |
278 | yy = yy - hh; | |
279 | } | |
280 | ||
281 | if ( m_brush.GetStyle() != wxTRANSPARENT ) | |
282 | { | |
283 | SelectColour(m_brush.GetColour()); | |
52c8d32a | 284 | m_surface->FillRectangle(xx, yy, ww, hh); |
3dcc2231 VS |
285 | // restore pen's colour, because other drawing functions expect the |
286 | // colour to be set to the pen: | |
b3c86150 VS |
287 | SelectColour(m_pen.GetColour()); |
288 | } | |
289 | ||
290 | if ( m_pen.GetStyle() != wxTRANSPARENT ) | |
291 | { | |
52c8d32a | 292 | m_surface->DrawRectangle(xx, yy, ww, hh); |
b3c86150 VS |
293 | } |
294 | ||
295 | CalcBoundingBox(x, y); | |
296 | CalcBoundingBox(x + width, y + height); | |
297 | } | |
298 | ||
4a624f6e VZ |
299 | void wxDFBDCImpl::DoDrawRoundedRectangle(wxCoord WXUNUSED(x), |
300 | wxCoord WXUNUSED(y), | |
301 | wxCoord WXUNUSED(width), | |
302 | wxCoord WXUNUSED(height), | |
303 | double WXUNUSED(radius)) | |
b3c86150 | 304 | { |
4a624f6e | 305 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 306 | |
a5001e93 | 307 | wxFAIL_MSG( "DrawRoundedRectangle not implemented" ); |
b3c86150 VS |
308 | } |
309 | ||
4a624f6e VZ |
310 | void wxDFBDCImpl::DoDrawEllipse(wxCoord WXUNUSED(x), |
311 | wxCoord WXUNUSED(y), | |
312 | wxCoord WXUNUSED(width), | |
313 | wxCoord WXUNUSED(height)) | |
b3c86150 | 314 | { |
4a624f6e | 315 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 316 | |
a5001e93 | 317 | wxFAIL_MSG( "DrawElipse not implemented" ); |
b3c86150 VS |
318 | } |
319 | ||
4a624f6e VZ |
320 | void wxDFBDCImpl::DoDrawEllipticArc(wxCoord WXUNUSED(x), |
321 | wxCoord WXUNUSED(y), | |
322 | wxCoord WXUNUSED(w), | |
323 | wxCoord WXUNUSED(h), | |
324 | double WXUNUSED(sa), | |
325 | double WXUNUSED(ea)) | |
b3c86150 | 326 | { |
4a624f6e | 327 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 328 | |
a5001e93 | 329 | wxFAIL_MSG( "DrawElipticArc not implemented" ); |
b3c86150 VS |
330 | } |
331 | ||
4a624f6e | 332 | void wxDFBDCImpl::DoDrawText(const wxString& text, wxCoord x, wxCoord y) |
b3c86150 | 333 | { |
4a624f6e | 334 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 VS |
335 | |
336 | wxCoord xx = XLOG2DEV(x); | |
5034c9db | 337 | wxCoord yy = YLOG2DEV(y); |
b3c86150 VS |
338 | |
339 | // update the bounding box | |
340 | wxCoord w, h; | |
341 | CalcBoundingBox(x, y); | |
4a624f6e | 342 | DoGetTextExtent(text, &w, &h); |
b3c86150 VS |
343 | CalcBoundingBox(x + w, y + h); |
344 | ||
345 | // if background mode is solid, DrawText must paint text's background: | |
346 | if ( m_backgroundMode == wxSOLID ) | |
347 | { | |
a1b806b9 | 348 | wxCHECK_RET( m_textBackgroundColour.IsOk(), |
3dcc2231 | 349 | wxT("invalid background color") ); |
b3c86150 | 350 | |
3dcc2231 | 351 | SelectColour(m_textBackgroundColour); |
52c8d32a | 352 | m_surface->FillRectangle(xx, yy, XLOG2DEVREL(w), YLOG2DEVREL(h)); |
b3c86150 VS |
353 | } |
354 | ||
355 | // finally draw the text itself: | |
a1b806b9 | 356 | wxCHECK_RET( m_textForegroundColour.IsOk(), |
3dcc2231 VS |
357 | wxT("invalid foreground color") ); |
358 | SelectColour(m_textForegroundColour); | |
26f03c44 | 359 | m_surface->DrawString(text.utf8_str(), -1, xx, yy, DSTF_LEFT | DSTF_TOP); |
3dcc2231 VS |
360 | |
361 | // restore pen's colour, because other drawing functions expect the colour | |
362 | // to be set to the pen: | |
363 | SelectColour(m_pen.GetColour()); | |
b3c86150 VS |
364 | } |
365 | ||
4a624f6e VZ |
366 | void wxDFBDCImpl::DoDrawRotatedText(const wxString& WXUNUSED(text), |
367 | wxCoord WXUNUSED(x), wxCoord WXUNUSED(y), | |
368 | double WXUNUSED(angle)) | |
b3c86150 | 369 | { |
4a624f6e | 370 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 371 | |
a5001e93 | 372 | wxFAIL_MSG( "DrawRotatedText not implemented" ); |
b3c86150 VS |
373 | } |
374 | ||
375 | // --------------------------------------------------------------------------- | |
376 | // set GDI objects | |
377 | // --------------------------------------------------------------------------- | |
378 | ||
4a624f6e | 379 | void wxDFBDCImpl::SetPen(const wxPen& pen) |
b3c86150 | 380 | { |
a1b806b9 | 381 | m_pen = pen.IsOk() ? pen : DEFAULT_PEN; |
b3c86150 VS |
382 | |
383 | SelectColour(m_pen.GetColour()); | |
384 | } | |
385 | ||
4a624f6e | 386 | void wxDFBDCImpl::SetBrush(const wxBrush& brush) |
b3c86150 | 387 | { |
a1b806b9 | 388 | m_brush = brush.IsOk() ? brush : DEFAULT_BRUSH; |
b3c86150 VS |
389 | } |
390 | ||
4a624f6e | 391 | void wxDFBDCImpl::SelectColour(const wxColour& clr) |
b3c86150 | 392 | { |
52c8d32a | 393 | m_surface->SetColor(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha()); |
b3c86150 VS |
394 | #warning "use SetColorIndex?" |
395 | } | |
396 | ||
397 | #if wxUSE_PALETTE | |
4a624f6e | 398 | void wxDFBDCImpl::SetPalette(const wxPalette& WXUNUSED(palette)) |
b3c86150 | 399 | { |
4a624f6e | 400 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 401 | |
a5001e93 | 402 | wxFAIL_MSG( "SetPalette not implemented" ); |
b3c86150 VS |
403 | } |
404 | #endif // wxUSE_PALETTE | |
405 | ||
4a624f6e | 406 | void wxDFBDCImpl::SetFont(const wxFont& font) |
b3c86150 | 407 | { |
4a624f6e | 408 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 409 | |
a1b806b9 | 410 | wxFont f(font.IsOk() ? font : DEFAULT_FONT); |
b3c86150 | 411 | |
d7ae4a62 | 412 | wxFont oldfont(m_font); |
b3c86150 | 413 | |
071f5576 | 414 | m_font = f; |
d7ae4a62 VS |
415 | |
416 | if ( !m_surface->SetFont(GetCurrentFont()) ) | |
417 | { | |
418 | m_font = oldfont; | |
419 | return; | |
420 | } | |
421 | } | |
422 | ||
4a624f6e | 423 | wxIDirectFBFontPtr wxDFBDCImpl::GetCurrentFont() const |
d7ae4a62 VS |
424 | { |
425 | bool aa = (GetDepth() > 8); | |
426 | return m_font.GetDirectFBFont(aa); | |
b3c86150 VS |
427 | } |
428 | ||
4a624f6e | 429 | void wxDFBDCImpl::SetBackground(const wxBrush& brush) |
b3c86150 | 430 | { |
4a624f6e | 431 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 432 | |
a1b806b9 | 433 | if (!brush.IsOk()) return; |
b3c86150 VS |
434 | |
435 | m_backgroundBrush = brush; | |
436 | } | |
437 | ||
4a624f6e | 438 | void wxDFBDCImpl::SetBackgroundMode(int mode) |
b3c86150 VS |
439 | { |
440 | m_backgroundMode = mode; | |
441 | } | |
442 | ||
89efaf2b | 443 | void wxDFBDCImpl::SetLogicalFunction(wxRasterOperationMode function) |
b3c86150 | 444 | { |
4a624f6e | 445 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 446 | |
5942996c VS |
447 | // NB: we could also support XOR, but for blitting only (via DSBLIT_XOR); |
448 | // and possibly others via SetSrc/DstBlendFunction() | |
449 | wxASSERT_MSG( function == wxCOPY, | |
a5001e93 | 450 | "only wxCOPY logical function supported" ); |
b3c86150 VS |
451 | |
452 | m_logicalFunction = function; | |
453 | } | |
454 | ||
4a624f6e | 455 | bool wxDFBDCImpl::StartDoc(const wxString& WXUNUSED(message)) |
b3c86150 VS |
456 | { |
457 | // We might be previewing, so return true to let it continue. | |
458 | return true; | |
459 | } | |
460 | ||
4a624f6e | 461 | void wxDFBDCImpl::EndDoc() |
b3c86150 VS |
462 | { |
463 | } | |
464 | ||
4a624f6e | 465 | void wxDFBDCImpl::StartPage() |
b3c86150 VS |
466 | { |
467 | } | |
468 | ||
4a624f6e | 469 | void wxDFBDCImpl::EndPage() |
b3c86150 VS |
470 | { |
471 | } | |
472 | ||
473 | // --------------------------------------------------------------------------- | |
474 | // text metrics | |
475 | // --------------------------------------------------------------------------- | |
476 | ||
4a624f6e | 477 | wxCoord wxDFBDCImpl::GetCharHeight() const |
b3c86150 | 478 | { |
4a624f6e | 479 | wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") ); |
a1b806b9 | 480 | wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") ); |
b3c86150 VS |
481 | |
482 | int h = -1; | |
d7ae4a62 | 483 | GetCurrentFont()->GetHeight(&h); |
b3c86150 VS |
484 | return YDEV2LOGREL(h); |
485 | } | |
486 | ||
4a624f6e | 487 | wxCoord wxDFBDCImpl::GetCharWidth() const |
b3c86150 | 488 | { |
4a624f6e | 489 | wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") ); |
a1b806b9 | 490 | wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") ); |
b3c86150 VS |
491 | |
492 | int w = -1; | |
d7ae4a62 | 493 | GetCurrentFont()->GetStringWidth("H", 1, &w); |
b3c86150 VS |
494 | // VS: YDEV is corrent, it should *not* be XDEV, because font's are only |
495 | // scaled according to m_scaleY | |
496 | return YDEV2LOGREL(w); | |
497 | } | |
498 | ||
4a624f6e | 499 | void wxDFBDCImpl::DoGetTextExtent(const wxString& string, wxCoord *x, wxCoord *y, |
b3c86150 | 500 | wxCoord *descent, wxCoord *externalLeading, |
c94f845b | 501 | const wxFont *theFont) const |
b3c86150 | 502 | { |
4a624f6e | 503 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
a1b806b9 DS |
504 | wxCHECK_RET( m_font.IsOk(), wxT("no font selected") ); |
505 | wxCHECK_RET( !theFont || theFont->IsOk(), wxT("invalid font") ); | |
b3c86150 VS |
506 | |
507 | wxFont oldFont; | |
508 | if ( theFont != NULL ) | |
509 | { | |
510 | oldFont = m_font; | |
4a624f6e | 511 | wxConstCast(this, wxDFBDCImpl)->SetFont(*theFont); |
b3c86150 VS |
512 | } |
513 | ||
514 | wxCoord xx = 0, yy = 0; | |
515 | DFBRectangle rect; | |
d7ae4a62 | 516 | wxIDirectFBFontPtr f = GetCurrentFont(); |
b3c86150 | 517 | |
26f03c44 | 518 | if ( f->GetStringExtents(string.utf8_str(), -1, &rect, NULL) ) |
b3c86150 VS |
519 | { |
520 | // VS: YDEV is corrent, it should *not* be XDEV, because font's are | |
521 | // only scaled according to m_scaleY | |
522 | xx = YDEV2LOGREL(rect.w); | |
523 | yy = YDEV2LOGREL(rect.h); | |
524 | ||
525 | if ( descent ) | |
526 | { | |
527 | int d; | |
52c8d32a | 528 | if ( f->GetDescender(&d) ) |
b3c86150 VS |
529 | *descent = YDEV2LOGREL(-d); |
530 | else | |
531 | *descent = 0; | |
532 | } | |
533 | } | |
534 | ||
535 | if ( x ) *x = xx; | |
536 | if ( y ) *y = yy; | |
537 | if ( externalLeading ) *externalLeading = 0; | |
538 | ||
539 | if ( theFont != NULL ) | |
4a624f6e | 540 | wxConstCast(this, wxDFBDCImpl)->SetFont(oldFont); |
b3c86150 VS |
541 | } |
542 | ||
543 | ||
544 | ||
545 | // --------------------------------------------------------------------------- | |
546 | // mapping modes | |
547 | // --------------------------------------------------------------------------- | |
548 | ||
04ab8b6d RR |
549 | // FIXME_DFB: scaling affects pixel size of font, pens, brushes, which |
550 | // is not currently implemented here; probably makes sense to | |
551 | // switch to Cairo instead of implementing everything for DFB | |
4a624f6e VZ |
552 | |
553 | void wxDFBDCImpl::DoGetSize(int *w, int *h) const | |
b3c86150 | 554 | { |
4a624f6e | 555 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
b3c86150 | 556 | |
52c8d32a | 557 | m_surface->GetSize(w, h); |
b3c86150 VS |
558 | } |
559 | ||
4a624f6e | 560 | void wxDFBDCImpl::DoGetSizeMM(int *width, int *height) const |
b3c86150 VS |
561 | { |
562 | #warning "move this to common code?" | |
563 | int w = 0; | |
564 | int h = 0; | |
565 | GetSize(&w, &h); | |
566 | if ( width ) *width = int(double(w) / (m_userScaleX*m_mm_to_pix_x)); | |
567 | if ( height ) *height = int(double(h) / (m_userScaleY*m_mm_to_pix_y)); | |
568 | } | |
569 | ||
4a624f6e | 570 | wxSize wxDFBDCImpl::GetPPI() const |
b3c86150 VS |
571 | { |
572 | #warning "move this to common code?" | |
573 | return wxSize(int(double(m_mm_to_pix_x) * inches2mm), | |
574 | int(double(m_mm_to_pix_y) * inches2mm)); | |
575 | } | |
576 | ||
577 | ||
578 | // --------------------------------------------------------------------------- | |
579 | // Blitting | |
580 | // --------------------------------------------------------------------------- | |
581 | ||
4a624f6e VZ |
582 | bool wxDFBDCImpl::DoBlit(wxCoord xdest, wxCoord ydest, |
583 | wxCoord width, wxCoord height, | |
584 | wxDC *source, wxCoord xsrc, wxCoord ysrc, | |
89efaf2b | 585 | wxRasterOperationMode rop, bool useMask, |
4a624f6e | 586 | wxCoord xsrcMask, wxCoord ysrcMask) |
b3c86150 | 587 | { |
4a624f6e | 588 | wxCHECK_MSG( IsOk(), false, "invalid dc" ); |
a5001e93 | 589 | wxCHECK_MSG( source, false, "invalid source dc" ); |
5942996c VS |
590 | |
591 | // NB: we could also support XOR here (via DSBLIT_XOR) | |
592 | // and possibly others via SetSrc/DstBlendFunction() | |
a5001e93 | 593 | wxCHECK_MSG( rop == wxCOPY, false, "only wxCOPY function supported" ); |
b3c86150 VS |
594 | |
595 | // transform the source DC coords to the device ones | |
596 | xsrc = source->LogicalToDeviceX(xsrc); | |
597 | ysrc = source->LogicalToDeviceY(ysrc); | |
598 | ||
5942996c VS |
599 | // FIXME_DFB: use the mask origin when drawing transparently |
600 | wxASSERT_MSG( xsrcMask == -1 && ysrcMask == -1, | |
a5001e93 | 601 | "non-default source mask offset not implemented" ); |
5942996c | 602 | #if 0 |
b3c86150 VS |
603 | if (xsrcMask == -1 && ysrcMask == -1) |
604 | { | |
605 | xsrcMask = xsrc; ysrcMask = ysrc; | |
606 | } | |
607 | else | |
608 | { | |
609 | xsrcMask = source->LogicalToDeviceX(xsrcMask); | |
610 | ysrcMask = source->LogicalToDeviceY(ysrcMask); | |
611 | } | |
5942996c | 612 | #endif |
b3c86150 | 613 | |
5942996c VS |
614 | wxMemoryDC *sourceAsMemDC = wxDynamicCast(source, wxMemoryDC); |
615 | if ( sourceAsMemDC ) | |
b3c86150 | 616 | { |
4a624f6e | 617 | DoDrawSubBitmap(sourceAsMemDC->GetSelectedBitmap(), |
5942996c VS |
618 | xsrc, ysrc, |
619 | width, height, | |
620 | xdest, ydest, | |
621 | rop, | |
622 | useMask); | |
b3c86150 VS |
623 | } |
624 | else | |
625 | { | |
4a624f6e VZ |
626 | return DoBlitFromSurface |
627 | ( | |
628 | static_cast<wxDFBDCImpl *>(source->GetImpl()) | |
629 | ->GetDirectFBSurface(), | |
630 | xsrc, ysrc, | |
631 | width, height, | |
632 | xdest, ydest | |
633 | ); | |
b3c86150 VS |
634 | } |
635 | ||
636 | return true; | |
b3c86150 VS |
637 | } |
638 | ||
4a624f6e | 639 | void wxDFBDCImpl::DoDrawBitmap(const wxBitmap &bmp, wxCoord x, wxCoord y, bool useMask) |
b3c86150 | 640 | { |
4a624f6e | 641 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
a1b806b9 | 642 | wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") ); |
b3c86150 | 643 | |
5942996c VS |
644 | DoDrawSubBitmap(bmp, |
645 | 0, 0, bmp.GetWidth(), bmp.GetHeight(), | |
646 | x, y, | |
647 | m_logicalFunction, useMask); | |
b3c86150 VS |
648 | } |
649 | ||
4a624f6e | 650 | void wxDFBDCImpl::DoDrawIcon(const wxIcon& icon, wxCoord x, wxCoord y) |
b3c86150 VS |
651 | { |
652 | // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why | |
653 | DoDrawBitmap((const wxBitmap&)icon, x, y, true); | |
654 | } | |
655 | ||
4a624f6e | 656 | void wxDFBDCImpl::DoDrawSubBitmap(const wxBitmap &bmp, |
b3c86150 VS |
657 | wxCoord x, wxCoord y, wxCoord w, wxCoord h, |
658 | wxCoord destx, wxCoord desty, int rop, bool useMask) | |
659 | { | |
4a624f6e | 660 | wxCHECK_RET( IsOk(), wxT("invalid dc") ); |
a1b806b9 | 661 | wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") ); |
b3c86150 | 662 | |
5942996c VS |
663 | // NB: we could also support XOR here (via DSBLIT_XOR) |
664 | // and possibly others via SetSrc/DstBlendFunction() | |
a5001e93 | 665 | wxCHECK_RET( rop == wxCOPY, "only wxCOPY function supported" ); |
b3c86150 VS |
666 | |
667 | if ( bmp.GetDepth() == 1 ) | |
668 | { | |
669 | // Mono bitmaps are handled in special way -- all 1s are drawn in | |
670 | // foreground colours, all 0s in background colour. | |
a5001e93 | 671 | wxFAIL_MSG( "drawing mono bitmaps not implemented" ); |
5942996c | 672 | return; |
b3c86150 VS |
673 | } |
674 | ||
675 | if ( useMask && bmp.GetMask() ) | |
676 | { | |
0e1f8ea4 | 677 | // FIXME_DFB: Could use blitting modes for this; also see |
5942996c | 678 | // DFB's SetSrcBlendFunction() and SetSrcColorKey() |
a5001e93 | 679 | wxFAIL_MSG( "drawing bitmaps with masks not implemented" ); |
5942996c VS |
680 | return; |
681 | } | |
b3c86150 | 682 | |
5942996c VS |
683 | DoBlitFromSurface(bmp.GetDirectFBSurface(), |
684 | x, y, | |
685 | w, h, | |
686 | destx, desty); | |
687 | } | |
b3c86150 | 688 | |
4a624f6e VZ |
689 | bool wxDFBDCImpl::DoBlitFromSurface(const wxIDirectFBSurfacePtr& src, |
690 | wxCoord srcx, wxCoord srcy, | |
691 | wxCoord w, wxCoord h, | |
692 | wxCoord dstx, wxCoord dsty) | |
5942996c | 693 | { |
21e3246c VS |
694 | // don't do anything if the source rectangle is outside of source surface, |
695 | // DirectFB would assert in that case: | |
696 | wxSize srcsize; | |
697 | src->GetSize(&srcsize.x, &srcsize.y); | |
698 | if ( !wxRect(srcx, srcy, w, h).Intersects(wxRect(srcsize)) ) | |
699 | { | |
a5001e93 | 700 | wxLogDebug("Blitting from area outside of the source surface, caller code needs fixing."); |
21e3246c VS |
701 | return false; |
702 | } | |
703 | ||
5942996c VS |
704 | CalcBoundingBox(dstx, dsty); |
705 | CalcBoundingBox(dstx + w, dsty + h); | |
b3c86150 | 706 | |
5942996c VS |
707 | DFBRectangle srcRect = { srcx, srcy, w, h }; |
708 | DFBRectangle dstRect = { XLOG2DEV(dstx), YLOG2DEV(dsty), | |
709 | XLOG2DEVREL(w), YLOG2DEVREL(h) }; | |
b3c86150 | 710 | |
5942996c | 711 | wxIDirectFBSurfacePtr dst(m_surface); |
b3c86150 | 712 | |
5942996c | 713 | // FIXME: this will have to be different in useMask case, see above |
e2badebb VS |
714 | DFBSurfaceBlittingFlags blitFlag = (src->GetPixelFormat() == DSPF_ARGB) |
715 | ? DSBLIT_BLEND_ALPHACHANNEL | |
716 | : DSBLIT_NOFX; | |
717 | if ( !dst->SetBlittingFlags(blitFlag) ) | |
5942996c | 718 | return false; |
b3c86150 | 719 | |
5942996c VS |
720 | if ( srcRect.w != dstRect.w || srcRect.h != dstRect.h ) |
721 | { | |
722 | // the bitmap is drawn stretched: | |
723 | dst->StretchBlit(src, &srcRect, &dstRect); | |
b3c86150 | 724 | } |
b3c86150 VS |
725 | else |
726 | { | |
5942996c VS |
727 | // no stretching, size is preserved: |
728 | dst->Blit(src, &srcRect, dstRect.x, dstRect.y); | |
b3c86150 | 729 | } |
5942996c VS |
730 | |
731 | return true; | |
b3c86150 | 732 | } |