LCOV - code coverage report
Current view: top level - libreoffice/vcl/unx/generic/gdi - salgdi.cxx (source / functions) Hit Total Coverage
Test: libreoffice_filtered.info Lines: 0 583 0.0 %
Date: 2012-12-27 Functions: 0 48 0.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include "tools/debug.hxx"
      21             : 
      22             : #include "basegfx/polygon/b2dpolygon.hxx"
      23             : #include "basegfx/polygon/b2dpolypolygon.hxx"
      24             : #include "basegfx/polygon/b2dpolypolygontools.hxx"
      25             : #include "basegfx/polygon/b2dpolygontools.hxx"
      26             : #include "basegfx/polygon/b2dpolygonclipper.hxx"
      27             : #include "basegfx/polygon/b2dlinegeometry.hxx"
      28             : #include "basegfx/matrix/b2dhommatrix.hxx"
      29             : #include "basegfx/matrix/b2dhommatrixtools.hxx"
      30             : #include "basegfx/polygon/b2dpolypolygoncutter.hxx"
      31             : #include "basegfx/polygon/b2dtrapezoid.hxx"
      32             : 
      33             : #include "vcl/jobdata.hxx"
      34             : 
      35             : #include "unx/Xproto.h"
      36             : #include "unx/salunx.h"
      37             : #include "unx/saldata.hxx"
      38             : #include "unx/saldisp.hxx"
      39             : #include "unx/salgdi.h"
      40             : #include "unx/salframe.h"
      41             : #include "unx/salvd.h"
      42             : #include <unx/x11/xlimits.hxx>
      43             : 
      44             : #include "generic/printergfx.hxx"
      45             : #include "xrender_peer.hxx"
      46             : #include "region.h"
      47             : 
      48             : #include <vector>
      49             : #include <queue>
      50             : #include <set>
      51             : 
      52             : #include <config_graphite.h>
      53             : 
      54             : // -=-= SalPolyLine =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
      55             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
      56             : #define STATIC_POINTS 64
      57             : 
      58             : class SalPolyLine
      59             : {
      60             :             XPoint              Points_[STATIC_POINTS];
      61             :             XPoint             *pFirst_;
      62             : public:
      63             :     inline                      SalPolyLine( sal_uLong nPoints );
      64             :     inline                      SalPolyLine( sal_uLong nPoints, const SalPoint *p );
      65             :     inline                      ~SalPolyLine();
      66           0 :     inline  XPoint             &operator [] ( sal_uLong n ) const
      67           0 :                                 { return pFirst_[n]; }
      68             : };
      69             : 
      70             : inline SalPolyLine::SalPolyLine( sal_uLong nPoints )
      71             :     : pFirst_( nPoints+1 > STATIC_POINTS ? new XPoint[nPoints+1] : Points_ )
      72             : {}
      73             : 
      74           0 : inline SalPolyLine::SalPolyLine( sal_uLong nPoints, const SalPoint *p )
      75           0 :     : pFirst_( nPoints+1 > STATIC_POINTS ? new XPoint[nPoints+1] : Points_ )
      76             : {
      77           0 :     for( sal_uLong i = 0; i < nPoints; i++ )
      78             :     {
      79           0 :         pFirst_[i].x = (short)p[i].mnX;
      80           0 :         pFirst_[i].y = (short)p[i].mnY;
      81             :     }
      82           0 :     pFirst_[nPoints] = pFirst_[0]; // close polyline
      83           0 : }
      84             : 
      85           0 : inline SalPolyLine::~SalPolyLine()
      86           0 : { if( pFirst_ != Points_ ) delete [] pFirst_; }
      87             : 
      88             : #undef STATIC_POINTS
      89             : // -=-= X11SalGraphics =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
      90             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
      91           0 : X11SalGraphics::X11SalGraphics()
      92           0 :     : m_nXScreen( 0 )
      93             : {
      94           0 :     m_pFrame            = NULL;
      95           0 :     m_pVDev             = NULL;
      96           0 :     m_pDeleteColormap   = NULL;
      97           0 :     hDrawable_          = None;
      98           0 :     m_aXRenderPicture    = 0;
      99           0 :     m_pXRenderFormat     = NULL;
     100             : 
     101           0 :     mpClipRegion            = NULL;
     102           0 :     pPaintRegion_       = NULL;
     103             : 
     104           0 :     pPenGC_         = NULL;
     105           0 :     nPenPixel_          = 0;
     106           0 :     nPenColor_          = MAKE_SALCOLOR( 0x00, 0x00, 0x00 ); // Black
     107             : 
     108           0 :     pFontGC_            = NULL;
     109           0 :     for( int i = 0; i < MAX_FALLBACK; ++i )
     110           0 :         mpServerFont[i] = NULL;
     111             : 
     112           0 :     nTextPixel_         = 0;
     113           0 :     nTextColor_         = MAKE_SALCOLOR( 0x00, 0x00, 0x00 ); // Black
     114             : 
     115             : #ifdef ENABLE_GRAPHITE
     116             :     // check if graphite fonts have been disabled
     117           0 :     static const char* pDisableGraphiteStr = getenv( "SAL_DISABLE_GRAPHITE" );
     118           0 :     bDisableGraphite_       = pDisableGraphiteStr ? (pDisableGraphiteStr[0]!='0') : sal_False;
     119             : #endif
     120             : 
     121           0 :     pBrushGC_           = NULL;
     122           0 :     nBrushPixel_            = 0;
     123           0 :     nBrushColor_        = MAKE_SALCOLOR( 0xFF, 0xFF, 0xFF ); // White
     124           0 :     hBrush_             = None;
     125             : 
     126           0 :     pMonoGC_            = NULL;
     127           0 :     pCopyGC_            = NULL;
     128           0 :     pMaskGC_            = NULL;
     129           0 :     pInvertGC_          = NULL;
     130           0 :     pInvert50GC_        = NULL;
     131           0 :     pStippleGC_         = NULL;
     132           0 :     pTrackingGC_        = NULL;
     133             : 
     134           0 :     bWindow_            = sal_False;
     135           0 :     bPrinter_           = sal_False;
     136           0 :     bVirDev_            = sal_False;
     137           0 :     bPenGC_         = sal_False;
     138           0 :     bFontGC_            = sal_False;
     139           0 :     bBrushGC_           = sal_False;
     140           0 :     bMonoGC_            = sal_False;
     141           0 :     bCopyGC_            = sal_False;
     142           0 :     bInvertGC_          = sal_False;
     143           0 :     bInvert50GC_        = sal_False;
     144           0 :     bStippleGC_         = sal_False;
     145           0 :     bTrackingGC_        = sal_False;
     146           0 :     bXORMode_           = sal_False;
     147           0 :     bDitherBrush_       = sal_False;
     148           0 : }
     149             : 
     150             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     151           0 : X11SalGraphics::~X11SalGraphics()
     152             : {
     153           0 :     ReleaseFonts();
     154           0 :     freeResources();
     155           0 : }
     156             : 
     157             : // -=-= SalGraphics / X11SalGraphics =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     158             : 
     159           0 : void X11SalGraphics::freeResources()
     160             : {
     161           0 :     Display *pDisplay = GetXDisplay();
     162             : 
     163             :     DBG_ASSERT( !pPaintRegion_, "pPaintRegion_" );
     164           0 :     if( mpClipRegion ) XDestroyRegion( mpClipRegion ), mpClipRegion = None;
     165             : 
     166           0 :     if( hBrush_ )       XFreePixmap( pDisplay, hBrush_ ), hBrush_ = None;
     167           0 :     if( pPenGC_ )       XFreeGC( pDisplay, pPenGC_ ), pPenGC_ = None;
     168           0 :     if( pFontGC_ )      XFreeGC( pDisplay, pFontGC_ ), pFontGC_ = None;
     169           0 :     if( pBrushGC_ )     XFreeGC( pDisplay, pBrushGC_ ), pBrushGC_ = None;
     170           0 :     if( pMonoGC_ )      XFreeGC( pDisplay, pMonoGC_ ), pMonoGC_ = None;
     171           0 :     if( pCopyGC_ )      XFreeGC( pDisplay, pCopyGC_ ), pCopyGC_ = None;
     172           0 :     if( pMaskGC_ )      XFreeGC( pDisplay, pMaskGC_ ), pMaskGC_ = None;
     173           0 :     if( pInvertGC_ )    XFreeGC( pDisplay, pInvertGC_ ), pInvertGC_ = None;
     174           0 :     if( pInvert50GC_ )  XFreeGC( pDisplay, pInvert50GC_ ), pInvert50GC_ = None;
     175           0 :     if( pStippleGC_ )   XFreeGC( pDisplay, pStippleGC_ ), pStippleGC_ = None;
     176           0 :     if( pTrackingGC_ )  XFreeGC( pDisplay, pTrackingGC_ ), pTrackingGC_ = None;
     177           0 :     if( m_pDeleteColormap )
     178           0 :         delete m_pDeleteColormap, m_pColormap = m_pDeleteColormap = NULL;
     179             : 
     180           0 :     if( m_aXRenderPicture )
     181           0 :         XRenderPeer::GetInstance().FreePicture( m_aXRenderPicture ), m_aXRenderPicture = 0;
     182             : 
     183           0 :     bPenGC_ = bFontGC_ = bBrushGC_ = bMonoGC_ = bCopyGC_ = bInvertGC_ = bInvert50GC_ = bStippleGC_ = bTrackingGC_ = false;
     184           0 : }
     185             : 
     186           0 : void X11SalGraphics::SetDrawable( Drawable aDrawable, SalX11Screen nXScreen )
     187             : {
     188             :     // shortcut if nothing changed
     189           0 :     if( hDrawable_ == aDrawable )
     190           0 :         return;
     191             : 
     192             :     // free screen specific resources if needed
     193           0 :     if( nXScreen != m_nXScreen )
     194             :     {
     195           0 :         freeResources();
     196           0 :         m_pColormap = &GetGenericData()->GetSalDisplay()->GetColormap( nXScreen );
     197           0 :         m_nXScreen = nXScreen;
     198             :     }
     199             : 
     200           0 :     hDrawable_ = aDrawable;
     201           0 :     SetXRenderFormat( NULL );
     202           0 :     if( m_aXRenderPicture )
     203             :     {
     204           0 :         XRenderPeer::GetInstance().FreePicture( m_aXRenderPicture );
     205           0 :         m_aXRenderPicture = 0;
     206             :     }
     207             : 
     208           0 :     if( hDrawable_ )
     209             :     {
     210           0 :         nPenPixel_      = GetPixel( nPenColor_ );
     211           0 :         nTextPixel_     = GetPixel( nTextColor_ );
     212           0 :         nBrushPixel_    = GetPixel( nBrushColor_ );
     213             :     }
     214             : }
     215             : 
     216           0 : void X11SalGraphics::Init( SalFrame *pFrame, Drawable aTarget,
     217             :                            SalX11Screen nXScreen )
     218             : {
     219           0 :     m_pColormap = &GetGenericData()->GetSalDisplay()->GetColormap(nXScreen);
     220           0 :     m_nXScreen  = nXScreen;
     221           0 :     SetDrawable( aTarget, nXScreen );
     222             : 
     223           0 :     bWindow_    = sal_True;
     224           0 :     m_pFrame    = pFrame;
     225           0 :     m_pVDev     = NULL;
     226           0 : }
     227             : 
     228             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     229           0 : void X11SalGraphics::DeInit()
     230             : {
     231           0 :     SetDrawable( None, m_nXScreen );
     232           0 : }
     233             : 
     234             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     235           0 : void X11SalGraphics::SetClipRegion( GC pGC, XLIB_Region pXReg ) const
     236             : {
     237           0 :     Display *pDisplay = GetXDisplay();
     238             : 
     239           0 :     int n = 0;
     240             :     XLIB_Region Regions[3];
     241             : 
     242           0 :     if( mpClipRegion )
     243           0 :         Regions[n++] = mpClipRegion;
     244             : 
     245           0 :     if( pXReg && !XEmptyRegion( pXReg ) )
     246           0 :         Regions[n++] = pXReg;
     247             : 
     248           0 :     if( 0 == n )
     249           0 :         XSetClipMask( pDisplay, pGC, None );
     250           0 :     else if( 1 == n )
     251           0 :         XSetRegion( pDisplay, pGC, Regions[0] );
     252             :     else
     253             :     {
     254           0 :         XLIB_Region pTmpRegion = XCreateRegion();
     255           0 :         XIntersectRegion( Regions[0], Regions[1], pTmpRegion );
     256             : 
     257           0 :         XSetRegion( pDisplay, pGC, pTmpRegion );
     258           0 :         XDestroyRegion( pTmpRegion );
     259             :     }
     260           0 : }
     261             : 
     262             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     263           0 : GC X11SalGraphics::SelectPen()
     264             : {
     265           0 :     Display *pDisplay = GetXDisplay();
     266             : 
     267           0 :     if( !pPenGC_ )
     268             :     {
     269             :         XGCValues values;
     270           0 :         values.subwindow_mode       = ClipByChildren;
     271           0 :         values.fill_rule            = EvenOddRule;      // Pict import/ Gradient
     272           0 :         values.graphics_exposures   = False;
     273             : 
     274             :         pPenGC_ = XCreateGC( pDisplay, hDrawable_,
     275             :                              GCSubwindowMode | GCFillRule | GCGraphicsExposures,
     276           0 :                              &values );
     277             :     }
     278             : 
     279           0 :     if( !bPenGC_ )
     280             :     {
     281           0 :         if( nPenColor_ != SALCOLOR_NONE )
     282           0 :             XSetForeground( pDisplay, pPenGC_, nPenPixel_ );
     283           0 :         XSetFunction  ( pDisplay, pPenGC_, bXORMode_ ? GXxor : GXcopy );
     284           0 :         SetClipRegion( pPenGC_ );
     285           0 :         bPenGC_ = sal_True;
     286             :     }
     287             : 
     288           0 :     return pPenGC_;
     289             : }
     290             : 
     291             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     292           0 : GC X11SalGraphics::SelectBrush()
     293             : {
     294           0 :     Display *pDisplay = GetXDisplay();
     295             : 
     296             :     DBG_ASSERT( nBrushColor_ != SALCOLOR_NONE, "Brush Transparent" );
     297             : 
     298           0 :     if( !pBrushGC_ )
     299             :     {
     300             :         XGCValues values;
     301           0 :         values.subwindow_mode       = ClipByChildren;
     302           0 :         values.fill_rule            = EvenOddRule;      // Pict import/ Gradient
     303           0 :         values.graphics_exposures   = False;
     304             : 
     305             :         pBrushGC_ = XCreateGC( pDisplay, hDrawable_,
     306             :                                GCSubwindowMode | GCFillRule | GCGraphicsExposures,
     307           0 :                                &values );
     308             :     }
     309             : 
     310           0 :     if( !bBrushGC_ )
     311             :     {
     312           0 :         if( !bDitherBrush_ )
     313             :         {
     314           0 :             XSetFillStyle ( pDisplay, pBrushGC_, FillSolid );
     315           0 :             XSetForeground( pDisplay, pBrushGC_, nBrushPixel_ );
     316           0 :             if( bPrinter_ )
     317           0 :                 XSetTile( pDisplay, pBrushGC_, None );
     318             :         }
     319             :         else
     320             :         {
     321             :             // Bug in Sun Solaris 2.5.1, XFillPolygon doesn't allways reflect
     322             :             // changes of the tile. PROPERTY_BUG_Tile doesn't fix this !
     323           0 :             if (GetDisplay()->GetProperties() & PROPERTY_BUG_FillPolygon_Tile)
     324           0 :                 XSetFillStyle ( pDisplay, pBrushGC_, FillSolid );
     325             : 
     326           0 :             XSetFillStyle ( pDisplay, pBrushGC_, FillTiled );
     327           0 :             XSetTile      ( pDisplay, pBrushGC_, hBrush_ );
     328             :         }
     329           0 :         XSetFunction  ( pDisplay, pBrushGC_, bXORMode_ ? GXxor : GXcopy );
     330           0 :         SetClipRegion( pBrushGC_ );
     331             : 
     332           0 :         bBrushGC_ = sal_True;
     333             :     }
     334             : 
     335           0 :     return pBrushGC_;
     336             : }
     337             : 
     338             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     339           0 : GC X11SalGraphics::GetTrackingGC()
     340             : {
     341           0 :     const char    dash_list[2] = {2, 2};
     342             : 
     343           0 :     if( !pTrackingGC_ )
     344             :     {
     345             :         XGCValues     values;
     346             : 
     347           0 :         values.graphics_exposures   = False;
     348           0 :         values.foreground           = m_pColormap->GetBlackPixel()
     349           0 :                                       ^ m_pColormap->GetWhitePixel();
     350           0 :         values.function             = GXxor;
     351           0 :         values.line_width           = 1;
     352           0 :         values.line_style           = LineOnOffDash;
     353             : 
     354             :         pTrackingGC_ = XCreateGC( GetXDisplay(), GetDrawable(),
     355             :                                   GCGraphicsExposures | GCForeground | GCFunction
     356             :                                   | GCLineWidth | GCLineStyle,
     357           0 :                                   &values );
     358           0 :         XSetDashes( GetXDisplay(), pTrackingGC_, 0, dash_list, 2 );
     359             :     }
     360             : 
     361           0 :     if( !bTrackingGC_ )
     362             :     {
     363           0 :         SetClipRegion( pTrackingGC_ );
     364           0 :         bTrackingGC_ = sal_True;
     365             :     }
     366             : 
     367           0 :     return pTrackingGC_;
     368             : }
     369             : 
     370             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     371           0 : void X11SalGraphics::DrawLines( sal_uLong              nPoints,
     372             :                                 const SalPolyLine &rPoints,
     373             :                                 GC                 pGC,
     374             :                                 bool               bClose
     375             :                                 )
     376             : {
     377             :     // errechne wie viele Linien XWindow auf einmal zeichnen kann
     378           0 :     sal_uLong nMaxLines = (GetDisplay()->GetMaxRequestSize() - sizeof(xPolyPointReq))
     379           0 :                       / sizeof(xPoint);
     380           0 :     if( nMaxLines > nPoints ) nMaxLines = nPoints;
     381             : 
     382             :     // gebe alle Linien aus, die XWindows zeichnen kann.
     383             :     sal_uLong n;
     384           0 :     for( n = 0; nPoints - n > nMaxLines; n += nMaxLines - 1 )
     385             :         XDrawLines( GetXDisplay(),
     386             :                     GetDrawable(),
     387             :                     pGC,
     388           0 :                     &rPoints[n],
     389             :                     nMaxLines,
     390           0 :                     CoordModeOrigin );
     391             : 
     392           0 :     if( n < nPoints )
     393             :         XDrawLines( GetXDisplay(),
     394             :                     GetDrawable(),
     395             :                     pGC,
     396           0 :                     &rPoints[n],
     397             :                     nPoints - n,
     398           0 :                     CoordModeOrigin );
     399           0 :     if( bClose )
     400             :     {
     401           0 :         if( rPoints[nPoints-1].x != rPoints[0].x || rPoints[nPoints-1].y != rPoints[0].y )
     402           0 :             drawLine( rPoints[nPoints-1].x, rPoints[nPoints-1].y, rPoints[0].x, rPoints[0].y );
     403             :     }
     404           0 : }
     405             : 
     406             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     407             : // Dithern: Calculate a dither-pixmap and make a brush of it
     408             : #define P_DELTA         51
     409             : #define DMAP( v, m )    ((v % P_DELTA) > m ? (v / P_DELTA) + 1 : (v / P_DELTA))
     410             : 
     411           0 : BOOL X11SalGraphics::GetDitherPixmap( SalColor nSalColor )
     412             : {
     413             :     static const short nOrdDither8Bit[ 8 ][ 8 ] =
     414             :     {
     415             :         { 0, 38,  9, 48,  2, 40, 12, 50},
     416             :         {25, 12, 35, 22, 28, 15, 37, 24},
     417             :         { 6, 44,  3, 41,  8, 47,  5, 44},
     418             :         {32, 19, 28, 16, 34, 21, 31, 18},
     419             :         { 1, 40, 11, 49,  0, 39, 10, 48},
     420             :         {27, 14, 36, 24, 26, 13, 36, 23},
     421             :         { 8, 46,  4, 43,  7, 45,  4, 42},
     422             :         {33, 20, 30, 17, 32, 20, 29, 16}
     423             :     };
     424             : 
     425             :     // test for correct depth (8bit)
     426           0 :     if( GetColormap().GetVisual().GetDepth() != 8 )
     427           0 :         return sal_False;
     428             : 
     429             :     char    pBits[64];
     430           0 :     char   *pBitsPtr = pBits;
     431             : 
     432             :     // Set the pallette-entries for the dithering tile
     433           0 :     sal_uInt8 nSalColorRed   = SALCOLOR_RED   ( nSalColor );
     434           0 :     sal_uInt8 nSalColorGreen = SALCOLOR_GREEN ( nSalColor );
     435           0 :     sal_uInt8 nSalColorBlue  = SALCOLOR_BLUE  ( nSalColor );
     436             : 
     437           0 :     for( int nY = 0; nY < 8; nY++ )
     438             :     {
     439           0 :         for( int nX = 0; nX < 8; nX++ )
     440             :         {
     441           0 :             short nMagic = nOrdDither8Bit[nY][nX];
     442           0 :             sal_uInt8 nR   = P_DELTA * DMAP( nSalColorRed,   nMagic );
     443           0 :             sal_uInt8 nG   = P_DELTA * DMAP( nSalColorGreen, nMagic );
     444           0 :             sal_uInt8 nB   = P_DELTA * DMAP( nSalColorBlue,  nMagic );
     445             : 
     446           0 :             *pBitsPtr++ = GetColormap().GetPixel( MAKE_SALCOLOR( nR, nG, nB ) );
     447             :         }
     448             :     }
     449             : 
     450             :     // create the tile as ximage and an according pixmap -> caching
     451             :     XImage *pImage = XCreateImage( GetXDisplay(),
     452           0 :                                    GetColormap().GetXVisual(),
     453             :                                    8,
     454             :                                    ZPixmap,
     455             :                                    0,               // offset
     456             :                                    pBits,           // data
     457             :                                    8, 8,            // width & height
     458             :                                    8,               // bitmap_pad
     459           0 :                                    0 );             // (default) bytes_per_line
     460             : 
     461           0 :     if ( GetDisplay()->GetProperties() & PROPERTY_BUG_Tile )
     462             :     {
     463           0 :         if (hBrush_)
     464           0 :             XFreePixmap (GetXDisplay(), hBrush_);
     465           0 :         hBrush_ = limitXCreatePixmap( GetXDisplay(), GetDrawable(), 8, 8, 8 );
     466             :     }
     467             :     else
     468           0 :     if( !hBrush_ )
     469           0 :         hBrush_ = limitXCreatePixmap( GetXDisplay(), GetDrawable(), 8, 8, 8 );
     470             : 
     471             :     // put the ximage to the pixmap
     472             :     XPutImage( GetXDisplay(),
     473             :                hBrush_,
     474             :                GetDisplay()->GetCopyGC( m_nXScreen ),
     475             :                pImage,
     476             :                0, 0,                        // Source
     477             :                0, 0,                        // Destination
     478           0 :                8, 8 );                      // width & height
     479             : 
     480             :     // destroy image-frame but not palette-data
     481           0 :     pImage->data = NULL;
     482           0 :     XDestroyImage( pImage );
     483             : 
     484           0 :     return sal_True;
     485             : }
     486             : 
     487             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     488           0 : void X11SalGraphics::GetResolution( sal_Int32 &rDPIX, sal_Int32 &rDPIY ) // const
     489             : {
     490           0 :     const SalDisplay *pDisplay = GetDisplay();
     491           0 :     if (!pDisplay)
     492             :     {
     493             :         OSL_TRACE("Null display");
     494           0 :         rDPIX = rDPIY = 96;
     495           0 :         return;
     496             :     }
     497             : 
     498           0 :     rDPIX = pDisplay->GetResolution().A();
     499           0 :     rDPIY = pDisplay->GetResolution().B();
     500           0 :     if( !pDisplay->GetExactResolution() && rDPIY < 96 )
     501             :     {
     502           0 :         rDPIX = Divide( rDPIX * 96, rDPIY );
     503           0 :             rDPIY = 96;
     504             :     }
     505           0 :     else if ( rDPIY > 200 )
     506             :     {
     507           0 :         rDPIX = Divide( rDPIX * 200, rDPIY );
     508           0 :         rDPIY = 200;
     509             :     }
     510             : 
     511             :     // #i12705# equalize x- and y-resolution if they are close enough
     512           0 :     if( rDPIX != rDPIY )
     513             :     {
     514             :         // different x- and y- resolutions are usually artifacts of
     515             :         // a wrongly calculated screen size.
     516             : #ifdef DEBUG
     517             :         printf("Forcing Resolution from %" SAL_PRIdINT32 "x%" SAL_PRIdINT32 " to %" SAL_PRIdINT32 "x%" SAL_PRIdINT32 "\n",
     518             :                 rDPIX,rDPIY,rDPIY,rDPIY);
     519             : #endif
     520           0 :         rDPIX = rDPIY; // y-resolution is more trustworthy
     521             :     }
     522             : }
     523             : 
     524             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     525           0 : sal_uInt16 X11SalGraphics::GetBitCount() const
     526             : {
     527           0 :     return GetVisual().GetDepth();
     528             : }
     529             : 
     530             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     531           0 : long X11SalGraphics::GetGraphicsWidth() const
     532             : {
     533           0 :     if( m_pFrame )
     534           0 :         return m_pFrame->maGeometry.nWidth;
     535           0 :     else if( m_pVDev )
     536           0 :         return m_pVDev->GetWidth();
     537             :     else
     538           0 :         return 0;
     539             : }
     540             : 
     541             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     542           0 : long X11SalGraphics::GetGraphicsHeight() const
     543             : {
     544           0 :     if( m_pFrame )
     545           0 :         return m_pFrame->maGeometry.nHeight;
     546           0 :     else if( m_pVDev )
     547           0 :         return m_pVDev->GetHeight();
     548             :     else
     549           0 :         return 0;
     550             : }
     551             : 
     552             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     553           0 : void X11SalGraphics::ResetClipRegion()
     554             : {
     555           0 :     if( mpClipRegion )
     556             :     {
     557           0 :         bPenGC_         = sal_False;
     558           0 :         bFontGC_        = sal_False;
     559           0 :         bBrushGC_       = sal_False;
     560           0 :         bMonoGC_        = sal_False;
     561           0 :         bCopyGC_        = sal_False;
     562           0 :         bInvertGC_      = sal_False;
     563           0 :         bInvert50GC_    = sal_False;
     564           0 :         bStippleGC_     = sal_False;
     565           0 :         bTrackingGC_    = sal_False;
     566             : 
     567           0 :         XDestroyRegion( mpClipRegion );
     568           0 :         mpClipRegion    = NULL;
     569             :     }
     570           0 : }
     571             : 
     572           0 : bool X11SalGraphics::setClipRegion( const Region& i_rClip )
     573             : {
     574           0 :     if( mpClipRegion )
     575           0 :         XDestroyRegion( mpClipRegion );
     576           0 :     mpClipRegion = XCreateRegion();
     577             : 
     578             :     ImplRegionInfo aInfo;
     579             :     long nX, nY, nW, nH;
     580           0 :     bool bRegionRect = i_rClip.ImplGetFirstRect(aInfo, nX, nY, nW, nH );
     581           0 :     while( bRegionRect )
     582             :     {
     583           0 :         if ( nW && nH )
     584             :         {
     585             :             XRectangle aRect;
     586           0 :             aRect.x         = (short)nX;
     587           0 :             aRect.y         = (short)nY;
     588           0 :             aRect.width     = (unsigned short)nW;
     589           0 :             aRect.height    = (unsigned short)nH;
     590             : 
     591           0 :             XUnionRectWithRegion( &aRect, mpClipRegion, mpClipRegion );
     592             :         }
     593           0 :         bRegionRect = i_rClip.ImplGetNextRect( aInfo, nX, nY, nW, nH );
     594             :     }
     595             : 
     596             :     // done, invalidate GCs
     597           0 :     bPenGC_         = sal_False;
     598           0 :     bFontGC_        = sal_False;
     599           0 :     bBrushGC_       = sal_False;
     600           0 :     bMonoGC_        = sal_False;
     601           0 :     bCopyGC_        = sal_False;
     602           0 :     bInvertGC_      = sal_False;
     603           0 :     bInvert50GC_    = sal_False;
     604           0 :     bStippleGC_     = sal_False;
     605           0 :     bTrackingGC_    = sal_False;
     606             : 
     607           0 :     if( XEmptyRegion( mpClipRegion ) )
     608             :     {
     609           0 :         XDestroyRegion( mpClipRegion );
     610           0 :         mpClipRegion= NULL;
     611             :     }
     612           0 :     return true;
     613             : }
     614             : 
     615             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     616           0 : void X11SalGraphics::SetLineColor()
     617             : {
     618           0 :     if( nPenColor_ != SALCOLOR_NONE )
     619             :     {
     620           0 :         nPenColor_      = SALCOLOR_NONE;
     621           0 :         bPenGC_         = sal_False;
     622             :     }
     623           0 : }
     624             : 
     625             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     626           0 : void X11SalGraphics::SetLineColor( SalColor nSalColor )
     627             : {
     628           0 :     if( nPenColor_ != nSalColor )
     629             :     {
     630           0 :         nPenColor_      = nSalColor;
     631           0 :         nPenPixel_      = GetPixel( nSalColor );
     632           0 :         bPenGC_         = sal_False;
     633             :     }
     634           0 : }
     635             : 
     636             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     637           0 : void X11SalGraphics::SetFillColor()
     638             : {
     639           0 :     if( nBrushColor_ != SALCOLOR_NONE )
     640             :     {
     641           0 :         bDitherBrush_   = sal_False;
     642           0 :         nBrushColor_    = SALCOLOR_NONE;
     643           0 :         bBrushGC_       = sal_False;
     644             :     }
     645           0 : }
     646             : 
     647             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     648           0 : void X11SalGraphics::SetFillColor( SalColor nSalColor )
     649             : {
     650           0 :     if( nBrushColor_ != nSalColor )
     651             :     {
     652           0 :         bDitherBrush_   = sal_False;
     653           0 :         nBrushColor_    = nSalColor;
     654           0 :         nBrushPixel_    = GetPixel( nSalColor );
     655           0 :         if( TrueColor != GetColormap().GetVisual().GetClass()
     656           0 :             && GetColormap().GetColor( nBrushPixel_ ) != nBrushColor_
     657             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0x00, 0x00 ) // black
     658             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0x00, 0x80 ) // blue
     659             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0x80, 0x00 ) // green
     660             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0x80, 0x80 ) // cyan
     661             :             && nSalColor != MAKE_SALCOLOR( 0x80, 0x00, 0x00 ) // red
     662             :             && nSalColor != MAKE_SALCOLOR( 0x80, 0x00, 0x80 ) // magenta
     663             :             && nSalColor != MAKE_SALCOLOR( 0x80, 0x80, 0x00 ) // brown
     664             :             && nSalColor != MAKE_SALCOLOR( 0x80, 0x80, 0x80 ) // gray
     665             :             && nSalColor != MAKE_SALCOLOR( 0xC0, 0xC0, 0xC0 ) // light gray
     666             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0x00, 0xFF ) // light blue
     667             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0xFF, 0x00 ) // light green
     668             :             && nSalColor != MAKE_SALCOLOR( 0x00, 0xFF, 0xFF ) // light cyan
     669             :             && nSalColor != MAKE_SALCOLOR( 0xFF, 0x00, 0x00 ) // light red
     670             :             && nSalColor != MAKE_SALCOLOR( 0xFF, 0x00, 0xFF ) // light magenta
     671             :             && nSalColor != MAKE_SALCOLOR( 0xFF, 0xFF, 0x00 ) // light brown
     672             :             && nSalColor != MAKE_SALCOLOR( 0xFF, 0xFF, 0xFF ) )
     673           0 :             bDitherBrush_ = GetDitherPixmap(nSalColor);
     674           0 :         bBrushGC_       = sal_False;
     675             :     }
     676           0 : }
     677             : 
     678             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     679           0 : void X11SalGraphics::SetROPLineColor( SalROPColor nROPColor )
     680             : {
     681           0 :     switch( nROPColor )
     682             :     {
     683             :         case SAL_ROP_0 : // 0
     684           0 :             nPenPixel_ = (Pixel)0;
     685           0 :             break;
     686             :         case SAL_ROP_1 : // 1
     687           0 :             nPenPixel_ = (Pixel)(1 << GetVisual().GetDepth()) - 1;
     688           0 :             break;
     689             :         case SAL_ROP_INVERT : // 2
     690           0 :             nPenPixel_ = (Pixel)(1 << GetVisual().GetDepth()) - 1;
     691           0 :             break;
     692             :     }
     693           0 :     nPenColor_  = GetColormap().GetColor( nPenPixel_ );
     694           0 :     bPenGC_     = sal_False;
     695           0 : }
     696             : 
     697             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     698           0 : void X11SalGraphics::SetROPFillColor( SalROPColor nROPColor )
     699             : {
     700           0 :     switch( nROPColor )
     701             :     {
     702             :         case SAL_ROP_0 : // 0
     703           0 :             nBrushPixel_ = (Pixel)0;
     704           0 :             break;
     705             :         case SAL_ROP_1 : // 1
     706           0 :             nBrushPixel_ = (Pixel)(1 << GetVisual().GetDepth()) - 1;
     707           0 :             break;
     708             :         case SAL_ROP_INVERT : // 2
     709           0 :             nBrushPixel_ = (Pixel)(1 << GetVisual().GetDepth()) - 1;
     710           0 :             break;
     711             :     }
     712           0 :     bDitherBrush_   = sal_False;
     713           0 :     nBrushColor_    = GetColormap().GetColor( nBrushPixel_ );
     714           0 :     bBrushGC_       = sal_False;
     715           0 : }
     716             : 
     717             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     718           0 : void X11SalGraphics::SetXORMode( bool bSet, bool )
     719             : {
     720           0 :     if( !bXORMode_ == bSet )
     721             :     {
     722           0 :         bXORMode_   = bSet;
     723           0 :         bPenGC_     = sal_False;
     724           0 :         bFontGC_    = sal_False;
     725           0 :         bBrushGC_   = sal_False;
     726           0 :         bMonoGC_        = sal_False;
     727           0 :         bCopyGC_        = sal_False;
     728           0 :         bInvertGC_  = sal_False;
     729           0 :         bInvert50GC_    = sal_False;
     730           0 :         bStippleGC_ = sal_False;
     731           0 :         bTrackingGC_    = sal_False;
     732             :     }
     733           0 : }
     734             : 
     735             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     736           0 : void X11SalGraphics::drawPixel( long nX, long nY )
     737             : {
     738           0 :     if( nPenColor_ !=  SALCOLOR_NONE )
     739           0 :         XDrawPoint( GetXDisplay(), GetDrawable(), SelectPen(), nX, nY );
     740           0 : }
     741             : 
     742           0 : void X11SalGraphics::drawPixel( long nX, long nY, SalColor nSalColor )
     743             : {
     744           0 :     if( nSalColor != SALCOLOR_NONE )
     745             :     {
     746           0 :         Display *pDisplay = GetXDisplay();
     747             : 
     748           0 :         if( (nPenColor_ == SALCOLOR_NONE) && !bPenGC_ )
     749             :         {
     750           0 :             SetLineColor( nSalColor );
     751           0 :             XDrawPoint( pDisplay, GetDrawable(), SelectPen(), nX, nY );
     752           0 :             nPenColor_ = SALCOLOR_NONE;
     753           0 :             bPenGC_ = False;
     754             :         }
     755             :         else
     756             :         {
     757           0 :             GC pGC = SelectPen();
     758             : 
     759           0 :             if( nSalColor != nPenColor_ )
     760           0 :                 XSetForeground( pDisplay, pGC, GetPixel( nSalColor ) );
     761             : 
     762           0 :             XDrawPoint( pDisplay, GetDrawable(), pGC, nX, nY );
     763             : 
     764           0 :             if( nSalColor != nPenColor_ )
     765           0 :                 XSetForeground( pDisplay, pGC, nPenPixel_ );
     766             :         }
     767             :     }
     768           0 : }
     769             : 
     770             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     771           0 : void X11SalGraphics::drawLine( long nX1, long nY1, long nX2, long nY2 )
     772             : {
     773           0 :     if( nPenColor_ != SALCOLOR_NONE )
     774             :     {
     775           0 :         if ( GetDisplay()->GetProperties() & PROPERTY_BUG_DrawLine )
     776             :         {
     777           0 :             GC aGC = SelectPen();
     778           0 :             XDrawPoint (GetXDisplay(), GetDrawable(), aGC, (int)nX1, (int)nY1);
     779           0 :             XDrawPoint (GetXDisplay(), GetDrawable(), aGC, (int)nX2, (int)nY2);
     780           0 :             XDrawLine  (GetXDisplay(), GetDrawable(), aGC, nX1, nY1, nX2, nY2 );
     781             :         }
     782             :         else
     783             :             XDrawLine( GetXDisplay(), GetDrawable(),SelectPen(),
     784           0 :                        nX1, nY1, nX2, nY2 );
     785             :     }
     786           0 : }
     787             : 
     788             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     789           0 : void X11SalGraphics::drawRect( long nX, long nY, long nDX, long nDY )
     790             : {
     791           0 :     if( nBrushColor_ != SALCOLOR_NONE )
     792             :     {
     793             :         XFillRectangle( GetXDisplay(),
     794             :                         GetDrawable(),
     795             :                         SelectBrush(),
     796           0 :                         nX, nY, nDX, nDY );
     797             :     }
     798             :     // Beschreibung DrawRect verkehrt, deshalb -1
     799           0 :     if( nPenColor_ != SALCOLOR_NONE )
     800             :         XDrawRectangle( GetXDisplay(),
     801             :                         GetDrawable(),
     802             :                         SelectPen(),
     803           0 :                         nX, nY, nDX-1, nDY-1 );
     804           0 : }
     805             : 
     806             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     807           0 : void X11SalGraphics::drawPolyLine( sal_uLong nPoints, const SalPoint *pPtAry )
     808             : {
     809           0 :     drawPolyLine( nPoints, pPtAry, false );
     810           0 : }
     811             : 
     812             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     813           0 : void X11SalGraphics::drawPolyLine( sal_uLong nPoints, const SalPoint *pPtAry, bool bClose )
     814             : {
     815           0 :     if( nPenColor_ != SALCOLOR_NONE )
     816             :     {
     817           0 :         SalPolyLine Points( nPoints, pPtAry );
     818             : 
     819           0 :         DrawLines( nPoints, Points, SelectPen(), bClose );
     820             :     }
     821           0 : }
     822             : 
     823             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     824           0 : void X11SalGraphics::drawPolygon( sal_uLong nPoints, const SalPoint* pPtAry )
     825             : {
     826           0 :     if( nPoints == 0 )
     827             :         return;
     828             : 
     829           0 :     if( nPoints < 3 )
     830             :     {
     831           0 :         if( !bXORMode_ )
     832             :         {
     833           0 :             if( 1 == nPoints  )
     834           0 :                 drawPixel( pPtAry[0].mnX, pPtAry[0].mnY );
     835             :             else
     836             :                 drawLine( pPtAry[0].mnX, pPtAry[0].mnY,
     837           0 :                           pPtAry[1].mnX, pPtAry[1].mnY );
     838             :         }
     839             :         return;
     840             :     }
     841             : 
     842           0 :     SalPolyLine Points( nPoints, pPtAry );
     843             : 
     844           0 :     nPoints++;
     845             : 
     846             :     /* WORKAROUND: some Xservers (Xorg, VIA chipset in this case)
     847             :      * do not draw the visible part of a polygon
     848             :      * if it overlaps to the left of screen 0,y.
     849             :      * This happens to be the case in the gradient drawn in the
     850             :      * menubar background. workaround for the special case of
     851             :      * of a rectangle overlapping to the left.
     852             :      */
     853           0 :     if( nPoints == 5 &&
     854           0 :     Points[ 0 ].x == Points[ 1 ].x &&
     855           0 :         Points[ 1 ].y == Points[ 2 ].y &&
     856           0 :         Points[ 2 ].x == Points[ 3 ].x &&
     857           0 :         Points[ 0 ].x == Points[ 4 ].x && Points[ 0 ].y == Points[ 4 ].y
     858             :        )
     859             :     {
     860           0 :         bool bLeft = false;
     861           0 :         bool bRight = false;
     862           0 :         for(unsigned int i = 0; i < nPoints; i++ )
     863             :     {
     864           0 :             if( Points[i].x < 0 )
     865           0 :                 bLeft = true;
     866             :             else
     867           0 :                 bRight= true;
     868             :     }
     869           0 :     if( bLeft && ! bRight )
     870             :         return;
     871           0 :     if( bLeft && bRight )
     872             :         {
     873           0 :             for( unsigned int i = 0; i < nPoints; i++ )
     874           0 :                 if( Points[i].x < 0 )
     875           0 :                     Points[i].x = 0;
     876             :         }
     877             :     }
     878             : 
     879           0 :     if( nBrushColor_ != SALCOLOR_NONE )
     880             :         XFillPolygon( GetXDisplay(),
     881             :                       GetDrawable(),
     882             :                       SelectBrush(),
     883           0 :                       &Points[0], nPoints,
     884           0 :                       Complex, CoordModeOrigin );
     885             : 
     886           0 :     if( nPenColor_ != SALCOLOR_NONE )
     887           0 :         DrawLines( nPoints, Points, SelectPen(), true );
     888             : }
     889             : 
     890             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     891           0 : void X11SalGraphics::drawPolyPolygon( sal_uInt32        nPoly,
     892             :                                    const sal_uInt32    *pPoints,
     893             :                                    PCONSTSALPOINT  *pPtAry )
     894             : {
     895           0 :     if( nBrushColor_ != SALCOLOR_NONE )
     896             :     {
     897             :         sal_uInt32      i, n;
     898           0 :         XLIB_Region pXRegA  = NULL;
     899             : 
     900           0 :         for( i = 0; i < nPoly; i++ ) {
     901           0 :             n = pPoints[i];
     902           0 :             SalPolyLine Points( n, pPtAry[i] );
     903           0 :             if( n > 2 )
     904             :             {
     905           0 :                 XLIB_Region pXRegB = XPolygonRegion( &Points[0], n+1, WindingRule );
     906           0 :                 if( !pXRegA )
     907           0 :                     pXRegA = pXRegB;
     908             :                 else
     909             :                 {
     910           0 :                     XXorRegion( pXRegA, pXRegB, pXRegA );
     911           0 :                     XDestroyRegion( pXRegB );
     912             :                 }
     913             :             }
     914           0 :         }
     915             : 
     916           0 :         if( pXRegA )
     917             :         {
     918             :             XRectangle aXRect;
     919           0 :             XClipBox( pXRegA, &aXRect );
     920             : 
     921           0 :             GC pGC = SelectBrush();
     922           0 :             SetClipRegion( pGC, pXRegA ); // ??? doppelt
     923           0 :             XDestroyRegion( pXRegA );
     924           0 :             bBrushGC_ = sal_False;
     925             : 
     926             :             XFillRectangle( GetXDisplay(),
     927             :                             GetDrawable(),
     928             :                             pGC,
     929           0 :                             aXRect.x, aXRect.y, aXRect.width, aXRect.height );
     930             :         }
     931             :    }
     932             : 
     933           0 :    if( nPenColor_ != SALCOLOR_NONE )
     934           0 :        for( sal_uInt32 i = 0; i < nPoly; i++ )
     935           0 :            drawPolyLine( pPoints[i], pPtAry[i], true );
     936           0 : }
     937             : 
     938             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     939             : 
     940           0 : sal_Bool X11SalGraphics::drawPolyLineBezier( sal_uLong, const SalPoint*, const BYTE* )
     941             : {
     942           0 :     return sal_False;
     943             : }
     944             : 
     945             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     946             : 
     947           0 : sal_Bool X11SalGraphics::drawPolygonBezier( sal_uLong, const SalPoint*, const BYTE* )
     948             : {
     949           0 :     return sal_False;
     950             : }
     951             : 
     952             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     953             : 
     954           0 : sal_Bool X11SalGraphics::drawPolyPolygonBezier( sal_uInt32, const sal_uInt32*,
     955             :                                                 const SalPoint* const*, const BYTE* const* )
     956             : {
     957           0 :     return sal_False;
     958             : }
     959             : 
     960             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     961             : 
     962           0 : void X11SalGraphics::invert( sal_uLong nPoints,
     963             :                              const SalPoint* pPtAry,
     964             :                              SalInvert nFlags )
     965             : {
     966           0 :     SalPolyLine Points ( nPoints, pPtAry );
     967             : 
     968             :     GC pGC;
     969           0 :     if( SAL_INVERT_50 & nFlags )
     970           0 :         pGC = GetInvert50GC();
     971             :     else
     972           0 :         if ( SAL_INVERT_TRACKFRAME & nFlags )
     973           0 :             pGC = GetTrackingGC();
     974             :         else
     975           0 :             pGC = GetInvertGC();
     976             : 
     977           0 :     if( SAL_INVERT_TRACKFRAME & nFlags )
     978           0 :         DrawLines ( nPoints, Points, pGC, true );
     979             :     else
     980             :         XFillPolygon( GetXDisplay(),
     981             :                       GetDrawable(),
     982             :                       pGC,
     983           0 :                       &Points[0], nPoints,
     984           0 :                       Complex, CoordModeOrigin );
     985           0 : }
     986             : 
     987             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     988             : 
     989           0 : BOOL X11SalGraphics::drawEPS( long,long,long,long,void*,sal_uLong )
     990             : {
     991           0 :     return sal_False;
     992             : }
     993             : 
     994             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
     995             : 
     996           0 : XID X11SalGraphics::GetXRenderPicture()
     997             : {
     998           0 :     XRenderPeer& rRenderPeer = XRenderPeer::GetInstance();
     999             : 
    1000           0 :     if( !m_aXRenderPicture )
    1001             :     {
    1002             :         // check xrender support for matching visual
    1003           0 :         XRenderPictFormat* pXRenderFormat = GetXRenderFormat();
    1004           0 :         if( !pXRenderFormat )
    1005           0 :             return 0;
    1006             :         // get the matching xrender target for drawable
    1007           0 :         m_aXRenderPicture = rRenderPeer.CreatePicture( hDrawable_, pXRenderFormat, 0, NULL );
    1008             :     }
    1009             : 
    1010             :     {
    1011             :         // reset clip region
    1012             :         // TODO: avoid clip reset if already done
    1013             :         XRenderPictureAttributes aAttr;
    1014           0 :         aAttr.clip_mask = None;
    1015           0 :         rRenderPeer.ChangePicture( m_aXRenderPicture, CPClipMask, &aAttr );
    1016             :     }
    1017             : 
    1018           0 :     return m_aXRenderPicture;
    1019             : }
    1020             : 
    1021           0 : XRenderPictFormat* X11SalGraphics::GetXRenderFormat() const
    1022             : {
    1023           0 :     if( m_pXRenderFormat == NULL )
    1024           0 :         m_pXRenderFormat = XRenderPeer::GetInstance().FindVisualFormat( GetVisual().visual );
    1025           0 :     return m_pXRenderFormat;
    1026             : }
    1027             : 
    1028             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
    1029             : 
    1030           0 : SystemGraphicsData X11SalGraphics::GetGraphicsData() const
    1031             : {
    1032           0 :     SystemGraphicsData aRes;
    1033             : 
    1034           0 :     aRes.nSize = sizeof(aRes);
    1035           0 :     aRes.pDisplay  = GetXDisplay();
    1036           0 :     aRes.hDrawable = hDrawable_;
    1037           0 :     aRes.pVisual   = GetVisual().visual;
    1038           0 :     aRes.nScreen   = m_nXScreen.getXScreen();
    1039           0 :     aRes.nDepth    = GetBitCount();
    1040           0 :     aRes.aColormap = GetColormap().GetXColormap();
    1041           0 :     aRes.pXRenderFormat = m_pXRenderFormat;
    1042           0 :     return aRes;
    1043             : }
    1044             : 
    1045             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
    1046             : 
    1047             : // draw a poly-polygon
    1048           0 : bool X11SalGraphics::drawPolyPolygon( const ::basegfx::B2DPolyPolygon& rOrigPolyPoly, double fTransparency )
    1049             : {
    1050             :     // nothing to do for empty polypolygons
    1051           0 :     const int nOrigPolyCount = rOrigPolyPoly.count();
    1052           0 :     if( nOrigPolyCount <= 0 )
    1053           0 :         return sal_True;
    1054             : 
    1055             :     // nothing to do if everything is transparent
    1056           0 :     if( (nBrushColor_ == SALCOLOR_NONE)
    1057             :     &&  (nPenColor_ == SALCOLOR_NONE) )
    1058           0 :         return sal_True;
    1059             : 
    1060             :     // cannot handle pencolor!=brushcolor yet
    1061           0 :     if( (nPenColor_ != SALCOLOR_NONE)
    1062             :     &&  (nPenColor_ != nBrushColor_) )
    1063           0 :         return sal_False;
    1064             : 
    1065             :     // TODO: remove the env-variable when no longer needed
    1066           0 :     static const char* pRenderEnv = getenv( "SAL_DISABLE_RENDER_POLY" );
    1067           0 :     if( pRenderEnv )
    1068           0 :         return sal_False;
    1069             : 
    1070             :     // snap to raster if requested
    1071           0 :     basegfx::B2DPolyPolygon aPolyPoly = rOrigPolyPoly;
    1072           0 :     const bool bSnapToRaster = !getAntiAliasB2DDraw();
    1073           0 :     if( bSnapToRaster )
    1074           0 :         aPolyPoly = basegfx::tools::snapPointsOfHorizontalOrVerticalEdges( aPolyPoly );
    1075             : 
    1076             :     // don't bother with polygons outside of visible area
    1077           0 :     const basegfx::B2DRange aViewRange( 0, 0, GetGraphicsWidth(), GetGraphicsHeight() );
    1078           0 :     aPolyPoly = basegfx::tools::clipPolyPolygonOnRange( aPolyPoly, aViewRange, true, false );
    1079           0 :     if( !aPolyPoly.count() )
    1080           0 :         return true;
    1081             : 
    1082             :     // tesselate the polypolygon into trapezoids
    1083           0 :     basegfx::B2DTrapezoidVector aB2DTrapVector;
    1084           0 :     basegfx::tools::trapezoidSubdivide( aB2DTrapVector, aPolyPoly );
    1085           0 :     const int nTrapCount = aB2DTrapVector.size();
    1086           0 :     if( !nTrapCount )
    1087           0 :         return true;
    1088           0 :     const bool bDrawn = drawFilledTrapezoids( &aB2DTrapVector[0], nTrapCount, fTransparency );
    1089           0 :     return bDrawn;
    1090             : }
    1091             : 
    1092             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
    1093             : 
    1094           0 : bool X11SalGraphics::drawFilledTrapezoids( const ::basegfx::B2DTrapezoid* pB2DTraps, int nTrapCount, double fTransparency )
    1095             : {
    1096           0 :     if( nTrapCount <= 0 )
    1097           0 :         return true;
    1098             : 
    1099           0 :     Picture aDstPic = GetXRenderPicture();
    1100             :     // check xrender support for this drawable
    1101           0 :     if( !aDstPic )
    1102           0 :         return false;
    1103             : 
    1104             :      // convert the B2DTrapezoids into XRender-Trapezoids
    1105             :     typedef std::vector<XTrapezoid> TrapezoidVector;
    1106           0 :     TrapezoidVector aTrapVector( nTrapCount );
    1107           0 :     const basegfx::B2DTrapezoid* pB2DTrap = pB2DTraps;
    1108           0 :     for( int i = 0; i < nTrapCount; ++pB2DTrap, ++i )
    1109             :     {
    1110           0 :         XTrapezoid& rTrap = aTrapVector[ i ] ;
    1111             : 
    1112             :          // set y-coordinates
    1113           0 :         const double fY1 = pB2DTrap->getTopY();
    1114           0 :         rTrap.left.p1.y = rTrap.right.p1.y = rTrap.top = XDoubleToFixed( fY1 );
    1115           0 :         const double fY2 = pB2DTrap->getBottomY();
    1116           0 :         rTrap.left.p2.y = rTrap.right.p2.y = rTrap.bottom = XDoubleToFixed( fY2 );
    1117             : 
    1118             :          // set x-coordinates
    1119           0 :         const double fXL1 = pB2DTrap->getTopXLeft();
    1120           0 :         rTrap.left.p1.x = XDoubleToFixed( fXL1 );
    1121           0 :         const double fXR1 = pB2DTrap->getTopXRight();
    1122           0 :         rTrap.right.p1.x = XDoubleToFixed( fXR1 );
    1123           0 :         const double fXL2 = pB2DTrap->getBottomXLeft();
    1124           0 :         rTrap.left.p2.x = XDoubleToFixed( fXL2 );
    1125           0 :         const double fXR2 = pB2DTrap->getBottomXRight();
    1126           0 :         rTrap.right.p2.x = XDoubleToFixed( fXR2 );
    1127             :     }
    1128             : 
    1129             :     // get xrender Picture for polygon foreground
    1130             :     // TODO: cache it like the target picture which uses GetXRenderPicture()
    1131           0 :     XRenderPeer& rRenderPeer = XRenderPeer::GetInstance();
    1132           0 :     SalDisplay::RenderEntry& rEntry = GetDisplay()->GetRenderEntries( m_nXScreen )[ 32 ];
    1133           0 :     if( !rEntry.m_aPicture )
    1134             :     {
    1135           0 :         Display* pXDisplay = GetXDisplay();
    1136             : 
    1137           0 :         rEntry.m_aPixmap = limitXCreatePixmap( pXDisplay, hDrawable_, 1, 1, 32 );
    1138             :         XRenderPictureAttributes aAttr;
    1139           0 :         aAttr.repeat = true;
    1140             : 
    1141           0 :         XRenderPictFormat* pXRPF = rRenderPeer.FindStandardFormat( PictStandardARGB32 );
    1142           0 :         rEntry.m_aPicture = rRenderPeer.CreatePicture( rEntry.m_aPixmap, pXRPF, CPRepeat, &aAttr );
    1143             :     }
    1144             : 
    1145             :     // set polygon foreground color and opacity
    1146           0 :     XRenderColor aRenderColor = GetXRenderColor( nBrushColor_ , fTransparency );
    1147           0 :     rRenderPeer.FillRectangle( PictOpSrc, rEntry.m_aPicture, &aRenderColor, 0, 0, 1, 1 );
    1148             : 
    1149             :     // set clipping
    1150             :     // TODO: move into GetXRenderPicture?
    1151           0 :     if( mpClipRegion && !XEmptyRegion( mpClipRegion ) )
    1152           0 :         rRenderPeer.SetPictureClipRegion( aDstPic, mpClipRegion );
    1153             : 
    1154             :     // render the trapezoids
    1155           0 :     const XRenderPictFormat* pMaskFormat = rRenderPeer.GetStandardFormatA8();
    1156             :     rRenderPeer.CompositeTrapezoids( PictOpOver,
    1157           0 :         rEntry.m_aPicture, aDstPic, pMaskFormat, 0, 0, &aTrapVector[0], aTrapVector.size() );
    1158             : 
    1159           0 :     return true;
    1160             : }
    1161             : 
    1162             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
    1163             : 
    1164           0 : bool X11SalGraphics::drawPolyLine(
    1165             :     const ::basegfx::B2DPolygon& rPolygon,
    1166             :     double fTransparency,
    1167             :     const ::basegfx::B2DVector& rLineWidth,
    1168             :     basegfx::B2DLineJoin eLineJoin,
    1169             :     com::sun::star::drawing::LineCap eLineCap)
    1170             : {
    1171           0 :     const bool bIsHairline = (rLineWidth.getX() == rLineWidth.getY()) && (rLineWidth.getX() <= 1.2);
    1172             : 
    1173             :     // #i101491#
    1174           0 :     if( !bIsHairline && (rPolygon.count() > 1000) )
    1175             :     {
    1176             :         // the used basegfx::tools::createAreaGeometry is simply too
    1177             :         // expensive with very big polygons; fallback to caller (who
    1178             :         // should use ImplLineConverter normally)
    1179             :         // AW: ImplLineConverter had to be removed since it does not even
    1180             :         // know LineJoins, so the fallback will now prepare the line geometry
    1181             :         // the same way.
    1182           0 :         return false;
    1183             :     }
    1184             : 
    1185             :     // temporarily adjust brush color to pen color
    1186             :     // since the line is drawn as an area-polygon
    1187           0 :     const SalColor aKeepBrushColor = nBrushColor_;
    1188           0 :     nBrushColor_ = nPenColor_;
    1189             : 
    1190             :     // #i11575#desc5#b adjust B2D tesselation result to raster positions
    1191           0 :     basegfx::B2DPolygon aPolygon = rPolygon;
    1192           0 :     const double fHalfWidth = 0.5 * rLineWidth.getX();
    1193           0 :     aPolygon.transform( basegfx::tools::createTranslateB2DHomMatrix(+fHalfWidth,+fHalfWidth) );
    1194             : 
    1195             :     // shortcut for hairline drawing to improve performance
    1196           0 :     bool bDrawnOk = true;
    1197           0 :     if( bIsHairline )
    1198             :     {
    1199             :         // hairlines can benefit from a simplified tesselation
    1200             :         // e.g. for hairlines the linejoin style can be ignored
    1201           0 :         basegfx::B2DTrapezoidVector aB2DTrapVector;
    1202           0 :         basegfx::tools::createLineTrapezoidFromB2DPolygon( aB2DTrapVector, aPolygon, rLineWidth.getX() );
    1203             : 
    1204             :         // draw tesselation result
    1205           0 :         const int nTrapCount = aB2DTrapVector.size();
    1206           0 :         if( nTrapCount > 0 )
    1207           0 :             bDrawnOk = drawFilledTrapezoids( &aB2DTrapVector[0], nTrapCount, fTransparency );
    1208             : 
    1209             :         // restore the original brush GC
    1210           0 :         nBrushColor_ = aKeepBrushColor;
    1211           0 :         return bDrawnOk;
    1212             :     }
    1213             : 
    1214             :     // get the area polygon for the line polygon
    1215           0 :     if( (rLineWidth.getX() != rLineWidth.getY())
    1216           0 :     && !basegfx::fTools::equalZero( rLineWidth.getY() ) )
    1217             :     {
    1218             :         // prepare for createAreaGeometry() with anisotropic linewidth
    1219           0 :         aPolygon.transform( basegfx::tools::createScaleB2DHomMatrix(1.0, rLineWidth.getX() / rLineWidth.getY()));
    1220             :     }
    1221             : 
    1222             :     // create the area-polygon for the line
    1223           0 :     const basegfx::B2DPolyPolygon aAreaPolyPoly( basegfx::tools::createAreaGeometry(aPolygon, fHalfWidth, eLineJoin, eLineCap) );
    1224             : 
    1225           0 :     if( (rLineWidth.getX() != rLineWidth.getY())
    1226           0 :     && !basegfx::fTools::equalZero( rLineWidth.getX() ) )
    1227             :     {
    1228             :         // postprocess createAreaGeometry() for anisotropic linewidth
    1229           0 :         aPolygon.transform(basegfx::tools::createScaleB2DHomMatrix(1.0, rLineWidth.getY() / rLineWidth.getX()));
    1230             :     }
    1231             : 
    1232             :     // draw each area polypolygon component individually
    1233             :     // to emulate the polypolygon winding rule "non-zero"
    1234           0 :     const int nPolyCount = aAreaPolyPoly.count();
    1235           0 :     for( int nPolyIdx = 0; nPolyIdx < nPolyCount; ++nPolyIdx )
    1236             :     {
    1237           0 :         const ::basegfx::B2DPolyPolygon aOnePoly( aAreaPolyPoly.getB2DPolygon( nPolyIdx ) );
    1238           0 :         bDrawnOk = drawPolyPolygon( aOnePoly, fTransparency );
    1239           0 :         if( !bDrawnOk )
    1240             :             break;
    1241           0 :     }
    1242             : 
    1243             :     // restore the original brush GC
    1244           0 :     nBrushColor_ = aKeepBrushColor;
    1245           0 :     return bDrawnOk;
    1246             : }
    1247             : 
    1248             : // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
    1249             : 
    1250             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10