]> git.saurik.com Git - wxWidgets.git/blob - src/dfb/dc.cpp
Make public headers compatible with Objective-C++ with ARC.
[wxWidgets.git] / src / dfb / dc.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/dfb/dc.cpp
3 // Purpose: wxDFBDCImpl class
4 // Author: Vaclav Slavik
5 // Created: 2006-08-07
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
26 #include "wx/dcmemory.h"
27 #include "wx/log.h"
28 #endif
29
30 #include "wx/dfb/dc.h"
31 #include "wx/dfb/private.h"
32
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
38 // ===========================================================================
39 // implementation
40 // ===========================================================================
41
42 //-----------------------------------------------------------------------------
43 // wxDFBDCImpl
44 //-----------------------------------------------------------------------------
45
46 IMPLEMENT_ABSTRACT_CLASS(wxDFBDCImpl, wxDCImpl)
47
48 void wxDFBDCImpl::DFBInit(const wxIDirectFBSurfacePtr& surface)
49 {
50 m_surface = surface;
51
52 wxCHECK_RET( surface != NULL, "invalid surface" );
53
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
59 SetFont(DEFAULT_FONT);
60 SetPen(DEFAULT_PEN);
61 SetBrush(DEFAULT_BRUSH);
62 }
63
64
65 // ---------------------------------------------------------------------------
66 // clipping
67 // ---------------------------------------------------------------------------
68
69 void wxDFBDCImpl::DoSetClippingRegion(wxCoord cx, wxCoord cy, wxCoord cw, wxCoord ch)
70 {
71 wxCHECK_RET( IsOk(), wxT("invalid dc") );
72
73 wxSize size(GetSize());
74
75 wxASSERT_MSG( !m_clipping,
76 "narrowing clipping region not implemented yet" );
77
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.
81 DFBRegion r;
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;
86
87 if ( !m_surface->SetClip(&r) )
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
97 void wxDFBDCImpl::DoSetDeviceClippingRegion(const wxRegion& region)
98 {
99 // NB: this can be done because wxDFB only supports rectangular regions
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
109 DoSetClippingRegion(rect.x, rect.y, rect.width, rect.height);
110 }
111
112 void wxDFBDCImpl::DestroyClippingRegion()
113 {
114 wxCHECK_RET( IsOk(), wxT("invalid dc") );
115
116 m_surface->SetClip(NULL);
117
118 ResetClipping();
119 }
120
121 // ---------------------------------------------------------------------------
122 // query capabilities
123 // ---------------------------------------------------------------------------
124
125 int wxDFBDCImpl::GetDepth() const
126 {
127 return m_surface->GetDepth();
128 }
129
130 // ---------------------------------------------------------------------------
131 // drawing
132 // ---------------------------------------------------------------------------
133
134 void wxDFBDCImpl::Clear()
135 {
136 wxCHECK_RET( IsOk(), wxT("invalid dc") );
137
138 if ( m_backgroundBrush.GetStyle() == wxTRANSPARENT )
139 return;
140
141 wxColour clr = m_backgroundBrush.GetColour();
142 m_surface->Clear(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha());
143
144 wxSize size(GetSize());
145 CalcBoundingBox(XDEV2LOG(0), YDEV2LOG(0));
146 CalcBoundingBox(XDEV2LOG(size.x), YDEV2LOG(size.y));
147 }
148
149 extern bool wxDoFloodFill(wxDC *dc, wxCoord x, wxCoord y,
150 const wxColour & col, wxFloodFillStyle style);
151
152 bool wxDFBDCImpl::DoFloodFill(wxCoord x, wxCoord y,
153 const wxColour& col, wxFloodFillStyle style)
154 {
155 return wxDoFloodFill(GetOwner(), x, y, col, style);
156 }
157
158 bool wxDFBDCImpl::DoGetPixel(wxCoord x, wxCoord y, wxColour *col) const
159 {
160 wxCHECK_MSG( col, false, "NULL colour parameter in wxDFBDCImpl::GetPixel");
161
162 wxFAIL_MSG( "GetPixel not implemented" );
163
164 wxUnusedVar(x);
165 wxUnusedVar(y);
166
167 return false;
168 }
169
170 void wxDFBDCImpl::DoCrossHair(wxCoord x, wxCoord y)
171 {
172 wxCHECK_RET( IsOk(), wxT("invalid dc") );
173
174 wxFAIL_MSG( "CrossHair not implemented" );
175
176 wxUnusedVar(x);
177 wxUnusedVar(y);
178 }
179
180 void wxDFBDCImpl::DoDrawLine(wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2)
181 {
182 wxCHECK_RET( IsOk(), wxT("invalid dc") );
183
184 if ( m_pen.GetStyle() == wxTRANSPARENT )
185 return;
186
187 wxCoord xx1 = XLOG2DEV(x1);
188 wxCoord yy1 = YLOG2DEV(y1);
189 wxCoord xx2 = XLOG2DEV(x2);
190 wxCoord yy2 = YLOG2DEV(y2);
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);
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)
220 void wxDFBDCImpl::DoDrawArc(wxCoord WXUNUSED(x1), wxCoord WXUNUSED(y1),
221 wxCoord WXUNUSED(x2), wxCoord WXUNUSED(y2),
222 wxCoord WXUNUSED(xc), wxCoord WXUNUSED(yc))
223 {
224 wxCHECK_RET( IsOk(), wxT("invalid dc") );
225
226 wxFAIL_MSG( "DrawArc not implemented" );
227 }
228
229 void wxDFBDCImpl::DoDrawPoint(wxCoord x, wxCoord y)
230 {
231 wxCHECK_RET( IsOk(), wxT("invalid dc") );
232
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)
239 }
240
241 void wxDFBDCImpl::DoDrawPolygon(int WXUNUSED(n), const wxPoint WXUNUSED(points)[],
242 wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset),
243 wxPolygonFillMode WXUNUSED(fillStyle))
244 {
245 wxCHECK_RET( IsOk(), wxT("invalid dc") );
246
247 wxFAIL_MSG( "DrawPolygon not implemented" );
248 }
249
250 void wxDFBDCImpl::DoDrawLines(int WXUNUSED(n), const wxPoint WXUNUSED(points)[],
251 wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset))
252 {
253 wxCHECK_RET( IsOk(), wxT("invalid dc") );
254
255 // TODO: impl. using DirectDB's DrawLines
256 wxFAIL_MSG( "DrawLines not implemented" );
257 }
258
259 void wxDFBDCImpl::DoDrawRectangle(wxCoord x, wxCoord y, wxCoord width, wxCoord height)
260 {
261 wxCHECK_RET( IsOk(), wxT("invalid dc") );
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());
284 m_surface->FillRectangle(xx, yy, ww, hh);
285 // restore pen's colour, because other drawing functions expect the
286 // colour to be set to the pen:
287 SelectColour(m_pen.GetColour());
288 }
289
290 if ( m_pen.GetStyle() != wxTRANSPARENT )
291 {
292 m_surface->DrawRectangle(xx, yy, ww, hh);
293 }
294
295 CalcBoundingBox(x, y);
296 CalcBoundingBox(x + width, y + height);
297 }
298
299 void wxDFBDCImpl::DoDrawRoundedRectangle(wxCoord WXUNUSED(x),
300 wxCoord WXUNUSED(y),
301 wxCoord WXUNUSED(width),
302 wxCoord WXUNUSED(height),
303 double WXUNUSED(radius))
304 {
305 wxCHECK_RET( IsOk(), wxT("invalid dc") );
306
307 wxFAIL_MSG( "DrawRoundedRectangle not implemented" );
308 }
309
310 void wxDFBDCImpl::DoDrawEllipse(wxCoord WXUNUSED(x),
311 wxCoord WXUNUSED(y),
312 wxCoord WXUNUSED(width),
313 wxCoord WXUNUSED(height))
314 {
315 wxCHECK_RET( IsOk(), wxT("invalid dc") );
316
317 wxFAIL_MSG( "DrawElipse not implemented" );
318 }
319
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))
326 {
327 wxCHECK_RET( IsOk(), wxT("invalid dc") );
328
329 wxFAIL_MSG( "DrawElipticArc not implemented" );
330 }
331
332 void wxDFBDCImpl::DoDrawText(const wxString& text, wxCoord x, wxCoord y)
333 {
334 wxCHECK_RET( IsOk(), wxT("invalid dc") );
335
336 wxCoord xx = XLOG2DEV(x);
337 wxCoord yy = YLOG2DEV(y);
338
339 // update the bounding box
340 wxCoord w, h;
341 CalcBoundingBox(x, y);
342 DoGetTextExtent(text, &w, &h);
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 {
348 wxCHECK_RET( m_textBackgroundColour.IsOk(),
349 wxT("invalid background color") );
350
351 SelectColour(m_textBackgroundColour);
352 m_surface->FillRectangle(xx, yy, XLOG2DEVREL(w), YLOG2DEVREL(h));
353 }
354
355 // finally draw the text itself:
356 wxCHECK_RET( m_textForegroundColour.IsOk(),
357 wxT("invalid foreground color") );
358 SelectColour(m_textForegroundColour);
359 m_surface->DrawString(text.utf8_str(), -1, xx, yy, DSTF_LEFT | DSTF_TOP);
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());
364 }
365
366 void wxDFBDCImpl::DoDrawRotatedText(const wxString& WXUNUSED(text),
367 wxCoord WXUNUSED(x), wxCoord WXUNUSED(y),
368 double WXUNUSED(angle))
369 {
370 wxCHECK_RET( IsOk(), wxT("invalid dc") );
371
372 wxFAIL_MSG( "DrawRotatedText not implemented" );
373 }
374
375 // ---------------------------------------------------------------------------
376 // set GDI objects
377 // ---------------------------------------------------------------------------
378
379 void wxDFBDCImpl::SetPen(const wxPen& pen)
380 {
381 m_pen = pen.IsOk() ? pen : DEFAULT_PEN;
382
383 SelectColour(m_pen.GetColour());
384 }
385
386 void wxDFBDCImpl::SetBrush(const wxBrush& brush)
387 {
388 m_brush = brush.IsOk() ? brush : DEFAULT_BRUSH;
389 }
390
391 void wxDFBDCImpl::SelectColour(const wxColour& clr)
392 {
393 m_surface->SetColor(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha());
394 #warning "use SetColorIndex?"
395 }
396
397 #if wxUSE_PALETTE
398 void wxDFBDCImpl::SetPalette(const wxPalette& WXUNUSED(palette))
399 {
400 wxCHECK_RET( IsOk(), wxT("invalid dc") );
401
402 wxFAIL_MSG( "SetPalette not implemented" );
403 }
404 #endif // wxUSE_PALETTE
405
406 void wxDFBDCImpl::SetFont(const wxFont& font)
407 {
408 wxCHECK_RET( IsOk(), wxT("invalid dc") );
409
410 wxFont f(font.IsOk() ? font : DEFAULT_FONT);
411
412 wxFont oldfont(m_font);
413
414 m_font = f;
415
416 if ( !m_surface->SetFont(GetCurrentFont()) )
417 {
418 m_font = oldfont;
419 return;
420 }
421 }
422
423 wxIDirectFBFontPtr wxDFBDCImpl::GetCurrentFont() const
424 {
425 bool aa = (GetDepth() > 8);
426 return m_font.GetDirectFBFont(aa);
427 }
428
429 void wxDFBDCImpl::SetBackground(const wxBrush& brush)
430 {
431 wxCHECK_RET( IsOk(), wxT("invalid dc") );
432
433 if (!brush.IsOk()) return;
434
435 m_backgroundBrush = brush;
436 }
437
438 void wxDFBDCImpl::SetBackgroundMode(int mode)
439 {
440 m_backgroundMode = mode;
441 }
442
443 void wxDFBDCImpl::SetLogicalFunction(wxRasterOperationMode function)
444 {
445 wxCHECK_RET( IsOk(), wxT("invalid dc") );
446
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,
450 "only wxCOPY logical function supported" );
451
452 m_logicalFunction = function;
453 }
454
455 bool wxDFBDCImpl::StartDoc(const wxString& WXUNUSED(message))
456 {
457 // We might be previewing, so return true to let it continue.
458 return true;
459 }
460
461 void wxDFBDCImpl::EndDoc()
462 {
463 }
464
465 void wxDFBDCImpl::StartPage()
466 {
467 }
468
469 void wxDFBDCImpl::EndPage()
470 {
471 }
472
473 // ---------------------------------------------------------------------------
474 // text metrics
475 // ---------------------------------------------------------------------------
476
477 wxCoord wxDFBDCImpl::GetCharHeight() const
478 {
479 wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") );
480 wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") );
481
482 int h = -1;
483 GetCurrentFont()->GetHeight(&h);
484 return YDEV2LOGREL(h);
485 }
486
487 wxCoord wxDFBDCImpl::GetCharWidth() const
488 {
489 wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") );
490 wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") );
491
492 int w = -1;
493 GetCurrentFont()->GetStringWidth("H", 1, &w);
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
499 void wxDFBDCImpl::DoGetTextExtent(const wxString& string, wxCoord *x, wxCoord *y,
500 wxCoord *descent, wxCoord *externalLeading,
501 const wxFont *theFont) const
502 {
503 wxCHECK_RET( IsOk(), wxT("invalid dc") );
504 wxCHECK_RET( m_font.IsOk(), wxT("no font selected") );
505 wxCHECK_RET( !theFont || theFont->IsOk(), wxT("invalid font") );
506
507 wxFont oldFont;
508 if ( theFont != NULL )
509 {
510 oldFont = m_font;
511 wxConstCast(this, wxDFBDCImpl)->SetFont(*theFont);
512 }
513
514 wxCoord xx = 0, yy = 0;
515 DFBRectangle rect;
516 wxIDirectFBFontPtr f = GetCurrentFont();
517
518 if ( f->GetStringExtents(string.utf8_str(), -1, &rect, NULL) )
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;
528 if ( f->GetDescender(&d) )
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 )
540 wxConstCast(this, wxDFBDCImpl)->SetFont(oldFont);
541 }
542
543
544
545 // ---------------------------------------------------------------------------
546 // mapping modes
547 // ---------------------------------------------------------------------------
548
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
552
553 void wxDFBDCImpl::DoGetSize(int *w, int *h) const
554 {
555 wxCHECK_RET( IsOk(), wxT("invalid dc") );
556
557 m_surface->GetSize(w, h);
558 }
559
560 void wxDFBDCImpl::DoGetSizeMM(int *width, int *height) const
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
570 wxSize wxDFBDCImpl::GetPPI() const
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
582 bool wxDFBDCImpl::DoBlit(wxCoord xdest, wxCoord ydest,
583 wxCoord width, wxCoord height,
584 wxDC *source, wxCoord xsrc, wxCoord ysrc,
585 wxRasterOperationMode rop, bool useMask,
586 wxCoord xsrcMask, wxCoord ysrcMask)
587 {
588 wxCHECK_MSG( IsOk(), false, "invalid dc" );
589 wxCHECK_MSG( source, false, "invalid source dc" );
590
591 // NB: we could also support XOR here (via DSBLIT_XOR)
592 // and possibly others via SetSrc/DstBlendFunction()
593 wxCHECK_MSG( rop == wxCOPY, false, "only wxCOPY function supported" );
594
595 // transform the source DC coords to the device ones
596 xsrc = source->LogicalToDeviceX(xsrc);
597 ysrc = source->LogicalToDeviceY(ysrc);
598
599 // FIXME_DFB: use the mask origin when drawing transparently
600 wxASSERT_MSG( xsrcMask == -1 && ysrcMask == -1,
601 "non-default source mask offset not implemented" );
602 #if 0
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 }
612 #endif
613
614 wxMemoryDC *sourceAsMemDC = wxDynamicCast(source, wxMemoryDC);
615 if ( sourceAsMemDC )
616 {
617 DoDrawSubBitmap(sourceAsMemDC->GetSelectedBitmap(),
618 xsrc, ysrc,
619 width, height,
620 xdest, ydest,
621 rop,
622 useMask);
623 }
624 else
625 {
626 return DoBlitFromSurface
627 (
628 static_cast<wxDFBDCImpl *>(source->GetImpl())
629 ->GetDirectFBSurface(),
630 xsrc, ysrc,
631 width, height,
632 xdest, ydest
633 );
634 }
635
636 return true;
637 }
638
639 void wxDFBDCImpl::DoDrawBitmap(const wxBitmap &bmp, wxCoord x, wxCoord y, bool useMask)
640 {
641 wxCHECK_RET( IsOk(), wxT("invalid dc") );
642 wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") );
643
644 DoDrawSubBitmap(bmp,
645 0, 0, bmp.GetWidth(), bmp.GetHeight(),
646 x, y,
647 m_logicalFunction, useMask);
648 }
649
650 void wxDFBDCImpl::DoDrawIcon(const wxIcon& icon, wxCoord x, wxCoord y)
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
656 void wxDFBDCImpl::DoDrawSubBitmap(const wxBitmap &bmp,
657 wxCoord x, wxCoord y, wxCoord w, wxCoord h,
658 wxCoord destx, wxCoord desty, int rop, bool useMask)
659 {
660 wxCHECK_RET( IsOk(), wxT("invalid dc") );
661 wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") );
662
663 // NB: we could also support XOR here (via DSBLIT_XOR)
664 // and possibly others via SetSrc/DstBlendFunction()
665 wxCHECK_RET( rop == wxCOPY, "only wxCOPY function supported" );
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.
671 wxFAIL_MSG( "drawing mono bitmaps not implemented" );
672 return;
673 }
674
675 if ( useMask && bmp.GetMask() )
676 {
677 // FIXME_DFB: Could use blitting modes for this; also see
678 // DFB's SetSrcBlendFunction() and SetSrcColorKey()
679 wxFAIL_MSG( "drawing bitmaps with masks not implemented" );
680 return;
681 }
682
683 DoBlitFromSurface(bmp.GetDirectFBSurface(),
684 x, y,
685 w, h,
686 destx, desty);
687 }
688
689 bool wxDFBDCImpl::DoBlitFromSurface(const wxIDirectFBSurfacePtr& src,
690 wxCoord srcx, wxCoord srcy,
691 wxCoord w, wxCoord h,
692 wxCoord dstx, wxCoord dsty)
693 {
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 {
700 wxLogDebug("Blitting from area outside of the source surface, caller code needs fixing.");
701 return false;
702 }
703
704 CalcBoundingBox(dstx, dsty);
705 CalcBoundingBox(dstx + w, dsty + h);
706
707 DFBRectangle srcRect = { srcx, srcy, w, h };
708 DFBRectangle dstRect = { XLOG2DEV(dstx), YLOG2DEV(dsty),
709 XLOG2DEVREL(w), YLOG2DEVREL(h) };
710
711 wxIDirectFBSurfacePtr dst(m_surface);
712
713 // FIXME: this will have to be different in useMask case, see above
714 DFBSurfaceBlittingFlags blitFlag = (src->GetPixelFormat() == DSPF_ARGB)
715 ? DSBLIT_BLEND_ALPHACHANNEL
716 : DSBLIT_NOFX;
717 if ( !dst->SetBlittingFlags(blitFlag) )
718 return false;
719
720 if ( srcRect.w != dstRect.w || srcRect.h != dstRect.h )
721 {
722 // the bitmap is drawn stretched:
723 dst->StretchBlit(src, &srcRect, &dstRect);
724 }
725 else
726 {
727 // no stretching, size is preserved:
728 dst->Blit(src, &srcRect, dstRect.x, dstRect.y);
729 }
730
731 return true;
732 }