extensions and fixes
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/branches/WX_2_8_BRANCH@49266 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
@@ -292,6 +292,13 @@ public:
|
|||||||
|
|
||||||
virtual void StrokePath( const wxGraphicsPath& p );
|
virtual void StrokePath( const wxGraphicsPath& p );
|
||||||
virtual void FillPath( const wxGraphicsPath& p , int fillStyle = wxODDEVEN_RULE );
|
virtual void FillPath( const wxGraphicsPath& p , int fillStyle = wxODDEVEN_RULE );
|
||||||
|
virtual void DrawPath( const wxGraphicsPath& path, int fillStyle = wxODDEVEN_RULE );
|
||||||
|
|
||||||
|
// stroke lines connecting each of the points
|
||||||
|
virtual void StrokeLines( size_t n, const wxPoint2DDouble *points);
|
||||||
|
|
||||||
|
// draws a polygon
|
||||||
|
virtual void DrawLines( size_t n, const wxPoint2DDouble *points, int fillStyle = wxODDEVEN_RULE );
|
||||||
|
|
||||||
virtual void Translate( wxDouble dx , wxDouble dy );
|
virtual void Translate( wxDouble dx , wxDouble dy );
|
||||||
virtual void Scale( wxDouble xScale , wxDouble yScale );
|
virtual void Scale( wxDouble xScale , wxDouble yScale );
|
||||||
@@ -329,6 +336,26 @@ private:
|
|||||||
DECLARE_DYNAMIC_CLASS_NO_COPY(wxGDIPlusContext)
|
DECLARE_DYNAMIC_CLASS_NO_COPY(wxGDIPlusContext)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class WXDLLIMPEXP_CORE wxGDIPlusMeasuringContext : public wxGDIPlusContext
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
wxGDIPlusMeasuringContext( wxGraphicsRenderer* renderer ) : wxGDIPlusContext( renderer , m_hdc = GetDC(NULL) )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
wxGDIPlusMeasuringContext()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~wxGDIPlusMeasuringContext()
|
||||||
|
{
|
||||||
|
ReleaseDC( NULL, m_hdc );
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
HDC m_hdc ;
|
||||||
|
DECLARE_DYNAMIC_CLASS_NO_COPY(wxGDIPlusMeasuringContext)
|
||||||
|
} ;
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// wxGDIPlusPen implementation
|
// wxGDIPlusPen implementation
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
@@ -858,6 +885,27 @@ void * wxGDIPlusMatrixData::GetNativeMatrix() const
|
|||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
IMPLEMENT_DYNAMIC_CLASS(wxGDIPlusContext,wxGraphicsContext)
|
IMPLEMENT_DYNAMIC_CLASS(wxGDIPlusContext,wxGraphicsContext)
|
||||||
|
IMPLEMENT_DYNAMIC_CLASS(wxGDIPlusMeasuringContext,wxGDIPlusContext)
|
||||||
|
|
||||||
|
class wxGDIPlusOffsetHelper
|
||||||
|
{
|
||||||
|
public :
|
||||||
|
wxGDIPlusOffsetHelper( Graphics* gr , bool offset )
|
||||||
|
{
|
||||||
|
m_gr = gr;
|
||||||
|
m_offset = offset;
|
||||||
|
if ( m_offset )
|
||||||
|
m_gr->TranslateTransform( 0.5, 0.5 );
|
||||||
|
}
|
||||||
|
~wxGDIPlusOffsetHelper( )
|
||||||
|
{
|
||||||
|
if ( m_offset )
|
||||||
|
m_gr->TranslateTransform( -0.5, -0.5 );
|
||||||
|
}
|
||||||
|
public :
|
||||||
|
Graphics* m_gr;
|
||||||
|
bool m_offset;
|
||||||
|
} ;
|
||||||
|
|
||||||
wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer* renderer, HDC hdc )
|
wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer* renderer, HDC hdc )
|
||||||
: wxGraphicsContext(renderer)
|
: wxGraphicsContext(renderer)
|
||||||
@@ -897,6 +945,8 @@ void wxGDIPlusContext::Init()
|
|||||||
|
|
||||||
void wxGDIPlusContext::SetDefaults()
|
void wxGDIPlusContext::SetDefaults()
|
||||||
{
|
{
|
||||||
|
m_context->SetTextRenderingHint(TextRenderingHintSystemDefault);
|
||||||
|
m_context->SetPixelOffsetMode(PixelOffsetModeHalf);
|
||||||
m_context->SetSmoothingMode(SmoothingModeHighQuality);
|
m_context->SetSmoothingMode(SmoothingModeHighQuality);
|
||||||
m_state1 = m_context->Save();
|
m_state1 = m_context->Save();
|
||||||
m_state2 = m_context->Save();
|
m_state2 = m_context->Save();
|
||||||
@@ -929,10 +979,45 @@ void wxGDIPlusContext::ResetClip()
|
|||||||
m_context->ResetClip();
|
m_context->ResetClip();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void wxGDIPlusContext::StrokeLines( size_t n, const wxPoint2DDouble *points)
|
||||||
|
{
|
||||||
|
if ( !m_pen.IsNull() )
|
||||||
|
{
|
||||||
|
wxGDIPlusOffsetHelper helper( m_context , ShouldOffset() );
|
||||||
|
Point *cpoints = new Point[n];
|
||||||
|
for (size_t i = 0; i < n; i++)
|
||||||
|
{
|
||||||
|
cpoints[i].X = (int)(points[i].m_x );
|
||||||
|
cpoints[i].Y = (int)(points[i].m_y );
|
||||||
|
|
||||||
|
} // for (size_t i = 0; i < n; i++)
|
||||||
|
m_context->DrawLines( ((wxGDIPlusPenData*)m_pen.GetGraphicsData())->GetGDIPlusPen() , cpoints , n ) ;
|
||||||
|
delete[] cpoints;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void wxGDIPlusContext::DrawLines( size_t n, const wxPoint2DDouble *points, int WXUNUSED(fillStyle) )
|
||||||
|
{
|
||||||
|
wxGDIPlusOffsetHelper helper( m_context , ShouldOffset() );
|
||||||
|
Point *cpoints = new Point[n];
|
||||||
|
for (size_t i = 0; i < n; i++)
|
||||||
|
{
|
||||||
|
cpoints[i].X = (int)(points[i].m_x );
|
||||||
|
cpoints[i].Y = (int)(points[i].m_y );
|
||||||
|
|
||||||
|
} // for (int i = 0; i < n; i++)
|
||||||
|
if ( !m_brush.IsNull() )
|
||||||
|
m_context->FillPolygon( ((wxGDIPlusBrushData*)m_brush.GetRefData())->GetGDIPlusBrush() , cpoints , n ) ;
|
||||||
|
if ( !m_pen.IsNull() )
|
||||||
|
m_context->DrawLines( ((wxGDIPlusPenData*)m_pen.GetGraphicsData())->GetGDIPlusPen() , cpoints , n ) ;
|
||||||
|
delete[] cpoints;
|
||||||
|
}
|
||||||
|
|
||||||
void wxGDIPlusContext::StrokePath( const wxGraphicsPath& path )
|
void wxGDIPlusContext::StrokePath( const wxGraphicsPath& path )
|
||||||
{
|
{
|
||||||
if ( !m_pen.IsNull() )
|
if ( !m_pen.IsNull() )
|
||||||
{
|
{
|
||||||
|
wxGDIPlusOffsetHelper helper( m_context , ShouldOffset() );
|
||||||
m_context->DrawPath( ((wxGDIPlusPenData*)m_pen.GetGraphicsData())->GetGDIPlusPen() , (GraphicsPath*) path.GetNativePath() );
|
m_context->DrawPath( ((wxGDIPlusPenData*)m_pen.GetGraphicsData())->GetGDIPlusPen() , (GraphicsPath*) path.GetNativePath() );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -947,6 +1032,15 @@ void wxGDIPlusContext::FillPath( const wxGraphicsPath& path , int fillStyle )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void wxGDIPlusContext::DrawPath( const wxGraphicsPath& path, int fillStyle )
|
||||||
|
{
|
||||||
|
{
|
||||||
|
wxGDIPlusOffsetHelper helper( m_context , ShouldOffset() );
|
||||||
|
FillPath( path , fillStyle );
|
||||||
|
}
|
||||||
|
StrokePath( path );
|
||||||
|
}
|
||||||
|
|
||||||
void wxGDIPlusContext::Rotate( wxDouble angle )
|
void wxGDIPlusContext::Rotate( wxDouble angle )
|
||||||
{
|
{
|
||||||
m_context->RotateTransform( RadToDeg(angle) );
|
m_context->RotateTransform( RadToDeg(angle) );
|
||||||
@@ -1042,7 +1136,7 @@ void wxGDIPlusContext::DrawBitmap( const wxBitmap &bmp, wxDouble x, wxDouble y,
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
image = Bitmap::FromHBITMAP((HBITMAP)bmp.GetHBITMAP(),(HPALETTE)bmp.GetPalette()->GetHPALETTE());
|
image = Bitmap::FromHBITMAP((HBITMAP)bmp.GetHBITMAP(),(HPALETTE)bmp.GetPalette()->GetHPALETTE());
|
||||||
if ( GetPixelFormatSize(image->GetPixelFormat()) == 32 )
|
if ( bmp.HasAlpha() && GetPixelFormatSize(image->GetPixelFormat()) == 32 )
|
||||||
{
|
{
|
||||||
size_t width = image->GetWidth();
|
size_t width = image->GetWidth();
|
||||||
size_t height = image->GetHeight();
|
size_t height = image->GetHeight();
|
||||||
@@ -1068,18 +1162,20 @@ void wxGDIPlusContext::DrawBitmap( const wxBitmap &bmp, wxDouble x, wxDouble y,
|
|||||||
|
|
||||||
void wxGDIPlusContext::DrawIcon( const wxIcon &icon, wxDouble x, wxDouble y, wxDouble w, wxDouble h )
|
void wxGDIPlusContext::DrawIcon( const wxIcon &icon, wxDouble x, wxDouble y, wxDouble w, wxDouble h )
|
||||||
{
|
{
|
||||||
|
// the built-in conversion fails when there is alpha in the HICON (eg XP style icons), we can only
|
||||||
|
// find out by looking at the bitmap data whether there really was alpha in it
|
||||||
HICON hIcon = (HICON)icon.GetHICON();
|
HICON hIcon = (HICON)icon.GetHICON();
|
||||||
ICONINFO iconInfo ;
|
ICONINFO iconInfo ;
|
||||||
// IconInfo creates the bitmaps for color and mask, we must dispose of them after use
|
// IconInfo creates the bitmaps for color and mask, we must dispose of them after use
|
||||||
if (!GetIconInfo(hIcon,&iconInfo))
|
if (!GetIconInfo(hIcon,&iconInfo))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
BITMAP iconBmpData ;
|
|
||||||
GetObject(iconInfo.hbmColor,sizeof(BITMAP),&iconBmpData);
|
|
||||||
Bitmap interim(iconInfo.hbmColor,NULL);
|
Bitmap interim(iconInfo.hbmColor,NULL);
|
||||||
|
|
||||||
Bitmap* image = NULL ;
|
Bitmap* image = NULL ;
|
||||||
|
|
||||||
|
// if it's not 32 bit, it doesn't have an alpha channel, note that since the conversion doesn't
|
||||||
|
// work correctly, asking IsAlphaPixelFormat at this point fails as well
|
||||||
if( GetPixelFormatSize(interim.GetPixelFormat())!= 32 )
|
if( GetPixelFormatSize(interim.GetPixelFormat())!= 32 )
|
||||||
{
|
{
|
||||||
image = Bitmap::FromHICON(hIcon);
|
image = Bitmap::FromHICON(hIcon);
|
||||||
@@ -1093,8 +1189,28 @@ void wxGDIPlusContext::DrawIcon( const wxIcon &icon, wxDouble x, wxDouble y, wxD
|
|||||||
|
|
||||||
interim.LockBits(&bounds, ImageLockModeRead,
|
interim.LockBits(&bounds, ImageLockModeRead,
|
||||||
interim.GetPixelFormat(),&data);
|
interim.GetPixelFormat(),&data);
|
||||||
image = new Bitmap(data.Width, data.Height, data.Stride,
|
|
||||||
PixelFormat32bppARGB , (BYTE*) data.Scan0);
|
bool hasAlpha = false;
|
||||||
|
for ( size_t y = 0 ; y < height && !hasAlpha ; ++y)
|
||||||
|
{
|
||||||
|
for( size_t x = 0 ; x < width && !hasAlpha; ++x)
|
||||||
|
{
|
||||||
|
ARGB *dest = (ARGB*)((BYTE*)data.Scan0 + data.Stride*y + x*4);
|
||||||
|
if ( ( *dest & Color::AlphaMask ) != 0 )
|
||||||
|
hasAlpha = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( hasAlpha )
|
||||||
|
{
|
||||||
|
image = new Bitmap(data.Width, data.Height, data.Stride,
|
||||||
|
PixelFormat32bppARGB , (BYTE*) data.Scan0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
image = Bitmap::FromHICON(hIcon);
|
||||||
|
}
|
||||||
|
|
||||||
interim.UnlockBits(&data);
|
interim.UnlockBits(&data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1112,8 +1228,7 @@ void wxGDIPlusContext::DrawText( const wxString &str, wxDouble x, wxDouble y )
|
|||||||
|
|
||||||
wxWCharBuffer s = str.wc_str( *wxConvUI );
|
wxWCharBuffer s = str.wc_str( *wxConvUI );
|
||||||
m_context->DrawString( s , -1 , ((wxGDIPlusFontData*)m_font.GetRefData())->GetGDIPlusFont() ,
|
m_context->DrawString( s , -1 , ((wxGDIPlusFontData*)m_font.GetRefData())->GetGDIPlusFont() ,
|
||||||
PointF( x , y ) , ((wxGDIPlusFontData*)m_font.GetRefData())->GetGDIPlusBrush() );
|
PointF( x , y ) , StringFormat::GenericTypographic() , ((wxGDIPlusFontData*)m_font.GetRefData())->GetGDIPlusBrush() );
|
||||||
// TODO m_backgroundMode == wxSOLID
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void wxGDIPlusContext::GetTextExtent( const wxString &str, wxDouble *width, wxDouble *height,
|
void wxGDIPlusContext::GetTextExtent( const wxString &str, wxDouble *width, wxDouble *height,
|
||||||
@@ -1135,11 +1250,11 @@ void wxGDIPlusContext::GetTextExtent( const wxString &str, wxDouble *width, wxDo
|
|||||||
f->GetSize() / ffamily.GetEmHeight(FontStyleRegular);
|
f->GetSize() / ffamily.GetEmHeight(FontStyleRegular);
|
||||||
|
|
||||||
if ( height )
|
if ( height )
|
||||||
*height = rHeight * factorY + 0.5 ;
|
*height = rHeight * factorY;
|
||||||
if ( descent )
|
if ( descent )
|
||||||
*descent = rDescent * factorY + 0.5 ;
|
*descent = rDescent * factorY;
|
||||||
if ( externalLeading )
|
if ( externalLeading )
|
||||||
*externalLeading = (rHeight - rAscent - rDescent) * factorY + 0.5 ;
|
*externalLeading = (rHeight - rAscent - rDescent) * factorY;
|
||||||
// measuring empty strings is not guaranteed, so do it by hand
|
// measuring empty strings is not guaranteed, so do it by hand
|
||||||
if ( str.IsEmpty())
|
if ( str.IsEmpty())
|
||||||
{
|
{
|
||||||
@@ -1148,18 +1263,14 @@ void wxGDIPlusContext::GetTextExtent( const wxString &str, wxDouble *width, wxDo
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// MeasureString does return a rectangle that is way too large, so it is
|
|
||||||
// not usable here
|
|
||||||
RectF layoutRect(0,0, 100000.0f, 100000.0f);
|
RectF layoutRect(0,0, 100000.0f, 100000.0f);
|
||||||
StringFormat strFormat;
|
StringFormat strFormat( StringFormat::GenericTypographic() );
|
||||||
CharacterRange strRange(0,wcslen(s));
|
strFormat.SetFormatFlags( StringFormatFlagsMeasureTrailingSpaces | strFormat.GetFormatFlags() );
|
||||||
strFormat.SetMeasurableCharacterRanges(1,&strRange);
|
|
||||||
Region region ;
|
RectF bounds ;
|
||||||
m_context->MeasureCharacterRanges(s, -1 , f,layoutRect, &strFormat,1,®ion) ;
|
m_context->MeasureString((const wchar_t *) s , wcslen(s) , f, layoutRect, &strFormat, &bounds ) ;
|
||||||
RectF bbox ;
|
|
||||||
region.GetBounds(&bbox,m_context);
|
|
||||||
if ( width )
|
if ( width )
|
||||||
*width = bbox.GetRight()-bbox.GetLeft()+0.5;
|
*width = bounds.Width;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1177,7 +1288,7 @@ void wxGDIPlusContext::GetPartialTextExtents(const wxString& text, wxArrayDouble
|
|||||||
wxASSERT_MSG(text.length() == len , wxT("GetPartialTextExtents not yet implemented for multichar situations"));
|
wxASSERT_MSG(text.length() == len , wxT("GetPartialTextExtents not yet implemented for multichar situations"));
|
||||||
|
|
||||||
RectF layoutRect(0,0, 100000.0f, 100000.0f);
|
RectF layoutRect(0,0, 100000.0f, 100000.0f);
|
||||||
StringFormat strFormat;
|
StringFormat strFormat( StringFormat::GenericTypographic() );
|
||||||
|
|
||||||
CharacterRange* ranges = new CharacterRange[len] ;
|
CharacterRange* ranges = new CharacterRange[len] ;
|
||||||
Region* regions = new Region[len];
|
Region* regions = new Region[len];
|
||||||
@@ -1187,6 +1298,7 @@ void wxGDIPlusContext::GetPartialTextExtents(const wxString& text, wxArrayDouble
|
|||||||
ranges[i].Length = 1 ;
|
ranges[i].Length = 1 ;
|
||||||
}
|
}
|
||||||
strFormat.SetMeasurableCharacterRanges(len,ranges);
|
strFormat.SetMeasurableCharacterRanges(len,ranges);
|
||||||
|
strFormat.SetFormatFlags( StringFormatFlagsMeasureTrailingSpaces | strFormat.GetFormatFlags() );
|
||||||
m_context->MeasureCharacterRanges(ws, -1 , f,layoutRect, &strFormat,1,regions) ;
|
m_context->MeasureCharacterRanges(ws, -1 , f,layoutRect, &strFormat,1,regions) ;
|
||||||
|
|
||||||
RectF bbox ;
|
RectF bbox ;
|
||||||
@@ -1355,9 +1467,7 @@ wxGraphicsContext * wxGDIPlusRenderer::CreateContext( const wxMemoryDC& dc)
|
|||||||
wxGraphicsContext * wxGDIPlusRenderer::CreateMeasuringContext()
|
wxGraphicsContext * wxGDIPlusRenderer::CreateMeasuringContext()
|
||||||
{
|
{
|
||||||
EnsureIsLoaded();
|
EnsureIsLoaded();
|
||||||
return NULL;
|
return new wxGDIPlusMeasuringContext(this);
|
||||||
// TODO use GetDC(NULL) but then we have to release it from the context
|
|
||||||
//return new wxGDIPlusContext(this,(HDC) dc.GetHDC());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
wxGraphicsContext * wxGDIPlusRenderer::CreateContextFromNativeContext( void * context )
|
wxGraphicsContext * wxGDIPlusRenderer::CreateContextFromNativeContext( void * context )
|
||||||
|
Reference in New Issue
Block a user