summaryrefslogtreecommitdiffstats
path: root/cppcanvas/qa
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-15 05:54:39 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-15 05:54:39 +0000
commit267c6f2ac71f92999e969232431ba04678e7437e (patch)
tree358c9467650e1d0a1d7227a21dac2e3d08b622b2 /cppcanvas/qa
parentInitial commit. (diff)
downloadlibreoffice-267c6f2ac71f92999e969232431ba04678e7437e.tar.xz
libreoffice-267c6f2ac71f92999e969232431ba04678e7437e.zip
Adding upstream version 4:24.2.0.upstream/4%24.2.0
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'cppcanvas/qa')
-rw-r--r--cppcanvas/qa/extras/emfplus/data/fdo77229.emfbin0 -> 593224 bytes
-rw-r--r--cppcanvas/qa/extras/emfplus/emfplus.cxx64
-rw-r--r--cppcanvas/qa/unit/test.cxx208
3 files changed, 272 insertions, 0 deletions
diff --git a/cppcanvas/qa/extras/emfplus/data/fdo77229.emf b/cppcanvas/qa/extras/emfplus/data/fdo77229.emf
new file mode 100644
index 0000000000..1f12717fa4
--- /dev/null
+++ b/cppcanvas/qa/extras/emfplus/data/fdo77229.emf
Binary files differ
diff --git a/cppcanvas/qa/extras/emfplus/emfplus.cxx b/cppcanvas/qa/extras/emfplus/emfplus.cxx
new file mode 100644
index 0000000000..1ba179404f
--- /dev/null
+++ b/cppcanvas/qa/extras/emfplus/emfplus.cxx
@@ -0,0 +1,64 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ */
+
+#include <test/unoapi_test.hxx>
+
+#include <sfx2/objsh.hxx>
+#include <sfx2/sfxbasemodel.hxx>
+#include <vcl/BitmapReadAccess.hxx>
+#include <vcl/filter/PngImageWriter.hxx>
+#include <vcl/gdimtf.hxx>
+#include <tools/stream.hxx>
+
+using namespace com::sun::star;
+
+class Test : public UnoApiTest
+{
+public:
+ Test()
+ : UnoApiTest("/cppcanvas/qa/extras/emfplus/data/")
+ {
+ }
+
+ Bitmap load(const char* pName)
+ {
+ loadFromFile(OUString::createFromAscii(pName));
+ SfxBaseModel* pModel = dynamic_cast<SfxBaseModel*>(mxComponent.get());
+ CPPUNIT_ASSERT(pModel);
+ SfxObjectShell* pShell = pModel->GetObjectShell();
+ std::shared_ptr<GDIMetaFile> xMetaFile = pShell->GetPreviewMetaFile();
+ BitmapEx aResultBitmap;
+ CPPUNIT_ASSERT(xMetaFile->CreateThumbnail(aResultBitmap));
+ // If this is set, the metafile will be dumped as a PNG one for debug purposes.
+ char* pEnv = getenv("CPPCANVAS_DEBUG_EMFPLUS_DUMP_TO");
+ if (pEnv)
+ {
+ SvFileStream aStream(OUString::fromUtf8(pEnv), StreamMode::WRITE);
+ vcl::PngImageWriter aWriter(aStream);
+ CPPUNIT_ASSERT(aWriter.write(aResultBitmap));
+ }
+
+ return aResultBitmap.GetBitmap();
+ }
+};
+
+CPPUNIT_TEST_FIXTURE(Test, testFdo77229)
+{
+ Bitmap aBitmap = load("fdo77229.emf");
+ BitmapScopedReadAccess pAccess(aBitmap);
+ // The green star was missing.
+ Color aColor(pAccess->GetPixel(284, 280));
+ CPPUNIT_ASSERT_EQUAL(sal_uInt8(0), aColor.GetRed());
+ CPPUNIT_ASSERT_EQUAL(sal_uInt8(0), aColor.GetBlue());
+ CPPUNIT_ASSERT(aColor.GetGreen() == 0xfe || aColor.GetGreen() == 0xff);
+}
+
+CPPUNIT_PLUGIN_IMPLEMENT();
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
diff --git a/cppcanvas/qa/unit/test.cxx b/cppcanvas/qa/unit/test.cxx
new file mode 100644
index 0000000000..ad278f1083
--- /dev/null
+++ b/cppcanvas/qa/unit/test.cxx
@@ -0,0 +1,208 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ */
+
+#include <config_cairo_canvas.h>
+
+#include <sal/config.h>
+
+#include <vcl/wrkwin.hxx>
+#include <vcl/canvastools.hxx>
+#include <vcl/virdev.hxx>
+#include <vcl/gdimtf.hxx>
+#include <vcl/metaact.hxx>
+
+#include <com/sun/star/rendering/XBitmap.hpp>
+#include <com/sun/star/rendering/XCanvas.hpp>
+#include <com/sun/star/rendering/XBitmapCanvas.hpp>
+
+#include <cppcanvas/vclfactory.hxx>
+
+using namespace ::com::sun::star;
+
+static std::ostream& operator<<(std::ostream& rStream, const KernArray& rArray)
+{
+ if (rArray.empty())
+ {
+ rStream << "{ }";
+ return rStream;
+ }
+
+ rStream << "{ ";
+ for (size_t i = 0; i < rArray.size() - 1; i++)
+ rStream << rArray[i] << ", ";
+ rStream << rArray.back();
+ rStream << " }";
+ return rStream;
+}
+
+static std::ostream& operator<<(std::ostream& rStream, const std::vector<sal_Bool>& rVec)
+{
+ if (rVec.empty())
+ {
+ rStream << "{ }";
+ return rStream;
+ }
+
+ rStream << "{ ";
+ for (size_t i = 0; i < rVec.size() - 1; i++)
+ rStream << std::boolalpha << bool(rVec[i]) << ", ";
+ rStream << std::boolalpha << bool(rVec.back());
+ rStream << " }";
+ return rStream;
+}
+
+#include <test/bootstrapfixture.hxx>
+
+class CanvasTest : public test::BootstrapFixture
+{
+public:
+ CanvasTest()
+ : BootstrapFixture(true, false)
+ {
+ }
+};
+
+CPPUNIT_TEST_FIXTURE(CanvasTest, testComposite)
+{
+ ScopedVclPtrInstance<WorkWindow> pWin(nullptr, WB_STDWORK);
+
+ uno::Reference<rendering::XCanvas> xCanvas = pWin->GetOutDev()->GetCanvas();
+ CPPUNIT_ASSERT(xCanvas.is());
+
+ // a huge canvas ...
+ Size aSize(1, 1);
+ uno::Reference<rendering::XBitmap> xBitmap = xCanvas->getDevice()->createCompatibleAlphaBitmap(
+ vcl::unotools::integerSize2DFromSize(aSize));
+ CPPUNIT_ASSERT(xBitmap.is());
+
+ uno::Reference<rendering::XBitmapCanvas> xBitmapCanvas(xBitmap, uno::UNO_QUERY);
+ CPPUNIT_ASSERT(xBitmapCanvas.is());
+
+ BitmapEx aBitmapEx;
+ {
+ // clear the canvas and basic sanity check ...
+ xBitmapCanvas->clear();
+ CPPUNIT_ASSERT(aBitmapEx.Create(xBitmapCanvas, aSize));
+ CPPUNIT_ASSERT(aBitmapEx.IsAlpha());
+ CPPUNIT_ASSERT(!aBitmapEx.GetAlphaMask().IsEmpty());
+ }
+
+ {
+ // render something
+ rendering::RenderState aDefaultState;
+ uno::Sequence<double> aRedTransparent{ 1.0, // R
+ 0.0, // G
+ 0.0, // B
+ 0.5 }; // A
+ aDefaultState.DeviceColor = aRedTransparent;
+#if 0
+ // words fail me to describe the sheer beauty of allocating a UNO
+ // object to represent a polygon, and manually handling the ViewState
+ // and there being no public helper for this - to render ... a rectangle.
+ XCachedPrimitive fillPolyPolygon( [in] XPolyPolygon2D xPolyPolygon, [in] ViewState aViewState, [in] RenderState aRenderState )
+#endif
+ }
+}
+
+CPPUNIT_TEST_FIXTURE(CanvasTest, testTdf155810)
+{
+ GDIMetaFile aInputMetaFile, aOutputMetaFile;
+ KernArray aDXArray;
+ std::vector<sal_Bool> aKashidaArray;
+
+ // First create a meta file with a text array action that has Kashida adjustments.
+ {
+ ScopedVclPtrInstance<VirtualDevice> pDev;
+
+ vcl::Font aFont(u"Noto Naskh Arabic"_ustr, u"Regular"_ustr, Size(0, 72));
+ pDev->SetFont(aFont);
+
+ aInputMetaFile.Record(pDev.get());
+
+ OUString aText(u"خالد"_ustr);
+ pDev->GetTextArray(aText, &aDXArray);
+
+ auto nKashida = 200;
+ aDXArray.set(0, aDXArray[0] + nKashida);
+ aDXArray.set(2, aDXArray[2] + nKashida);
+ aKashidaArray = { true, false, true, false };
+
+ pDev->DrawTextArray(Point(0, 0), aText, aDXArray, aKashidaArray, 0, -1);
+
+ aInputMetaFile.Stop();
+ }
+
+ // Then draw it using canvas
+ {
+ ScopedVclPtrInstance<VirtualDevice> pDev;
+
+ aOutputMetaFile.Record(pDev.get());
+
+ auto xCanvas = pDev->GetCanvas();
+ CPPUNIT_ASSERT(xCanvas.is());
+ auto pCanvas = cppcanvas::VCLFactory::createCanvas(xCanvas);
+
+ auto pRenderer = cppcanvas::VCLFactory::createRenderer(pCanvas, aInputMetaFile, {});
+ pRenderer->draw();
+
+ aOutputMetaFile.Stop();
+ }
+
+ // Then check that the text array drawn by the canvas renderer didn’t loose
+ // the Kashida insertion positions.
+ {
+ auto pInputAction = aInputMetaFile.GetAction(aInputMetaFile.GetActionSize() - 1);
+ auto pOutputAction = aOutputMetaFile.GetAction(aOutputMetaFile.GetActionSize() - 2);
+
+ CPPUNIT_ASSERT_EQUAL(MetaActionType::TEXTARRAY, pInputAction->GetType());
+ CPPUNIT_ASSERT_EQUAL(pInputAction->GetType(), pOutputAction->GetType());
+
+ MetaTextArrayAction* pInputTextAction = static_cast<MetaTextArrayAction*>(pInputAction);
+ MetaTextArrayAction* pOutputTextAction = static_cast<MetaTextArrayAction*>(pOutputAction);
+
+ CPPUNIT_ASSERT_EQUAL(pInputTextAction->GetDXArray(), aDXArray);
+ CPPUNIT_ASSERT_EQUAL(pInputTextAction->GetDXArray(), pOutputTextAction->GetDXArray());
+
+ CPPUNIT_ASSERT_EQUAL(pInputTextAction->GetKashidaArray(), aKashidaArray);
+ CPPUNIT_ASSERT_EQUAL(pInputTextAction->GetKashidaArray(),
+ pOutputTextAction->GetKashidaArray());
+ }
+
+ // Now, test drawSubset
+ {
+ ScopedVclPtrInstance<VirtualDevice> pDev;
+
+ aOutputMetaFile.Clear();
+ aOutputMetaFile.Record(pDev.get());
+
+ auto xCanvas = pDev->GetCanvas();
+ CPPUNIT_ASSERT(xCanvas.is());
+ auto pCanvas = cppcanvas::VCLFactory::createCanvas(xCanvas);
+
+ auto pRenderer = cppcanvas::VCLFactory::createRenderer(pCanvas, aInputMetaFile, {});
+ pRenderer->drawSubset(1, 3);
+
+ aOutputMetaFile.Stop();
+ }
+
+ {
+ auto pOutputAction = aOutputMetaFile.GetAction(aOutputMetaFile.GetActionSize() - 2);
+
+ CPPUNIT_ASSERT_EQUAL(MetaActionType::TEXTARRAY, pOutputAction->GetType());
+
+ MetaTextArrayAction* pOutputTextAction = static_cast<MetaTextArrayAction*>(pOutputAction);
+ std::vector<sal_Bool> aSubsetKashidaArray({ false, true });
+
+ CPPUNIT_ASSERT_EQUAL(aSubsetKashidaArray, pOutputTextAction->GetKashidaArray());
+ }
+}
+
+CPPUNIT_PLUGIN_IMPLEMENT();
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */