/* -*- 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 "charttest.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #define EPS 1E-12 #if defined(MACOSX) #define INT_EPS 200.1 #elif defined(UNX) #define INT_EPS 2.1 #else #define INT_EPS 0.1 #endif #define DECLARE_DUMP_TEST(TestName, BaseClass, DumpMode) \ class TestName : public BaseClass { \ protected:\ virtual OUString getTestName() override { return OUString(#TestName); } \ public:\ TestName() : BaseClass(DumpMode) {}; \ CPPUNIT_TEST_SUITE(TestName); \ CPPUNIT_TEST(verify); \ CPPUNIT_TEST_SUITE_END(); \ virtual void verify() override;\ };\ CPPUNIT_TEST_SUITE_REGISTRATION(TestName); \ void TestName::verify() #define CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(aActual) \ if(isInDumpMode()) \ writeActual(OUString::number(aActual), #aActual); \ else \ { \ OString sTestFileName = OUStringToOString(getTestFileName(), RTL_TEXTENCODING_UTF8); \ CPPUNIT_ASSERT_EQUAL_MESSAGE(OString("Failing test file is: " + sTestFileName).getStr(), readExpected(#aActual), OUString::number(aActual)); \ } #define CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aActual, EPS_) \ if(isInDumpMode()) \ writeActual(OUString::number(aActual), #aActual); \ else \ { \ OString sTestFileName = OUStringToOString(getTestFileName(), RTL_TEXTENCODING_UTF8); \ CPPUNIT_ASSERT_DOUBLES_EQUAL_MESSAGE(OString("Failing test file is: " + sTestFileName).getStr(), readExpectedDouble(#aActual), aActual, EPS_); \ } #define CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(aActual) \ if(isInDumpMode()) \ writeActual(aActual, #aActual); \ else \ { \ OString sTestFileName = OUStringToOString(getTestFileName(), RTL_TEXTENCODING_UTF8); \ CPPUNIT_ASSERT_EQUAL_MESSAGE(OString("Failing test file is: " + sTestFileName).getStr(), readExpected(#aActual), aActual.trim()); \ } #define CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aActual) \ if(isInDumpMode()) \ writeActualTransformation(aActual, #aActual); \ else \ { \ OUString expectedTransform; \ if (!readAndCheckTransformation (aActual, #aActual, expectedTransform)) \ { \ OString sTestFileName = OUStringToOString(getTestFileName(), RTL_TEXTENCODING_UTF8); \ CPPUNIT_ASSERT_EQUAL_MESSAGE(OString("Failing test file is: " + sTestFileName).getStr(), expectedTransform, transformationToOneLineString(aActual)); \ } \ } #define CPPUNIT_DUMP_ASSERT_NOTE(Note) \ if(isInDumpMode()) \ writeNote(OUString(Note)); \ else \ readNote(OUString(Note));\ class Chart2DumpTest : public ChartTest, public XmlTestTools { protected: Chart2DumpTest(bool bDumpMode) { m_bDumpMode = bDumpMode; } virtual ~Chart2DumpTest() override { } bool isInDumpMode () const {return m_bDumpMode;} virtual OUString getTestName() { return OUString(); } OUString getTestFileName() { return m_sTestFileName; } OUString getTestFileDirName() { return OUString("/chart2/qa/extras/chart2dump/data/"); } OUString getReferenceDirName() { return "/chart2/qa/extras/chart2dump/reference/" + getTestName().toAsciiLowerCase() + "/"; } void setTestFileName (const OUString& sName) { m_sTestFileName = sName; OUString sFileName = m_sTestFileName; assert(sFileName.lastIndexOf('.') < sFileName.getLength()); sFileName = sFileName.copy(0, sFileName.lastIndexOf('.')) + ".txt"; if (!m_bDumpMode) { if (m_aReferenceFile.is_open()) m_aReferenceFile.close(); OString sReferenceFile = OUStringToOString(m_directories.getPathFromSrc(getReferenceDirName()) + sFileName, RTL_TEXTENCODING_UTF8); m_aReferenceFile.open(sReferenceFile.getStr(), std::ios_base::in); CPPUNIT_ASSERT_MESSAGE(OString("Can't open reference file: " + sReferenceFile).getStr(), m_aReferenceFile.is_open()); } else { if (m_aDumpFile.is_open()) m_aDumpFile.close(); OString sDumpFile = OUStringToOString(m_directories.getPathFromSrc(getReferenceDirName()) + sFileName, RTL_TEXTENCODING_UTF8); m_aDumpFile.open(sDumpFile.getStr(), std::ios_base::out | std::ofstream::binary | std::ofstream::trunc); CPPUNIT_ASSERT_MESSAGE(OString("Can't open dump file: " + sDumpFile).getStr(), m_aDumpFile.is_open()); } } virtual void verify() { CPPUNIT_FAIL("verify method must be overridden"); } OUString readExpected(const OUString& sCheck) { assert(!m_bDumpMode); assert(m_aReferenceFile.is_open()); std::string sTemp; getline(m_aReferenceFile, sTemp); OString sAssertMessage = OString("The reference file does not contain the right content. Maybe it needs an update:") + OUStringToOString(m_sTestFileName, RTL_TEXTENCODING_UTF8); CPPUNIT_ASSERT_EQUAL_MESSAGE(sAssertMessage.getStr(), OUString("// " + sCheck), OUString(sTemp.data(), sTemp.length(), RTL_TEXTENCODING_UTF8)); getline(m_aReferenceFile, sTemp); return OUString(sTemp.data(), sTemp.length(), RTL_TEXTENCODING_UTF8); } void writeActual(const OUString& sActualValue, const OUString& sCheck) { assert(m_bDumpMode); assert(m_aDumpFile.is_open()); m_aDumpFile << "// " << OUStringToOString(sCheck, RTL_TEXTENCODING_UTF8).getStr() << "\n"; // Add check string to make dump file readable m_aDumpFile << OUStringToOString(sActualValue.trim(), RTL_TEXTENCODING_UTF8).getStr() << "\n"; // Write out the checked value, will be used as reference later } void readNote(const OUString& sNote) { assert(!m_bDumpMode); assert(m_aReferenceFile.is_open()); std::string sTemp; getline(m_aReferenceFile, sTemp); OString sAssertMessage = OString("The reference file does not contain the right content. Maybe it needs an update:") + OUStringToOString(m_sTestFileName, RTL_TEXTENCODING_UTF8); CPPUNIT_ASSERT_EQUAL_MESSAGE(sAssertMessage.getStr(), OUString("/// " + sNote), OUString(sTemp.data(), sTemp.length(), RTL_TEXTENCODING_UTF8)); } void writeNote(const OUString& sNote) { assert(m_bDumpMode); assert(m_aDumpFile.is_open()); m_aDumpFile << "/// " << OUStringToOString(sNote, RTL_TEXTENCODING_UTF8).getStr() << "\n"; } double readExpectedDouble(const OUString& sCheck) { OUString sExpected = readExpected(sCheck); return sExpected.toDouble(); } void writeActualTransformation(const drawing::HomogenMatrix3& rTransform, const OUString& sCheck) { writeActual(transformationToOneLineString(rTransform), sCheck); } bool readAndCheckTransformation(const drawing::HomogenMatrix3& rTransform, const OUString& sCheck, OUString& rExpectedTranform) { rExpectedTranform = readExpected(sCheck); // Reference transfromation string // Covnert string back to a transformation; drawing::HomogenMatrix3 aExpectedTransform; aExpectedTransform.Line1.Column1 = rExpectedTranform.getToken(0, ';').toDouble(); aExpectedTransform.Line1.Column2 = rExpectedTranform.getToken(1, ';').toDouble(); aExpectedTransform.Line1.Column3 = rExpectedTranform.getToken(2, ';').toDouble(); aExpectedTransform.Line2.Column1 = rExpectedTranform.getToken(3, ';').toDouble(); aExpectedTransform.Line2.Column2 = rExpectedTranform.getToken(4, ';').toDouble(); aExpectedTransform.Line2.Column3 = rExpectedTranform.getToken(5, ';').toDouble(); aExpectedTransform.Line3.Column1 = rExpectedTranform.getToken(6, ';').toDouble(); aExpectedTransform.Line3.Column2 = rExpectedTranform.getToken(7, ';').toDouble(); aExpectedTransform.Line3.Column3 = rExpectedTranform.getToken(8, ';').toDouble(); // Check the equality of the two transformation return (std::abs(aExpectedTransform.Line1.Column1 - rTransform.Line1.Column1) < INT_EPS && std::abs(aExpectedTransform.Line1.Column2 - rTransform.Line1.Column2) < INT_EPS && std::abs(aExpectedTransform.Line1.Column3 - rTransform.Line1.Column3) < INT_EPS && std::abs(aExpectedTransform.Line2.Column1 - rTransform.Line2.Column1) < INT_EPS && std::abs(aExpectedTransform.Line2.Column2 - rTransform.Line2.Column2) < INT_EPS && std::abs(aExpectedTransform.Line2.Column3 - rTransform.Line2.Column3) < INT_EPS && std::abs(aExpectedTransform.Line3.Column1 - rTransform.Line3.Column1) < INT_EPS && std::abs(aExpectedTransform.Line3.Column2 - rTransform.Line3.Column2) < INT_EPS && std::abs(aExpectedTransform.Line3.Column3 - rTransform.Line3.Column3) < INT_EPS); } OUString sequenceToOneLineString(uno::Sequence& rSeq) { OUStringBuffer aBufer; for (const OUString& seqItem : rSeq) { aBufer.append(seqItem + ";"); } return aBufer.makeStringAndClear(); } OUString doubleVectorToOneLineString(const std::vector& rVector) { OUStringBuffer aBufer; for (const double& vectorItem : rVector) { aBufer.append(OUString::number(vectorItem) + ";"); } return aBufer.makeStringAndClear(); } OUString transformationToOneLineString(const drawing::HomogenMatrix3& rTransform) { return OUString::number(rTransform.Line1.Column1) + ";" + OUString::number(rTransform.Line1.Column2) + ";" + OUString::number(rTransform.Line1.Column3) + ";" + OUString::number(rTransform.Line2.Column1) + ";" + OUString::number(rTransform.Line2.Column2) + ";" + OUString::number(rTransform.Line2.Column3) + ";" + OUString::number(rTransform.Line3.Column1) + ";" + OUString::number(rTransform.Line3.Column2) + ";" + OUString::number(rTransform.Line3.Column3); } uno::Reference getShapeByName(const uno::Reference& rShapes, const OUString& rName, bool (*pCondition)(const uno::Reference&) = nullptr) { uno::Reference XIndexAccess(rShapes, uno::UNO_QUERY); for (sal_Int32 i = 0; i < XIndexAccess->getCount(); ++i) { uno::Reference xShapes(XIndexAccess->getByIndex(i), uno::UNO_QUERY); if (xShapes.is()) { uno::Reference xRet = getShapeByName(xShapes, rName, pCondition); if (xRet.is()) return xRet; } uno::Reference xNamedShape(XIndexAccess->getByIndex(i), uno::UNO_QUERY); if (xNamedShape->getName() == rName) { uno::Reference xShape(xNamedShape, uno::UNO_QUERY); if (pCondition == nullptr || (*pCondition)(xShape)) return xShape; } } return uno::Reference(); } private: OUString m_sTestFileName; bool m_bDumpMode; std::ifstream m_aReferenceFile; std::ofstream m_aDumpFile; }; DECLARE_DUMP_TEST(ChartDataTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "simple_chart.ods", "multiple_categories.ods" }; for (const OUString& aTestFile : aTestFiles) { setTestFileName(aTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc (getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); CPPUNIT_ASSERT(xChartDoc.is()); // Check title uno::Reference< chart2::XChartDocument > xChartDoc2(xChartDoc, UNO_QUERY_THROW); Reference xTitled(xChartDoc, uno::UNO_QUERY_THROW); uno::Reference xTitle = xTitled->getTitleObject(); if(xTitle.is()) { OUString sChartTitle = getTitleString(xTitled); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sChartTitle); } // Check chart type Reference xChartType = getChartTypeFromDoc(xChartDoc2, 0); CPPUNIT_ASSERT(xChartType.is()); OUString sChartType = xChartType->getChartType(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sChartType); // Check axis titles and number format // x Axis Reference xAxis = getAxisFromDoc(xChartDoc2, 0, 0, 0); Reference xAxisTitled(xAxis, UNO_QUERY_THROW); uno::Reference xAxisTitle = xAxisTitled->getTitleObject(); if (xAxisTitle.is()) { OUString sXAxisTitle = getTitleString(xAxisTitled); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sXAxisTitle); } sal_Int32 nXAxisNumberFormat = getNumberFormatFromAxis(xAxis); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nXAxisNumberFormat); sal_Int16 nXAxisNumberType = getNumberFormatType(xChartDoc2, nXAxisNumberFormat); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nXAxisNumberType); // y Axis xAxis.set(getAxisFromDoc(xChartDoc2, 0, 1, 0)); xAxisTitled.set(xAxis, UNO_QUERY_THROW); xAxisTitle.set(xAxisTitled->getTitleObject()); if (xAxisTitle.is()) { OUString sYAxisTitle = getTitleString(xAxisTitled); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sYAxisTitle); } sal_Int32 nYAxisNumberFormat = getNumberFormatFromAxis(xAxis); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nYAxisNumberFormat); sal_Int16 nYAxisNumberType = getNumberFormatType(xChartDoc2, nYAxisNumberFormat); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nYAxisNumberType); // Check column labels uno::Reference< chart::XChartDataArray > xChartData(xChartDoc->getData(), UNO_QUERY_THROW); CPPUNIT_ASSERT(xChartData.is()); uno::Sequence < OUString > aColumnLabels = xChartData->getColumnDescriptions(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(aColumnLabels.getLength()); OUString sColumnLabels = sequenceToOneLineString(aColumnLabels); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sColumnLabels); // Check row labels uno::Sequence< OUString > aRowLabels = xChartData->getRowDescriptions(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(aRowLabels.getLength()); OUString sRowLabels = sequenceToOneLineString(aRowLabels); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sRowLabels); // Check Y values std::vector > aDataSeriesYValues = getDataSeriesYValuesFromChartType(xChartType); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(aDataSeriesYValues.size()); for (const std::vector& aYValuesOfSeries : aDataSeriesYValues) { CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(aYValuesOfSeries.size()); OUString sYValuesOfSeries = doubleVectorToOneLineString(aYValuesOfSeries); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sYValuesOfSeries); } // Check source ranges for (size_t nIndex = 0; nIndex < aDataSeriesYValues.size(); ++nIndex) { Reference< chart2::data::XDataSequence > xDataSeq = getDataSequenceFromDocByRole(xChartDoc2, "values-x", nIndex); if (xDataSeq.is()) { OUString aXValuesSourceRange = xDataSeq->getSourceRangeRepresentation(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(aXValuesSourceRange); } xDataSeq.set(getDataSequenceFromDocByRole(xChartDoc2, "values-y", nIndex)); if (xDataSeq.is()) { OUString aYValuesSourceRange = xDataSeq->getSourceRangeRepresentation(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(aYValuesSourceRange); } xDataSeq.set(getDataSequenceFromDocByRole(xChartDoc2, "categories", nIndex)); if (xDataSeq.is()) { OUString aCategoriesSourceRange = xDataSeq->getSourceRangeRepresentation(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(aCategoriesSourceRange); } } } } DECLARE_DUMP_TEST(LegendTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "legend_on_right_side.ods", "legend_on_bottom.ods", "legend_on_left_side.ods", "legend_on_top.ods", "many_legend_entries.ods", "custom_legend_position.ods", "multiple_categories.ods" }; for (const OUString& aTestFile : aTestFiles) { setTestFileName(aTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); // Get legend shape uno::Reference xLegend = getShapeByName(xShapes, "CID/D=0:Legend="); CPPUNIT_ASSERT(xLegend.is()); // Check legend position and size awt::Point aLegendPosition = xLegend->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendPosition.Y, INT_EPS); awt::Size aLegendSize = xLegend->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendSize.Width, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendSize.Height, INT_EPS); // Check legend entries uno::Reference< chart2::XChartDocument > xChartDoc2(xChartDoc, UNO_QUERY_THROW); Reference xChartType = getChartTypeFromDoc(xChartDoc2, 0); CPPUNIT_ASSERT(xChartType.is()); std::vector > aDataSeriesYValues = getDataSeriesYValuesFromChartType(xChartType); size_t nLegendEntryCount = aDataSeriesYValues.size(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nLegendEntryCount); // Check legend entries geometry for (size_t nSeriesIndex = 0; nSeriesIndex < nLegendEntryCount; ++nSeriesIndex) { uno::Reference xLegendEntry = getShapeByName(xShapes, "CID/MultiClick/D=0:CS=0:CT=0:Series=" + OUString::number(nSeriesIndex) + ":LegendEntry=0"); CPPUNIT_ASSERT(xLegendEntry.is()); // Check position and size awt::Point aLegendEntryPosition = xLegendEntry->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendEntryPosition.X, 76.1); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendEntryPosition.Y, 38.1); awt::Size aLegendEntrySize = xLegendEntry->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendEntrySize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLegendEntrySize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xLegendEntryPropSet(xLegendEntry, UNO_QUERY_THROW); CPPUNIT_ASSERT(xLegendEntryPropSet.is()); drawing::HomogenMatrix3 aLegendEntryTransformation; xLegendEntryPropSet->getPropertyValue("Transformation") >>= aLegendEntryTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aLegendEntryTransformation); uno::Reference xLegendEntryContainer(xLegendEntry, UNO_QUERY_THROW); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(xLegendEntryContainer->getCount()); for (sal_Int32 nEntryGeometryElement = 1; nEntryGeometryElement < xLegendEntryContainer->getCount(); ++nEntryGeometryElement) { uno::Reference xLegendEntryGeom(xLegendEntryContainer->getByIndex(nEntryGeometryElement), UNO_QUERY_THROW); // Check geometry uno::Reference< drawing::XShapeDescriptor > xShapeDescriptor(xLegendEntryGeom, uno::UNO_QUERY_THROW); OUString sEntryGeomShapeType = xShapeDescriptor->getShapeType(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sEntryGeomShapeType); // Check display color Reference< beans::XPropertySet > xPropSet(xLegendEntryGeom, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); util::Color aEntryGeomColor = 0; xPropSet->getPropertyValue(UNO_NAME_FILLCOLOR) >>= aEntryGeomColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aEntryGeomColor)); } } // Check legend entries' text uno::Reference xLegendContainer(xLegend, UNO_QUERY_THROW); for (sal_Int32 i = 0; i < xLegendContainer->getCount(); ++i) { uno::Reference xShape(xLegendContainer->getByIndex(i), uno::UNO_QUERY); uno::Reference< drawing::XShapeDescriptor > xShapeDescriptor(xShape, uno::UNO_QUERY_THROW); OUString sShapeType = xShapeDescriptor->getShapeType(); if (sShapeType == "com.sun.star.drawing.TextShape") { uno::Reference xLegendEntryText = uno::Reference(xShape, uno::UNO_QUERY)->getText(); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(xLegendEntryText->getString()); } } } } DECLARE_DUMP_TEST(GridTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "vertical_grid.ods", "horizontal_grid.ods", "minor_grid.ods", "formated_grid_line.ods" }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); const std::vector aGridShapeNames = { "CID/D=0:CS=0:Axis=1,0:Grid=0", // Major vertical grid "CID/D=0:CS=0:Axis=0,0:Grid=0", // Major horizontal grid "CID/D=0:CS=0:Axis=1,0:Grid=0:SubGrid=0", // Minor vertical grid "CID/D=0:CS=0:Axis=0,0:Grid=0:SubGrid=0" // Minor horizontal grid }; for (const OUString& sGridShapeName : aGridShapeNames) { uno::Reference xGrid = getShapeByName(xShapes, sGridShapeName); if (xGrid.is()) { CPPUNIT_DUMP_ASSERT_NOTE(sGridShapeName); // Check position and size awt::Point aGridPosition = xGrid->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aGridPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aGridPosition.Y, INT_EPS); awt::Size aGridSize = xGrid->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aGridSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aGridSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xGrid, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aGridTransformation; xPropSet->getPropertyValue("Transformation") >>= aGridTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aGridTransformation); // Check line properties uno::Reference xIndexAccess(xGrid, UNO_QUERY_THROW); uno::Reference xGridLine(xIndexAccess->getByIndex(0), UNO_QUERY_THROW); Reference< beans::XPropertySet > xGridLinePropSet(xGridLine, UNO_QUERY_THROW); CPPUNIT_ASSERT(xGridLinePropSet.is()); // Line type drawing::LineDash aLineDash; xGridLinePropSet->getPropertyValue("LineDash") >>= aLineDash; OUString sGridLineDash = OUString::number(static_cast(aLineDash.Style)) + ";" + OUString::number(aLineDash.Dots) + ";" + OUString::number(aLineDash.DotLen) + OUString::number(aLineDash.Dashes) + ";" + OUString::number(aLineDash.DashLen) + ";" + OUString::number(aLineDash.Distance); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sGridLineDash); // Line color util::Color aLineColor = 0; xGridLinePropSet->getPropertyValue("LineColor") >>= aLineColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aLineColor)); // Line width sal_Int32 nLineWidth = 0; xGridLinePropSet->getPropertyValue("LineWidth") >>= nLineWidth; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nLineWidth); } } } } DECLARE_DUMP_TEST(AxisGeometryTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "default_formated_axis.ods", "axis_special_positioning.ods", "formated_axis_lines.ods", "rotated_axis_labels.ods" }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); const std::vector aAxisShapeNames = { "CID/D=0:CS=0:Axis=0,0", // X Axis "CID/D=0:CS=0:Axis=1,0", // Y Axis }; for (const OUString& sAxisShapeName : aAxisShapeNames) { uno::Reference xXAxis = getShapeByName(xShapes, sAxisShapeName); CPPUNIT_ASSERT(xXAxis.is()); CPPUNIT_DUMP_ASSERT_NOTE(sAxisShapeName); // Check position and size awt::Point aAxisPosition = xXAxis->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aAxisPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aAxisPosition.Y, INT_EPS); awt::Size aAxisSize = xXAxis->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aAxisSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aAxisSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xXAxis, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aAxisTransformation; xPropSet->getPropertyValue("Transformation") >>= aAxisTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aAxisTransformation); // Check line properties uno::Reference xIndexAccess(xXAxis, UNO_QUERY_THROW); sal_Int32 nAxisGeometriesCount = xIndexAccess->getCount(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nAxisGeometriesCount); uno::Reference xAxisLine(xIndexAccess->getByIndex(0), UNO_QUERY_THROW); Reference< beans::XPropertySet > xAxisLinePropSet(xAxisLine, UNO_QUERY_THROW); CPPUNIT_ASSERT(xAxisLinePropSet.is()); // Line type drawing::LineDash aLineDash; xAxisLinePropSet->getPropertyValue("LineDash") >>= aLineDash; OUString sAxisLineDash = OUString::number(static_cast(aLineDash.Style)) + ";" + OUString::number(aLineDash.Dots) + ";" + OUString::number(aLineDash.DotLen) + OUString::number(aLineDash.Dashes) + ";" + OUString::number(aLineDash.DashLen) + ";" + OUString::number(aLineDash.Distance); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sAxisLineDash); // Line color util::Color aAxisLineColor = 0; xAxisLinePropSet->getPropertyValue("LineColor") >>= aAxisLineColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aAxisLineColor)); // Line width sal_Int32 nAxisLineWidth = 0; xAxisLinePropSet->getPropertyValue("LineWidth") >>= nAxisLineWidth; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nAxisLineWidth); } } } DECLARE_DUMP_TEST(AxisLabelTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "default_formated_axis.ods", "rotated_axis_labels.ods", "formated_axis_labels.ods", "percent_stacked_column_chart.ods", }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); const std::vector aAxisShapeNames = { "CID/D=0:CS=0:Axis=0,0", // X Axis "CID/D=0:CS=0:Axis=1,0", // Y Axis }; for (const OUString& sAxisShapeName : aAxisShapeNames) { uno::Reference xXAxis = getShapeByName(xShapes, sAxisShapeName, // Axis occurs twice in chart xshape representation so need to get the one related to labels [](const uno::Reference& rXShape) -> bool { uno::Reference xAxisShapes(rXShape, uno::UNO_QUERY); CPPUNIT_ASSERT(xAxisShapes.is()); uno::Reference xChildShape(xAxisShapes->getByIndex(0), uno::UNO_QUERY); uno::Reference< drawing::XShapeDescriptor > xShapeDescriptor(xChildShape, uno::UNO_QUERY_THROW); return (xShapeDescriptor->getShapeType() == "com.sun.star.drawing.TextShape"); }); CPPUNIT_ASSERT(xXAxis.is()); CPPUNIT_DUMP_ASSERT_NOTE(sAxisShapeName); // Check label count uno::Reference xIndexAccess(xXAxis, UNO_QUERY_THROW); sal_Int32 nAxisLabelsCount = xIndexAccess->getCount(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nAxisLabelsCount); // Check labels's text, positioning and font properties for (sal_Int32 nLabelIndex = 0; nLabelIndex < nAxisLabelsCount; ++nLabelIndex) { // Check text uno::Reference xLabel(xIndexAccess->getByIndex(nLabelIndex), uno::UNO_QUERY); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(xLabel->getString()); // Check size and position uno::Reference xLabelShape(xLabel, uno::UNO_QUERY); awt::Point aLabelPosition = xLabelShape->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLabelPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLabelPosition.Y, INT_EPS); awt::Size aLabelSize = xLabelShape->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLabelSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aLabelSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xLabelShape, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aLabelTransformation; xPropSet->getPropertyValue("Transformation") >>= aLabelTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aLabelTransformation); // Check font color and height util::Color aLabelFontColor = 0; xPropSet->getPropertyValue("CharColor") >>= aLabelFontColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aLabelFontColor)); float fLabelFontHeight = 0.0f; xPropSet->getPropertyValue("CharHeight") >>= fLabelFontHeight; CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(fLabelFontHeight, EPS); } } } } DECLARE_DUMP_TEST(ColumnBarChartTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "normal_column_chart.ods", "stacked_column_chart.ods", "percent_stacked_column_chart.ods", "column_chart_small_spacing.ods", "normal_bar_chart.ods", "stacked_bar_chart.ods", "percent_stacked_bar_chart.ods", }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); uno::Reference< chart2::XChartDocument > xChartDoc2(xChartDoc, UNO_QUERY_THROW); Reference xChartType = getChartTypeFromDoc(xChartDoc2, 0); CPPUNIT_ASSERT(xChartType.is()); std::vector > aDataSeriesYValues = getDataSeriesYValuesFromChartType(xChartType); size_t nSeriesCount = aDataSeriesYValues.size(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nSeriesCount); for (size_t nSeries = 0; nSeries < nSeriesCount; ++nSeries) { uno::Reference xSeriesColumnsOrBars = getShapeByName(xShapes, "CID/D=0:CS=0:CT=0:Series=" + OUString::number(nSeries)); CPPUNIT_ASSERT(xSeriesColumnsOrBars.is()); CPPUNIT_DUMP_ASSERT_NOTE("Series " + OUString::number(nSeries) + " ColumnsOrBars"); // Check column/bar count in the series uno::Reference xIndexAccess(xSeriesColumnsOrBars, UNO_QUERY_THROW); sal_Int32 nColumnOrBarCountInSeries = xIndexAccess->getCount(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nColumnOrBarCountInSeries); // Check column/bar fill style and color Reference< beans::XPropertySet > xColumnOrBarPropSet(xIndexAccess->getByIndex(0), UNO_QUERY_THROW); drawing::FillStyle aSeriesColumnOrBarFillStyle; xColumnOrBarPropSet->getPropertyValue(UNO_NAME_FILLSTYLE) >>= aSeriesColumnOrBarFillStyle; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aSeriesColumnOrBarFillStyle)); util::Color aSeriesColumnOrBarFillColor = 0; xColumnOrBarPropSet->getPropertyValue(UNO_NAME_FILLCOLOR) >>= aSeriesColumnOrBarFillColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aSeriesColumnOrBarFillColor)); for (sal_Int32 nColumnOrBar = 0; nColumnOrBar < nColumnOrBarCountInSeries; ++nColumnOrBar) { uno::Reference xColumnOrBar(xIndexAccess->getByIndex(nColumnOrBar), UNO_QUERY_THROW); uno::Reference xNamedShape(xIndexAccess->getByIndex(nColumnOrBar), uno::UNO_QUERY); CPPUNIT_DUMP_ASSERT_NOTE(xNamedShape->getName()); // Check size and position awt::Point aColumnOrBarPosition = xColumnOrBar->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aColumnOrBarPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aColumnOrBarPosition.Y, INT_EPS); awt::Size aColumnOrBarSize = xColumnOrBar->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aColumnOrBarSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aColumnOrBarSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xColumnOrBar, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aColumnOrBarTransformation; xPropSet->getPropertyValue("Transformation") >>= aColumnOrBarTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aColumnOrBarTransformation); } } } } DECLARE_DUMP_TEST(ChartWallTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "chartwall_auto_adjust_with_titles.ods", "chartwall_auto_adjust_without_titles.ods", "chartwall_custom_positioning.ods" }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); uno::Reference xChartWall = getShapeByName(xShapes, "CID/DiagramWall="); CPPUNIT_ASSERT(xChartWall.is()); // Check position and size awt::Point aChartWallPosition = xChartWall->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aChartWallPosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aChartWallPosition.Y, INT_EPS); awt::Size aChartWallSize = xChartWall->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aChartWallSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aChartWallSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xChartWall, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aChartWallTransformation; xPropSet->getPropertyValue("Transformation") >>= aChartWallTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aChartWallTransformation); // Check fill properties drawing::FillStyle aChartWallFillStyle; xPropSet->getPropertyValue(UNO_NAME_FILLSTYLE) >>= aChartWallFillStyle; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aChartWallFillStyle)); util::Color aChartWallFillColor = 0; xPropSet->getPropertyValue(UNO_NAME_FILLCOLOR) >>= aChartWallFillColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aChartWallFillColor)); // Check line properties // Line type drawing::LineDash aLineDash; xPropSet->getPropertyValue("LineDash") >>= aLineDash; OUString sChartWallLineDash = OUString::number(static_cast(aLineDash.Style)) + ";" + OUString::number(aLineDash.Dots) + ";" + OUString::number(aLineDash.DotLen) + OUString::number(aLineDash.Dashes) + ";" + OUString::number(aLineDash.DashLen) + ";" + OUString::number(aLineDash.Distance); CPPUNIT_DUMP_ASSERT_STRINGS_EQUAL(sChartWallLineDash); // Line color util::Color aChartWallLineColor = 0; xPropSet->getPropertyValue("LineColor") >>= aChartWallLineColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aChartWallLineColor)); // Line width sal_Int32 nChartWallLineWidth = 0; xPropSet->getPropertyValue("LineWidth") >>= nChartWallLineWidth; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nChartWallLineWidth); } } DECLARE_DUMP_TEST(PieChartTest, Chart2DumpTest, false) { const std::vector aTestFiles = { "normal_pie_chart.ods", "rotated_pie_chart.ods", "exploded_pie_chart.ods", "donut_chart.ods", "pie_chart_many_slices.ods" }; for (const OUString& sTestFile : aTestFiles) { setTestFileName(sTestFile); load(getTestFileDirName(), getTestFileName()); uno::Reference< chart::XChartDocument > xChartDoc(getChartDocFromSheet(0, mxComponent), UNO_QUERY_THROW); uno::Reference xDrawPageSupplier(xChartDoc, uno::UNO_QUERY); uno::Reference xDrawPage = xDrawPageSupplier->getDrawPage(); uno::Reference xShapes(xDrawPage->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT(xShapes.is()); uno::Reference< chart2::XChartDocument > xChartDoc2(xChartDoc, UNO_QUERY_THROW); Reference xChartType = getChartTypeFromDoc(xChartDoc2, 0); CPPUNIT_ASSERT(xChartType.is()); std::vector > aDataSeriesYValues = getDataSeriesYValuesFromChartType(xChartType); size_t nSeriesCount = aDataSeriesYValues.size(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nSeriesCount); for (size_t nSeries = 0; nSeries < nSeriesCount; ++nSeries) { uno::Reference xSeriesSlices = getShapeByName(xShapes, "CID/D=0:CS=0:CT=0:Series=" + OUString::number(nSeries)); if (!xSeriesSlices.is()) break; // Normal pie chart displays only one series CPPUNIT_DUMP_ASSERT_NOTE("Series " + OUString::number(nSeries) + " slices"); // Check slice count in the series uno::Reference xIndexAccess(xSeriesSlices, UNO_QUERY_THROW); sal_Int32 nSlicesCountInSeries = xIndexAccess->getCount(); CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(nSlicesCountInSeries); // Check slices properties for (sal_Int32 nSlice = 0; nSlice < nSlicesCountInSeries; ++nSlice) { uno::Reference xSlice(xIndexAccess->getByIndex(nSlice), UNO_QUERY_THROW); uno::Reference xNamedShape(xIndexAccess->getByIndex(nSlice), uno::UNO_QUERY); OUString sName = xNamedShape->getName(); CPPUNIT_DUMP_ASSERT_NOTE(sName.copy(sName.lastIndexOf("/D=0"))); // Check size and position awt::Point aSlicePosition = xSlice->getPosition(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aSlicePosition.X, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aSlicePosition.Y, INT_EPS); awt::Size aSliceSize = xSlice->getSize(); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aSliceSize.Height, INT_EPS); CPPUNIT_DUMP_ASSERT_DOUBLES_EQUAL(aSliceSize.Width, INT_EPS); // Check transformation Reference< beans::XPropertySet > xPropSet(xSlice, UNO_QUERY_THROW); CPPUNIT_ASSERT(xPropSet.is()); drawing::HomogenMatrix3 aSliceTransformation; xPropSet->getPropertyValue("Transformation") >>= aSliceTransformation; CPPUNIT_DUMP_ASSERT_TRANSFORMATIONS_EQUAL(aSliceTransformation); // Check slice fill style and color drawing::FillStyle aSliceFillStyle; xPropSet->getPropertyValue(UNO_NAME_FILLSTYLE) >>= aSliceFillStyle; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aSliceFillStyle)); util::Color aSliceFillColor = 0; xPropSet->getPropertyValue(UNO_NAME_FILLCOLOR) >>= aSliceFillColor; CPPUNIT_DUMP_ASSERT_NUMBERS_EQUAL(static_cast(aSliceFillColor)); } } } } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */