/////////////////////////////////////////////////////////////////////////////// // Name: tests/graphics/affinetransform.cpp // Purpose: Unit test for transformations implemented for wxAffineMatrix2D // Author: Catalin Raceanu // Created: 2011-04-14 // Copyright: (c) 2011 wxWidgets development team /////////////////////////////////////////////////////////////////////////////// // ---------------------------------------------------------------------------- // headers // ---------------------------------------------------------------------------- #include "testprec.h" #ifdef __BORLANDC__ #pragma hdrstop #endif #include "wx/graphics.h" #include "wx/dcmemory.h" #include "wx/affinematrix2d.h" #include "wx/math.h" #if wxUSE_GRAPHICS_CONTEXT #include "wx/dcgraph.h" #endif // wxUSE_GRAPHICS_CONTEXT #include "testimage.h" // ---------------------------------------------------------------------------- // Affine transform test class // ---------------------------------------------------------------------------- class AffineTransformTestCase : public CppUnit::TestCase { public: AffineTransformTestCase() {} private: CPPUNIT_TEST_SUITE( AffineTransformTestCase ); CPPUNIT_TEST( InvertMatrix ); CPPUNIT_TEST( Concat ); CPPUNIT_TEST_SUITE_END(); void InvertMatrix(); void Concat(); wxDECLARE_NO_COPY_CLASS(AffineTransformTestCase); }; // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( AffineTransformTestCase ); // also include in its own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( AffineTransformTestCase, "AffineTransformTestCase" ); void AffineTransformTestCase::InvertMatrix() { wxAffineMatrix2D matrix1; matrix1.Set(wxMatrix2D(2, 1, 1, 1), wxPoint2DDouble(1, 1)); wxAffineMatrix2D matrix2(matrix1); matrix2.Invert(); wxMatrix2D m; wxPoint2DDouble p; matrix2.Get(&m, &p); CPPUNIT_ASSERT_EQUAL( 1, (int)m.m_11 ); CPPUNIT_ASSERT_EQUAL( -1, (int)m.m_12 ); CPPUNIT_ASSERT_EQUAL( -1, (int)m.m_21 ); CPPUNIT_ASSERT_EQUAL( 2, (int)m.m_22 ); CPPUNIT_ASSERT_EQUAL( 0, (int)p.m_x ); CPPUNIT_ASSERT_EQUAL( -1, (int)p.m_y ); matrix2.Concat(matrix1); CPPUNIT_ASSERT( matrix2.IsIdentity() ); } void AffineTransformTestCase::Concat() { wxAffineMatrix2D m1; m1.Set(wxMatrix2D(0.9, 0.4, -0.4, 0.9), wxPoint2DDouble(0.0, 0.0)); wxAffineMatrix2D m2; m2.Set(wxMatrix2D(1.0, 0.0, 0.0, 1.0), wxPoint2DDouble(3.0, 5.0)); m1.Concat(m2); wxMatrix2D m; wxPoint2DDouble p; m1.Get(&m, &p); const double delta = 0.01; CPPUNIT_ASSERT_DOUBLES_EQUAL( 0.9, m.m_11, delta ); CPPUNIT_ASSERT_DOUBLES_EQUAL( 0.4, m.m_12, delta ); CPPUNIT_ASSERT_DOUBLES_EQUAL( -0.4, m.m_21, delta ); CPPUNIT_ASSERT_DOUBLES_EQUAL( 0.9, m.m_22, delta ); CPPUNIT_ASSERT_DOUBLES_EQUAL( 0.7, p.m_x, delta ); CPPUNIT_ASSERT_DOUBLES_EQUAL( 5.7, p.m_y, delta ); } #if wxUSE_DC_TRANSFORM_MATRIX // ------------------------------- // Transform matrix test classes // ------------------------------- // ==================== // wxDC / wxGCDC tests // ==================== class TransformMatrixTestCaseDCBase : public CppUnit::TestCase { public: TransformMatrixTestCaseDCBase() { m_dc = NULL; wxImage::AddHandler(new wxJPEGHandler); m_imgOrig.LoadFile(wxS("horse.jpg")); CPPUNIT_ASSERT( m_imgOrig.IsOk() ); } virtual ~TransformMatrixTestCaseDCBase() { } virtual void setUp() { m_bmpOrig = wxBitmap(m_imgOrig); m_bmpUsingMatrix.Create(m_bmpOrig.GetSize(), m_bmpOrig.GetDepth()); } protected: virtual void FlushDC() = 0; void VMirrorAndTranslate(); void Rotate90Clockwise(); #if wxUSE_GRAPHICS_CONTEXT void CompareToGraphicsContext(); #endif // wxUSE_GRAPHICS_CONTEXT protected: wxImage m_imgOrig; wxBitmap m_bmpOrig; wxBitmap m_bmpUsingMatrix; wxDC* m_dc; wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseDCBase); }; // =========== // wxDC tests // =========== class TransformMatrixTestCaseDC : public TransformMatrixTestCaseDCBase { public: TransformMatrixTestCaseDC() { m_dc = &m_mdc; } virtual ~TransformMatrixTestCaseDC() { } virtual void setUp() wxOVERRIDE { TransformMatrixTestCaseDCBase::setUp(); m_mdc.SelectObject(m_bmpUsingMatrix); } virtual void tearDown() wxOVERRIDE { m_mdc.SelectObject(wxNullBitmap); TransformMatrixTestCaseDCBase::tearDown(); } protected: virtual void FlushDC() wxOVERRIDE {} private: CPPUNIT_TEST_SUITE( TransformMatrixTestCaseDC ); CPPUNIT_TEST( VMirrorAndTranslate ); CPPUNIT_TEST( Rotate90Clockwise ); #if wxUSE_GRAPHICS_CONTEXT CPPUNIT_TEST( CompareToGraphicsContext ); #endif // wxUSE_GRAPHICS_CONTEXT CPPUNIT_TEST_SUITE_END(); protected: wxMemoryDC m_mdc; wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseDC); }; // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( TransformMatrixTestCaseDC ); // also include in it's own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TransformMatrixTestCaseDC, "TransformMatrixTestCaseDC" ); #if wxUSE_GRAPHICS_CONTEXT // ============= // wxGCDC tests // ============= class TransformMatrixTestCaseGCDC : public TransformMatrixTestCaseDC { public: TransformMatrixTestCaseGCDC() {} virtual ~TransformMatrixTestCaseGCDC() {} virtual void setUp() wxOVERRIDE { TransformMatrixTestCaseDC::setUp(); m_gcdc = new wxGCDC(m_mdc); m_dc = m_gcdc; wxGraphicsContext* ctx = m_gcdc->GetGraphicsContext(); ctx->SetAntialiasMode(wxANTIALIAS_NONE); } virtual void tearDown() wxOVERRIDE { delete m_gcdc; TransformMatrixTestCaseDC::tearDown(); } protected: virtual void FlushDC() wxOVERRIDE { m_gcdc->GetGraphicsContext()->Flush(); } private: CPPUNIT_TEST_SUITE( TransformMatrixTestCaseGCDC ); CPPUNIT_TEST( VMirrorAndTranslate ); CPPUNIT_TEST( Rotate90Clockwise ); CPPUNIT_TEST_SUITE_END(); protected: wxGCDC* m_gcdc; wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseGCDC); }; // For MSW we have individual test cases for each graphics renderer // so we don't need to test wxGCDC with default renderer. #ifndef __WXMSW__ // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( TransformMatrixTestCaseGCDC ); // also include in it's own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TransformMatrixTestCaseGCDC, "TransformMatrixTestCaseGCDC" ); #endif // !__WXMSW__ #ifdef __WXMSW__ // GDI+ and Direct2D are available only under MSW. #if wxUSE_GRAPHICS_GDIPLUS class TransformMatrixTestCaseGCDCGDIPlus : public TransformMatrixTestCaseGCDC { public: TransformMatrixTestCaseGCDCGDIPlus() {} virtual ~TransformMatrixTestCaseGCDCGDIPlus() {} virtual void setUp() wxOVERRIDE { TransformMatrixTestCaseGCDC::setUp(); wxGraphicsRenderer* rend = wxGraphicsRenderer::GetGDIPlusRenderer(); wxGraphicsContext* ctx = rend->CreateContext(m_mdc); m_gcdc->SetGraphicsContext(ctx); } private: CPPUNIT_TEST_SUITE( TransformMatrixTestCaseGCDCGDIPlus ); CPPUNIT_TEST( VMirrorAndTranslate ); CPPUNIT_TEST( Rotate90Clockwise ); CPPUNIT_TEST_SUITE_END(); protected: wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseGCDCGDIPlus); }; // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( TransformMatrixTestCaseGCDCGDIPlus ); // also include in it's own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TransformMatrixTestCaseGCDCGDIPlus, "TransformMatrixTestCaseGCDCGDIPlus" ); #endif // wxUSE_GRAPHICS_GDIPLUS #if wxUSE_GRAPHICS_DIRECT2D class TransformMatrixTestCaseGCDCDirect2D : public TransformMatrixTestCaseGCDC { public: TransformMatrixTestCaseGCDCDirect2D() {} virtual ~TransformMatrixTestCaseGCDCDirect2D() {} virtual void setUp() wxOVERRIDE { TransformMatrixTestCaseGCDC::setUp(); wxGraphicsRenderer* rend = wxGraphicsRenderer::GetDirect2DRenderer(); wxGraphicsContext* ctx = rend->CreateContext(m_mdc); m_gcdc->SetGraphicsContext(ctx); } virtual void FlushDC() { // Apparently, flushing native Direct2D renderer // is not enough to update underlying DC (bitmap) // and therefore we have to destroy the renderer // to do so. TransformMatrixTestCaseGCDC::FlushDC(); m_gcdc->SetGraphicsContext(NULL); } private: CPPUNIT_TEST_SUITE( TransformMatrixTestCaseGCDCDirect2D ); CPPUNIT_TEST( VMirrorAndTranslate ); CPPUNIT_TEST( Rotate90Clockwise ); CPPUNIT_TEST_SUITE_END(); protected: wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseGCDCDirect2D); }; // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( TransformMatrixTestCaseGCDCDirect2D ); // also include in it's own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TransformMatrixTestCaseGCDCDirect2D, "TransformMatrixTestCaseGCDCDirect2D" ); #endif // wxUSE_GRAPHICS_DIRECT2D #endif // __WXMSW__ #if wxUSE_CAIRO class TransformMatrixTestCaseGCDCCairo : public TransformMatrixTestCaseGCDC { public: TransformMatrixTestCaseGCDCCairo() {} virtual ~TransformMatrixTestCaseGCDCCairo() {} virtual void setUp() wxOVERRIDE { TransformMatrixTestCaseGCDC::setUp(); wxGraphicsRenderer* rend = wxGraphicsRenderer::GetCairoRenderer(); wxGraphicsContext* ctx = rend->CreateContext(m_mdc); m_gcdc->SetGraphicsContext(ctx); } private: CPPUNIT_TEST_SUITE( TransformMatrixTestCaseGCDCCairo ); CPPUNIT_TEST( VMirrorAndTranslate ); CPPUNIT_TEST( Rotate90Clockwise ); CPPUNIT_TEST_SUITE_END(); protected: wxDECLARE_NO_COPY_CLASS(TransformMatrixTestCaseGCDCCairo); }; // register in the unnamed registry so that these tests are run by default CPPUNIT_TEST_SUITE_REGISTRATION( TransformMatrixTestCaseGCDCCairo ); // also include in it's own registry so that these tests can be run alone CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TransformMatrixTestCaseGCDCCairo, "TransformMatrixTestCaseGCDCCairo" ); #endif // wxUSE_CAIRO #endif // wxUSE_GRAPHICS_CONTEXT void TransformMatrixTestCaseDCBase::VMirrorAndTranslate() { // build the mirrored image using the transformation matrix if ( !m_dc->CanUseTransformMatrix() ) return; wxAffineMatrix2D matrix; matrix.Mirror(wxVERTICAL); // For wxDC pixel center is at (0, 0) so row 0 of the bitmap is the axis // of mirroring and it is left intact by the transformation. In this case // mirrored bitmap needs to be shifted by dim-1 pixels. // For wxGCDC pixel center of underlying wxGraphicsContext is at (0.5, 0.5) // so the axis of mirroring is above row 0 of the bitmap and this row // is affected by the transformation. In this case mirrored bitmap // needs to be shifthed by dim pixels. int ty; if ( m_dc->GetGraphicsContext() ) ty = m_bmpOrig.GetHeight(); else ty = m_bmpOrig.GetHeight() - 1; matrix.Translate(0, -ty); m_dc->SetTransformMatrix(matrix); m_dc->DrawBitmap(m_bmpOrig, 0, 0); FlushDC(); CPPUNIT_ASSERT_EQUAL( m_bmpUsingMatrix.ConvertToImage(), m_imgOrig.Mirror(false) ); } void TransformMatrixTestCaseDCBase::Rotate90Clockwise() { // build the rotated image using the transformation matrix if ( !m_dc->CanUseTransformMatrix() ) return; wxAffineMatrix2D matrix; matrix.Rotate(0.5 * M_PI); matrix.Translate(0, -m_bmpOrig.GetHeight()); m_dc->SetTransformMatrix(matrix); m_dc->DrawBitmap(m_bmpOrig, 0, 0); FlushDC(); CPPUNIT_ASSERT_EQUAL( m_bmpUsingMatrix.ConvertToImage(), m_imgOrig.Rotate90(true) ); } #if wxUSE_GRAPHICS_CONTEXT void TransformMatrixTestCaseDCBase::CompareToGraphicsContext() { wxPoint2DDouble pointA1(1.0, 3.0), pointA2(60.0, 50.0), pointG1(1.0, 3.0), pointG2(60.0, 50.0); // Create affine matrix and transform it wxAffineMatrix2D matrixA1, matrixA2; matrixA2.Rotate(M_PI / 3); matrixA1.Translate(-m_bmpOrig.GetWidth()/2, -m_bmpOrig.GetHeight()/2); matrixA1.Rotate(-M_PI *2/ 6); matrixA1.Translate(m_bmpOrig.GetWidth()/2, m_bmpOrig.GetHeight()/2); matrixA1.Mirror(wxHORIZONTAL); matrixA1.Concat(matrixA2); matrixA1.Mirror(wxVERTICAL); matrixA1.Translate(m_bmpOrig.GetWidth()/2, -m_bmpOrig.GetHeight()/2); matrixA1.Scale(0.9, 0.9); matrixA1.Invert(); // Create image using first matrix wxBitmap bmpUsingMatrixA1(m_bmpOrig.GetSize(), m_bmpOrig.GetDepth()); // Build the transformed image using the transformation matrix { wxMemoryDC dc(bmpUsingMatrixA1); if ( !dc.CanUseTransformMatrix() ) return; // Draw the bitmap dc.SetTransformMatrix(matrixA1); dc.DrawBitmap(m_bmpOrig, 0, 0); // Draw a line matrixA1.TransformPoint(&pointA1.m_x, &pointA1.m_y); matrixA1.TransformDistance(&pointA2.m_x, &pointA2.m_y); dc.DrawLine(wxRound(pointA1.m_x), wxRound(pointA1.m_y), wxRound(pointA1.m_x + pointA2.m_x), wxRound(pointA1.m_x + pointA2.m_y)); } // Create graphics matrix and transform it wxGraphicsRenderer* r; if ( m_dc->GetGraphicsContext() ) { r = m_dc->GetGraphicsContext()->GetRenderer(); } else { r = wxGraphicsRenderer::GetDefaultRenderer(); } wxBitmap bmp(10, 10); wxMemoryDC mDc(bmp); wxGraphicsContext* gDc = r->CreateContext(mDc); wxGraphicsMatrix matrixG1 = gDc->CreateMatrix(); wxGraphicsMatrix matrixG2 = gDc->CreateMatrix(); matrixG2.Rotate(M_PI / 3); matrixG1.Translate(-m_bmpOrig.GetWidth()/2, -m_bmpOrig.GetHeight()/2); matrixG1.Rotate(-M_PI*2 / 6); matrixG1.Translate(m_bmpOrig.GetWidth()/2, m_bmpOrig.GetHeight()/2); matrixG1.Scale(-1, 1); matrixG1.Concat(matrixG2); matrixG1.Scale(1, -1); matrixG1.Translate(m_bmpOrig.GetWidth()/2, -m_bmpOrig.GetHeight()/2); matrixG1.Scale(0.9, 0.9); matrixG1.Invert(); // Create affine matrix from the graphics matrix wxMatrix2D mat2D; wxPoint2DDouble tr; matrixG1.Get(&mat2D.m_11, &mat2D.m_12, &mat2D.m_21, &mat2D.m_22, &tr.m_x, &tr.m_y); wxAffineMatrix2D matrixAG; matrixAG.Set(mat2D, tr); delete gDc; // Create image using last matrix wxBitmap bmpUsingMatrixAG(m_bmpOrig.GetHeight(), m_bmpOrig.GetWidth()); // Build the transformed image using the transformation matrix { wxMemoryDC dc(bmpUsingMatrixAG); if ( !dc.CanUseTransformMatrix() ) return; // Draw the bitmap dc.SetTransformMatrix(matrixAG); dc.DrawBitmap(m_bmpOrig, 0, 0); // Draw a line matrixG1.TransformPoint(&pointG1.m_x, &pointG1.m_y); matrixG1.TransformDistance(&pointG2.m_x, &pointG2.m_y); dc.DrawLine(wxRound(pointG1.m_x), wxRound(pointG1.m_y), wxRound(pointG1.m_x + pointG2.m_x), wxRound(pointG1.m_x + pointG2.m_y)); } CPPUNIT_ASSERT_EQUAL( bmpUsingMatrixA1.ConvertToImage(), bmpUsingMatrixAG.ConvertToImage() ); // Save the images to check that something _is_ inside the visible area. //bmpUsingMatrixA1.SaveFile("matrixA1.jpg", wxBITMAP_TYPE_JPEG); //bmpUsingMatrixAG.SaveFile("matrixAG.jpg", wxBITMAP_TYPE_JPEG); } #endif // wxUSE_GRAPHICS_CONTEXT #endif // wxUSE_DC_TRANSFORM_MATRIX