/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include //#include // keep for testTableStylerPrSz #include #include #include #include class Test : public SwModelTestBase { public: Test() : SwModelTestBase("/sw/qa/extras/ooxmlexport/data/", "Office Open XML Text") {} protected: /** * Blacklist handling */ bool mustTestImportOf(const char* filename) const override { const char* aBlacklist[] = { "math-escape.docx", "math-mso2k7.docx", }; std::vector vBlacklist(aBlacklist, aBlacklist + SAL_N_ELEMENTS(aBlacklist)); // If the testcase is stored in some other format, it's pointless to test. return (OString(filename).endsWith(".docx") && std::find(vBlacklist.begin(), vBlacklist.end(), filename) == vBlacklist.end()); } /** * Validation handling */ bool mustValidate(const char* filename) const override { const char* aWhitelist[] = { "page-graphic-background.odt", "zoom.docx", "empty.odt", "fdo38244.docx", "comments-nested.odt" }; std::vector vWhitelist(aWhitelist, aWhitelist + SAL_N_ELEMENTS(aWhitelist)); return std::find(vWhitelist.begin(), vWhitelist.end(), filename) != vWhitelist.end(); } }; DECLARE_OOXMLEXPORT_TEST(testPageGraphicBackground, "page-graphic-background.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // No idea how the graphic background should be exported (seems there is no // way to do a non-tiling export to OOXML), but at least the background // color shouldn't be black. uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(-1), getProperty(xPageStyle, "BackColor")); } DECLARE_OOXMLEXPORT_EXPORTONLY_TEST(testCustomProperties, "custom-properties.docx") { // tdf#133377 tdf#103987 FILESAVE XLSX: Make sure the custom/core/application // file properties are stored correctly after roundtrip to .docx // Extended file properties - specific to Office package, // eg. docx - Number of Pages, pptx - Number of Slides xmlDocUniquePtr pXmlDoc = parseExport("docProps/app.xml"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Paragraphs", "1"); //assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Lines", "1"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Pages", "1"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Words", "3"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Characters", "22"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:CharactersWithSpaces", "24"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Company", "hhhhkompany"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Manager", "ffffmenadzer;iiiiisecondmanager"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:HyperlinkBase", "gggghiperlink"); //assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:DocSecurity", "2"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:AppVersion", "15.0000"); // Custom file properties - defined by user xmlDocUniquePtr pCustomXml = parseExport("docProps/custom.xml"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property", 9); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[1]", "name", "testDateProperty"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[1]/vt:filetime", "1982-04-19T10:00:00Z"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[3]", "name", "testNegativeNumberProperty"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[3]/vt:r8", "-100"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[4]", "name", "testNumberProperty"); //assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[4]/vt:i4", // "256"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[5]", "name", "testRealNumberProperty"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[5]/vt:r8", "-128.1"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[6]", "name", "testScientificNumber"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[6]/vt:r8", "1.23456789E+023"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[7]", "name", "testTextProperty"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[7]/vt:lpwstr", "testPropertyValue"); assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[8]", "name", "testYesNoProperty"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[8]/vt:bool", "1"); // Hidden Custom File Property. With Final set, MS Office notifies recipients that the document is final, and sets the document to read-only. assertXPath(pCustomXml, "/custom-properties:Properties/custom-properties:property[9]", "name", "_MarkAsFinal"); assertXPathContent(pCustomXml, "/custom-properties:Properties/custom-properties:property[9]/vt:bool", "1"); // Core file properties - common for all packages (eg. creation date, modify date) pXmlDoc = parseExport("docProps/core.xml"); assertXPathContent(pXmlDoc, "/cp:coreProperties/dc:creator", "Bartosz Kosiorek;secondauthor"); assertXPathContent(pXmlDoc, "/cp:coreProperties/dc:description", "cccckomentarzglowny"); assertXPathContent(pXmlDoc, "/cp:coreProperties/cp:lastPrinted", "2020-10-15T07:42:00Z"); assertXPathContent(pXmlDoc, "/cp:coreProperties/dcterms:created", "2020-10-14T16:23:00Z"); assertXPathContent(pXmlDoc, "/cp:coreProperties/cp:category", "eeeetokategoria"); assertXPathContent(pXmlDoc, "/cp:coreProperties/cp:version", "xxxxversion"); assertXPathContent(pXmlDoc, "/cp:coreProperties/cp:contentStatus", "ddddstatusnieznany"); assertXPathContent(pXmlDoc, "/cp:coreProperties/dc:identifier", "zzzzidentifier"); } DECLARE_OOXMLEXPORT_TEST(testZoom, "zoom.docx") { uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xViewSettingsSupplier(xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xPropertySet(xViewSettingsSupplier->getViewSettings()); sal_Int16 nValue = 0; xPropertySet->getPropertyValue("ZoomValue") >>= nValue; CPPUNIT_ASSERT_EQUAL(sal_Int16(42), nValue); // Validation test: order of elements were wrong. xmlDocUniquePtr pXmlDoc = parseExport("word/styles.xml"); if (!pXmlDoc) return; // Order was: rsid, next. int nNext = getXPathPosition(pXmlDoc, "/w:styles/w:style[3]", "next"); int nRsid = getXPathPosition(pXmlDoc, "/w:styles/w:style[3]", "rsid"); CPPUNIT_ASSERT(nNext < nRsid); pXmlDoc = parseExport("docProps/app.xml"); // One paragraph in the document. assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Paragraphs", "1"); assertXPathContent(pXmlDoc, "/extended-properties:Properties/extended-properties:Company", "Example Ltd"); } DECLARE_OOXMLEXPORT_TEST(defaultTabStopNotInStyles, "empty.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The default tab stop was mistakenly exported to a style. // xray ThisComponent.StyleFamilies(1)(0).ParaTabStop uno::Reference< container::XNameAccess > paragraphStyles = getStyles( "ParagraphStyles" ); uno::Sequence< style::TabStop > stops = getProperty< uno::Sequence< style::TabStop > >( paragraphStyles->getByName( "Standard" ), "ParaTabStops" ); // There actually be one tab stop, but it will be the default. CPPUNIT_ASSERT_EQUAL( static_cast(1), stops.getLength()); CPPUNIT_ASSERT_EQUAL( style::TabAlign_DEFAULT, stops[ 0 ].Alignment ); } DECLARE_OOXMLEXPORT_TEST(testFdo38244, "fdo38244.docx") { /* * Comments attached to a range was imported without the range, check for the annotation mark start/end positions. * * oParas = ThisComponent.Text.createEnumeration * oPara = oParas.nextElement * oRuns = oPara.createEnumeration * oRun = oRuns.nextElement * oRun = oRuns.nextElement 'Annotation * oRun = oRuns.nextElement * oRun = oRuns.nextElement 'AnnotationEnd * xray oRun.TextPortionType */ uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xParaEnumAccess(xTextDocument->getText(), uno::UNO_QUERY); uno::Reference xParaEnum = xParaEnumAccess->createEnumeration(); uno::Reference xRunEnumAccess(xParaEnum->nextElement(), uno::UNO_QUERY); uno::Reference xRunEnum = xRunEnumAccess->createEnumeration(); xRunEnum->nextElement(); uno::Reference xPropertySet(xRunEnum->nextElement(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("Annotation"), getProperty(xPropertySet, "TextPortionType")); xRunEnum->nextElement(); xPropertySet.set(xRunEnum->nextElement(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("AnnotationEnd"), getProperty(xPropertySet, "TextPortionType")); /* * Initials were not imported. * * oFields = ThisComponent.TextFields.createEnumeration * oField = oFields.nextElement * xray oField.Initials */ uno::Reference xTextFieldsSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xFieldsAccess(xTextFieldsSupplier->getTextFields()); uno::Reference xFields(xFieldsAccess->createEnumeration()); xPropertySet.set(xFields->nextElement(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("M"), getProperty(xPropertySet, "Initials")); /* * There was a fake empty paragraph at the end of the comment text. * * oFields = ThisComponent.TextFields.createEnumeration * oField = oFields.nextElement * oParas = oField.TextRange.createEnumeration * oPara = oParas.nextElement * oPara = oParas.nextElement */ xParaEnumAccess = getProperty< uno::Reference >(xPropertySet, "TextRange"); xParaEnum = xParaEnumAccess->createEnumeration(); xParaEnum->nextElement(); bool bCaught = false; try { xParaEnum->nextElement(); } catch (container::NoSuchElementException&) { bCaught = true; } CPPUNIT_ASSERT_EQUAL(true, bCaught); } DECLARE_OOXMLEXPORT_TEST(testCommentsNested, "comments-nested.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xOuter = getProperty< uno::Reference >(getRun(getParagraph(1), 2), "TextField"); CPPUNIT_ASSERT_EQUAL(OUString("Outer"), getProperty(xOuter, "Content")); uno::Reference xInner = getProperty< uno::Reference >(getRun(getParagraph(1), 4), "TextField"); CPPUNIT_ASSERT_EQUAL(OUString("Inner"), getProperty(xInner, "Content")); } DECLARE_OOXMLEXPORT_TEST(testMathEscape, "math-escape.docx") { CPPUNIT_ASSERT_EQUAL(OUString("\\{ left [ right ] left ( right ) \\}"), getFormula(getRun(getParagraph(1), 1))); } DECLARE_OOXMLEXPORT_TEST(testFdo51034, "fdo51034.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The problem was that the 'l' param of the HYPERLINK field was parsed with = "#", not += "#". CPPUNIT_ASSERT_EQUAL(OUString("http://Www.google.com/#a"), getProperty(getRun(getParagraph(1), 1), "HyperLinkURL")); } // Remove all spaces, as LO export/import may change that. // Replace symbol - (i.e. U+2212) with ASCII - , LO does this change and it shouldn't matter. static void CHECK_FORMULA(OUString const & expected, OUString const & actual) { CPPUNIT_ASSERT_EQUAL( expected.replaceAll( " ", "" ).replaceAll( OUString(u"\u2212"), "-" ), actual.replaceAll( " ", "" ).replaceAll( OUString(u"\u2212"), "-" )); } DECLARE_OOXMLEXPORT_TEST(testMathAccents, "math-accents.docx") { CHECK_FORMULA( "acute {a} grave {a} check {a} breve {a} circle {a} widevec {a} widetilde {a}" " widehat {a} dot {a} widevec {a} widevec {a} widetilde {a} underline {a}", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testMathD, "math-d.docx") { CHECK_FORMULA( "left (x mline y mline z right )", getFormula( getRun( getParagraph( 1 ), 1 ))); CHECK_FORMULA( "left (1 right )", getFormula( getRun( getParagraph( 1 ), 2 ))); CHECK_FORMULA( "left [2 right ]", getFormula( getRun( getParagraph( 1 ), 3 ))); CHECK_FORMULA( "left ldbracket 3 right rdbracket", getFormula( getRun( getParagraph( 1 ), 4 ))); CHECK_FORMULA( "left lline 4 right rline", getFormula( getRun( getParagraph( 1 ), 5 ))); CHECK_FORMULA( "left ldline 5 right rdline", getFormula( getRun( getParagraph( 1 ), 6 ))); CHECK_FORMULA( "left langle 6 right rangle", getFormula( getRun( getParagraph( 1 ), 7 ))); CHECK_FORMULA( "left langle a mline b right rangle", getFormula( getRun( getParagraph( 1 ), 8 ))); CHECK_FORMULA( "left ({x} over {y} right )", getFormula( getRun( getParagraph( 1 ), 9 ))); } DECLARE_OOXMLEXPORT_TEST(testMathEscaping, "math-escaping.docx") { CHECK_FORMULA( u"\u2212 \u221E < x < \u221E", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testMathLim, "math-lim.docx") { CHECK_FORMULA( u"lim from {x \u2192 1} {x}", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testMathMatrix, "math-matrix.docx") { CHECK_FORMULA( "left [matrix {1 # 2 ## 3 # 4} right ]", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testMathMso2k7, "math-mso2k7.docx") { CHECK_FORMULA( u"A = \u03C0 {r} ^ {2}", getFormula( getRun( getParagraph( 1 ), 1 ))); // TODO check the stack/binom difference // CHECK_FORMULA( "{left (x+a right )} ^ {n} = sum from {k=0} to {n} {left (binom {n} {k} right ) {x} ^ {k} {a} ^ {n-k}}", CHECK_FORMULA( "{left (x+a right )} ^ {n} = sum from {k=0} to {n} {left (stack {n # k} right ) {x} ^ {k} {a} ^ {n-k}}", getFormula( getRun( getParagraph( 2 ), 1 ))); CHECK_FORMULA( u"{left (1+x right )} ^ {n} =1+ {nx} over {1!} + {n left (n-1 right ) {x} ^ {2}} over {2!} +\u2026", getFormula( getRun( getParagraph( 3 ), 1 ))); // TODO check (cos/sin miss {}) // CHECK_FORMULA( "f left (x right ) = {a} rsub {0} + sum from {n=1} to {\xe2\x88\x9e} {left ({a} rsub {n} cos {{n\xcf\x80x} over {L}} + {b} rsub {n} sin {{n\xcf\x80x} over {L}} right )}", CHECK_FORMULA( u"f left (x right ) = {a} rsub {0} + sum from {n=1} to {\u221E} {left ({a} rsub {n} cos {n\u03C0x} over {L} + {b} rsub {n} sin {n\u03C0x} over {L} right )}", getFormula( getRun( getParagraph( 4 ), 1 ))); CHECK_FORMULA( "{a} ^ {2} + {b} ^ {2} = {c} ^ {2}", getFormula( getRun( getParagraph( 5 ), 1 ))); CHECK_FORMULA( u"x = {- b \u00B1 sqrt {{b} ^ {2} -4 ac}} over {2 a}", getFormula( getRun( getParagraph( 6 ), 1 ))); CHECK_FORMULA( u"{e} ^ {x} =1+ {x} over {1!} + {{x} ^ {2}} over {2!} + {{x} ^ {3}} over {3!} +\u2026, -\u221E to ", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testMathRad, "math-rad.docx") { CHECK_FORMULA( "sqrt {4}", getFormula( getRun( getParagraph( 1 ), 1 ))); CHECK_FORMULA( "nroot {3} {x + 1}", getFormula( getRun( getParagraph( 1 ), 2 ))); } DECLARE_OOXMLEXPORT_TEST(testMathSubscripts, "math-subscripts.docx") { CHECK_FORMULA( "{x} ^ {y} + {e} ^ {x}", getFormula( getRun( getParagraph( 1 ), 1 ))); CHECK_FORMULA( "{x} ^ {b}", getFormula( getRun( getParagraph( 1 ), 2 ))); CHECK_FORMULA( "{x} rsub {b}", getFormula( getRun( getParagraph( 1 ), 3 ))); CHECK_FORMULA( "{a} rsub {c} rsup {b}", getFormula( getRun( getParagraph( 1 ), 4 ))); CHECK_FORMULA( "{x} lsub {2} lsup {1}", getFormula( getRun( getParagraph( 1 ), 5 ))); CHECK_FORMULA( "{{x csup {6} csub {3}} lsub {4} lsup {5}} rsub {2} rsup {1}", getFormula( getRun( getParagraph( 1 ), 6 ))); } DECLARE_OOXMLEXPORT_TEST(testMathVerticalStacks, "math-vertical_stacks.docx") { CHECK_FORMULA( "{a} over {b}", getFormula( getRun( getParagraph( 1 ), 1 ))); CHECK_FORMULA( "{a} / {b}", getFormula( getRun( getParagraph( 2 ), 1 ))); // TODO check these // CHECK_FORMULA( "binom {a} {b}", getFormula( getRun( getParagraph( 3 ), 1 ))); // CHECK_FORMULA( "binom {a} {binom {b} {c}}", getFormula( getRun( getParagraph( 4 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testTable, "table.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // Make sure we write qFormat for well-known style names. xmlDocUniquePtr pXmlDocCT = parseExport("word/styles.xml"); CPPUNIT_ASSERT(pXmlDocCT); assertXPath(pXmlDocCT, "//w:style[@w:styleId='Normal']/w:qFormat", 1); } namespace { struct SingleLineBorders { sal_Int16 top, bottom, left, right; SingleLineBorders(int t=0, int b=0, int l=0, int r=0) : top(t), bottom(b), left(l), right(r) {} sal_Int16 getBorder(int i) const { switch (i) { case 0: return top; case 1: return bottom; case 2: return left; case 3: return right; default: assert(false); return 0; } } }; } DECLARE_OOXMLEXPORT_TEST(testTableBorders, "table-borders.docx") { uno::Reference xTablesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xTables(xTablesSupplier->getTextTables( ), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTables->getCount()); uno::Reference xTextTable (xTables->getByIndex(0), uno::UNO_QUERY); std::map cellBorders; cellBorders[OUString("A1")] = SingleLineBorders(106, 106, 106, 106); cellBorders[OUString("B1")] = SingleLineBorders(106, 0, 106, 35); cellBorders[OUString("C1")] = SingleLineBorders(106, 106, 35, 106); cellBorders[OUString("A2")] = SingleLineBorders(106, 35, 106, 0); cellBorders[OUString("B2")] = SingleLineBorders(0, 0, 0, 0); cellBorders[OUString("C2")] = SingleLineBorders(106, 106, 0, 106); cellBorders[OUString("A3")] = SingleLineBorders(35, 35, 106, 106); cellBorders[OUString("B3")] = SingleLineBorders(0, 106, 106, 106); cellBorders[OUString("C3")] = SingleLineBorders(106, 106, 106, 106); cellBorders[OUString("A4")] = SingleLineBorders(35, 106, 106, 35); cellBorders[OUString("B4")] = SingleLineBorders(106, 106, 35, 106); cellBorders[OUString("C4")] = SingleLineBorders(106, 106, 106, 106); const OUString borderNames[] = { OUString("TopBorder"), OUString("BottomBorder"), OUString("LeftBorder"), OUString("RightBorder"), }; uno::Sequence const cells = xTextTable->getCellNames(); sal_Int32 nLength = cells.getLength(); CPPUNIT_ASSERT_EQUAL(static_cast(cellBorders.size()), nLength); for (sal_Int32 i = 0; i < nLength; ++i) { uno::Reference xCell = xTextTable->getCellByName(cells[i]); uno::Reference< beans::XPropertySet > xPropSet(xCell, uno::UNO_QUERY_THROW); const SingleLineBorders& borders = cellBorders[cells[i]]; for (sal_Int32 j = 0; j < 4; ++j) { uno::Any aBorder = xPropSet->getPropertyValue(borderNames[j]); table::BorderLine aBorderLine; if (aBorder >>= aBorderLine) { std::stringstream message; message << cells[i] << "'s " << borderNames[j] << " is incorrect"; CPPUNIT_ASSERT_EQUAL_MESSAGE(message.str(), borders.getBorder(j), aBorderLine.OuterLineWidth); } } } } DECLARE_OOXMLEXPORT_TEST(testFdo51550, "fdo51550.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // The problem was that we lacked the fallback to export the replacement // graphic for OLE objects. But we can actually export the OLE itself now, // so check that instead. uno::Reference xTextEmbeddedObjectsSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xEmbeddedObjects(xTextEmbeddedObjectsSupplier->getEmbeddedObjects(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xEmbeddedObjects->getCount()); xmlDocUniquePtr pXmlDocCT = parseExport("[Content_Types].xml"); if (!pXmlDocCT) return; // initial import assertXPath(pXmlDocCT, "/ContentType:Types/ContentType:Override[@PartName='/word/embeddings/oleObject1.xlsx']", "ContentType", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); // check the rels too xmlDocUniquePtr pXmlDocRels = parseExport("word/_rels/document.xml.rels"); assertXPath(pXmlDocRels, "/rels:Relationships/rels:Relationship[@Target='embeddings/oleObject1.xlsx']", "Type", "http://schemas.openxmlformats.org/officeDocument/2006/relationships/package"); // check the content too xmlDocUniquePtr pXmlDocContent = parseExport("word/document.xml"); assertXPath(pXmlDocContent, "/w:document/w:body/w:p/w:r/w:object/o:OLEObject", "ProgID", "Excel.Sheet.12"); } DECLARE_OOXMLEXPORT_TEST(test1Table1Page, "1-table-1-page.docx") { // 2 problem for this document after export: // - invalid sectPr inserted at the beginning of the page // - font of empty cell is not preserved, leading to change in rows height CPPUNIT_ASSERT_EQUAL(1, getPages()); } DECLARE_OOXMLEXPORT_TEST(testTextFrames, "textframes.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The frames were simply missing, so let's check if all 3 frames were imported back. CPPUNIT_ASSERT_EQUAL(3, getShapes()); } DECLARE_OOXMLEXPORT_TEST(testTextFrameBorders, "textframe-borders.docx") { uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); if (xIndexAccess->getCount()) { // After import, a TextFrame is created by the VML import. uno::Reference xFrame(xIndexAccess->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xD99594), getProperty(xFrame, "BackColor")); table::BorderLine2 aBorder = getProperty(xFrame, "TopBorder"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xC0504D), aBorder.Color); CPPUNIT_ASSERT_EQUAL(sal_uInt32(35), aBorder.LineWidth); table::ShadowFormat aShadowFormat = getProperty(xFrame, "ShadowFormat"); CPPUNIT_ASSERT_EQUAL(table::ShadowLocation_BOTTOM_RIGHT, aShadowFormat.Location); CPPUNIT_ASSERT_EQUAL(sal_Int16(48), aShadowFormat.ShadowWidth); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x622423), aShadowFormat.Color); } else { // After export and import, the result is a shape. uno::Reference xShape(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xD99594), getProperty(xShape, "FillColor")); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xC0504D), getProperty(xShape, "LineColor")); CPPUNIT_ASSERT_EQUAL(sal_Int32(35), getProperty(xShape, "LineWidth")); CPPUNIT_ASSERT_EQUAL(sal_Int32(48), getProperty(xShape, "ShadowXDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(48), getProperty(xShape, "ShadowYDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x622423), getProperty(xShape, "ShadowColor")); } } DECLARE_OOXMLEXPORT_TEST(testTextframeGradient, "textframe-gradient.docx") { CPPUNIT_ASSERT_EQUAL(2, getShapes()); uno::Reference xFrame(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_GRADIENT, getProperty(xFrame, "FillStyle")); awt::Gradient aGradient = getProperty(xFrame, "FillGradient"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xC0504D), aGradient.StartColor); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xD99594), aGradient.EndColor); CPPUNIT_ASSERT_EQUAL(awt::GradientStyle_AXIAL, aGradient.Style); xFrame.set(getShape(2), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_GRADIENT, getProperty(xFrame, "FillStyle")); aGradient = getProperty(xFrame, "FillGradient"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x000000), aGradient.StartColor); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x666666), aGradient.EndColor); CPPUNIT_ASSERT_EQUAL(awt::GradientStyle_AXIAL, aGradient.Style); // Left / right margin was incorrect: the attribute was missing and we // didn't have the right default (had 0 instead of the below one). CPPUNIT_ASSERT_EQUAL(sal_Int32(318), getProperty(xFrame, "LeftMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(318), getProperty(xFrame, "RightMargin")); } DECLARE_OOXMLEXPORT_EXPORTONLY_TEST(testCellBtlr, "cell-btlr.docx") { /* * The problem was that the exporter didn't mirror the workaround of the * importer, regarding the btLr text direction: the token was completely missing in the output. */ xmlDocUniquePtr pXmlDoc = parseExport(); assertXPath(pXmlDoc, "/w:document/w:body/w:tbl/w:tr/w:tc/w:tcPr/w:textDirection", "val", "btLr"); } DECLARE_OOXMLEXPORT_TEST(testTableStylerPrSz, "table-style-rPr-sz.docx") { // Verify that font size inside the table is 20pt, despite the sz attribute in the table size. // Also check that other rPr attribute are used: italic, bold, underline // Office has the same behavior uno::Reference xTextTablesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xTables(xTextTablesSupplier->getTextTables(), uno::UNO_QUERY); uno::Reference xTable(xTables->getByIndex(0), uno::UNO_QUERY); uno::Reference xCell(xTable->getCellByName("A1"), uno::UNO_QUERY); uno::Reference xParaEnumAccess(xCell->getText(), uno::UNO_QUERY); uno::Reference xParaEnum = xParaEnumAccess->createEnumeration(); uno::Reference xPara(xParaEnum->nextElement(), uno::UNO_QUERY); // disabled temporarily, next commit enables it again #if 0 CPPUNIT_ASSERT_EQUAL(20.f, getProperty(getRun(xPara, 1), "CharHeight")); #endif // CPPUNIT_ASSERT_EQUAL(awt::FontUnderline::SINGLE, getProperty(getRun(xPara, 1), "CharUnderline")); // CPPUNIT_ASSERT_EQUAL(awt::FontWeight::BOLD, getProperty(getRun(xPara, 1), "CharWeight")); // CPPUNIT_ASSERT_EQUAL(awt::FontSlant_ITALIC, getProperty(getRun(xPara, 1), "CharPosture")); } DECLARE_OOXMLEXPORT_TEST(testMathLiteral, "math-literal.docx") { CHECK_FORMULA( u"iiint from {V} to {\"div\" \"F\"} dV= llint from {S} to {\"F\" \u2219 \"n \" dS}", getFormula( getRun( getParagraph( 1 ), 1 ))); } DECLARE_OOXMLEXPORT_TEST(testFdo48557, "fdo48557.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // Inner margins of the textframe wasn't exported. uno::Reference xFrame(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(150), getProperty(xFrame, "TextLeftDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(150), getProperty(xFrame, "TextRightDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(150), getProperty(xFrame, "TextUpperDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(150), getProperty(xFrame, "TextLowerDistance")); } DECLARE_OOXMLEXPORT_TEST(testI120928, "i120928.docx") { // w:numPicBullet was ignored, leading to missing graphic bullet in numbering. uno::Reference xPropertySet(getStyles("NumberingStyles")->getByName("WWNum1"), uno::UNO_QUERY); uno::Reference xLevels(xPropertySet->getPropertyValue("NumberingRules"), uno::UNO_QUERY); uno::Sequence aProps; xLevels->getByIndex(0) >>= aProps; // 1st level uno::Reference xBitmap; sal_Int16 nNumberingType = -1; for (beans::PropertyValue const & rProp : std::as_const(aProps)) { if (rProp.Name == "NumberingType") nNumberingType = rProp.Value.get(); else if (rProp.Name == "GraphicBitmap") xBitmap = rProp.Value.get>(); } CPPUNIT_ASSERT_EQUAL(style::NumberingType::BITMAP, nNumberingType); CPPUNIT_ASSERT(xBitmap.is()); } DECLARE_OOXMLEXPORT_TEST(testFdo64826, "fdo64826.docx") { // 'Track-Changes' (Track Revisions) wasn't exported. CPPUNIT_ASSERT_EQUAL(true, getProperty(mxComponent, "RecordChanges")); // 'Show-Changes' should not be exported - default is true. if (xmlDocUniquePtr pXmlSettings = parseExport("word/settings.xml")) { assertXPath(pXmlSettings, "/w:settings/w:revisionView", 0); } } DECLARE_OOXMLEXPORT_TEST(testPageBackground, "page-background.docx") { // 'Document Background' wasn't exported. uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x92D050), getProperty(xPageStyle, "BackColor")); } DECLARE_OOXMLEXPORT_TEST(testFdo65265, "fdo65265.docx") { // Redline (tracked changes) of text formatting were not exported uno::Reference xParagraph1 = getParagraph(1); uno::Reference xParagraph2 = getParagraph(2); CPPUNIT_ASSERT_EQUAL(OUString("Format"), getProperty(getRun(xParagraph1, 3), "RedlineType")); CPPUNIT_ASSERT_EQUAL(OUString("Format"), getProperty(getRun(xParagraph2, 2), "RedlineType")); } DECLARE_OOXMLEXPORT_TEST(testFdo65655, "fdo65655.docx") { // The problem was that the DOCX had a non-blank odd footer and a blank even footer // The 'Different Odd & Even Pages' was turned on // However - LO assumed that because the 'even' footer is blank - it should ignore the 'Different Odd & Even Pages' flag // So it did not import it and did not export it uno::Reference xPropertySet(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); bool bValue = false; xPropertySet->getPropertyValue("HeaderIsShared") >>= bValue; CPPUNIT_ASSERT_EQUAL(false, bValue); xPropertySet->getPropertyValue("FooterIsShared") >>= bValue; CPPUNIT_ASSERT_EQUAL(false, bValue); } DECLARE_OOXMLEXPORT_TEST(testFDO63053, "fdo63053.docx") { uno::Reference xDocumentPropertiesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties = xDocumentPropertiesSupplier->getDocumentProperties(); CPPUNIT_ASSERT_EQUAL(OUString("test1&test2"), xDocumentProperties->getTitle()); CPPUNIT_ASSERT_EQUAL(OUString("test1&test2"), xDocumentProperties->getSubject()); } DECLARE_OOXMLEXPORT_TEST(testWatermark, "watermark.docx") { uno::Reference xShape(getShape(1), uno::UNO_QUERY); // 1st problem: last character was missing CPPUNIT_ASSERT_EQUAL(OUString("SAMPLE"), xShape->getString()); const uno::Sequence aProps = getProperty< uno::Sequence >(xShape, "CustomShapeGeometry"); bool bFound = false; for (beans::PropertyValue const & prop : aProps) if (prop.Name == "TextPath") bFound = true; // 2nd problem: v:textpath wasn't imported CPPUNIT_ASSERT_EQUAL(true, bFound); // 3rd problem: rotation angle was 315, not 45. CPPUNIT_ASSERT_EQUAL(sal_Int32(45 * 100), getProperty(xShape, "RotateAngle")); // 4th problem: mso-position-vertical-relative:margin was ignored, VertOrientRelation was text::RelOrientation::FRAME. CPPUNIT_ASSERT_EQUAL(text::RelOrientation::PAGE_PRINT_AREA, getProperty(xShape, "VertOrientRelation")); // These problems were in the exporter // The textpath wasn't semi-transparent. CPPUNIT_ASSERT_EQUAL(sal_Int16(50), getProperty(xShape, "FillTransparence")); // The textpath had a stroke. CPPUNIT_ASSERT_EQUAL(drawing::LineStyle_NONE, getProperty(xShape, "LineStyle")); } DECLARE_OOXMLEXPORT_TEST(testWatermarkFont, "watermark-font.docx") { uno::Reference xShape(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("TestFont"), xShape->getString()); uno::Reference xPropertySet(xShape, uno::UNO_QUERY); OUString aFont; float nFontSize; // Check font family CPPUNIT_ASSERT(xPropertySet->getPropertyValue("CharFontName") >>= aFont); CPPUNIT_ASSERT_EQUAL(OUString("DejaVu Serif"), aFont); // Check font size CPPUNIT_ASSERT(xPropertySet->getPropertyValue("CharHeight") >>= nFontSize); CPPUNIT_ASSERT_EQUAL(float(72), nFontSize); } DECLARE_OOXMLEXPORT_TEST(testFdo43093, "fdo43093.docx") { // The problem was that the alignment are not exchange when the paragraph are RTL. uno::Reference xParaRtlLeft(getParagraph( 1, "RTL Left")); sal_Int32 nRtlLeft = getProperty< sal_Int32 >( xParaRtlLeft, "ParaAdjust" ); // test the text Direction value for the pragraph sal_Int16 nRLDir = getProperty< sal_Int32 >( xParaRtlLeft, "WritingMode" ); uno::Reference xParaRtlRight(getParagraph( 3, "RTL Right")); sal_Int32 nRtlRight = getProperty< sal_Int32 >( xParaRtlRight, "ParaAdjust" ); sal_Int16 nRRDir = getProperty< sal_Int32 >( xParaRtlRight, "WritingMode" ); uno::Reference xParaLtrLeft(getParagraph( 5, "LTR Left")); sal_Int32 nLtrLeft = getProperty< sal_Int32 >( xParaLtrLeft, "ParaAdjust" ); sal_Int16 nLLDir = getProperty< sal_Int32 >( xParaLtrLeft, "WritingMode" ); uno::Reference xParaLtrRight(getParagraph( 7, "LTR Right")); sal_Int32 nLtrRight = getProperty< sal_Int32 >( xParaLtrRight, "ParaAdjust" ); sal_Int16 nLRDir = getProperty< sal_Int32 >( xParaLtrRight, "WritingMode" ); // this will test the both the text direction and alignment for each paragraph CPPUNIT_ASSERT_EQUAL( sal_Int32 (style::ParagraphAdjust_LEFT), nRtlLeft); CPPUNIT_ASSERT_EQUAL(text::WritingMode2::RL_TB, nRLDir); CPPUNIT_ASSERT_EQUAL( sal_Int32 (style::ParagraphAdjust_RIGHT), nRtlRight); CPPUNIT_ASSERT_EQUAL(text::WritingMode2::RL_TB, nRRDir); CPPUNIT_ASSERT_EQUAL( sal_Int32 (style::ParagraphAdjust_LEFT), nLtrLeft); CPPUNIT_ASSERT_EQUAL(text::WritingMode2::LR_TB, nLLDir); CPPUNIT_ASSERT_EQUAL( sal_Int32 (style::ParagraphAdjust_RIGHT), nLtrRight); CPPUNIT_ASSERT_EQUAL(text::WritingMode2::LR_TB, nLRDir); } DECLARE_OOXMLEXPORT_TEST(testFdo64238_a, "fdo64238_a.docx") { // The problem was that when 'Show Only Odd Footer' was marked in Word and the Even footer *was filled* // then LO would still import the Even footer and concatenate it to the odd footer. // This case specifically is for : // 'Blank Odd Footer' with 'Non-Blank Even Footer' when 'Show Only Odd Footer' is marked in Word // In this case the imported footer in LO was supposed to be blank, but instead was the 'even' footer uno::Reference xFooterText = getProperty< uno::Reference >(getStyles("PageStyles")->getByName("Standard"), "FooterText"); uno::Reference< text::XTextRange > xFooterParagraph = getParagraphOfText( 1, xFooterText ); uno::Reference xRunEnumAccess(xFooterParagraph, uno::UNO_QUERY); uno::Reference xRunEnum = xRunEnumAccess->createEnumeration(); sal_Int32 numOfRuns = 0; while (xRunEnum->hasMoreElements()) { xRunEnum->nextElement(); numOfRuns++; } CPPUNIT_ASSERT_EQUAL(sal_Int32(2), numOfRuns); } DECLARE_OOXMLEXPORT_TEST(testFdo64238_b, "fdo64238_b.docx") { // The problem was that when 'Show Only Odd Footer' was marked in Word and the Even footer *was filled* // then LO would still import the Even footer and concatenate it to the odd footer. // This case specifically is for : // 'Non-Blank Odd Footer' with 'Non-Blank Even Footer' when 'Show Only Odd Footer' is marked in Word // In this case the imported footer in LO was supposed to be just the odd footer, but instead was the 'odd' and 'even' footers concatenated uno::Reference xFooterText = getProperty< uno::Reference >(getStyles("PageStyles")->getByName("Standard"), "FooterText"); uno::Reference< text::XTextRange > xFooterParagraph = getParagraphOfText( 1, xFooterText ); uno::Reference xRunEnumAccess(xFooterParagraph, uno::UNO_QUERY); uno::Reference xRunEnum = xRunEnumAccess->createEnumeration(); sal_Int32 numOfRuns = 0; while (xRunEnum->hasMoreElements()) { xRunEnum->nextElement(); numOfRuns++; } CPPUNIT_ASSERT_EQUAL(sal_Int32(5), numOfRuns); } DECLARE_OOXMLEXPORT_TEST(testFdo56679, "fdo56679.docx") { // The problem was that the DOCX importer and exporter did not handle the 'color' of an underline // (not the color of the text, the color of the underline itself) uno::Reference< text::XTextRange > xParagraph = getParagraph( 1 ); uno::Reference< text::XTextRange > xText = getRun( xParagraph, 2, "This is a simple sentence."); CPPUNIT_ASSERT_EQUAL(true, getProperty(xText, "CharUnderlineHasColor")); CPPUNIT_ASSERT_EQUAL(sal_Int32(0xFF0000), getProperty(xText, "CharUnderlineColor")); } DECLARE_OOXMLEXPORT_TEST(testFdo65400, "fdo65400.docx") { // The problem was that if in Word you choose 'Character Shading' - then the text portion // is marked with 'w:shd val=pct15'. LO did not store this value and so when importing and exporting // this value was lost (and so Word did not show 'Character Shading' was on) uno::Reference< text::XTextRange > paragraph1 = getParagraph( 1 ); uno::Reference< text::XTextRange > shaded = getRun( paragraph1, 2, "normal" ); CPPUNIT_ASSERT_EQUAL( sal_Int32( 0x0026 ), getProperty< sal_Int32 >( shaded, "CharShadingValue" )); CPPUNIT_ASSERT_EQUAL( sal_Int32( 0xd8d8d8 ), getProperty< sal_Int32 >( shaded, "CharBackColor" )); } DECLARE_OOXMLEXPORT_TEST(testFdo66543, "fdo66543.docx") { // The problem was that when importing DOCX with 'line numbers' - the 'start value' was imported // but nothing was done with it. uno::Reference< text::XTextRange > paragraph1 = getParagraph( 1 ); CPPUNIT_ASSERT_EQUAL(sal_Int32(2), getProperty(paragraph1, "ParaLineNumberStartValue")); } DECLARE_OOXMLEXPORT_TEST(testN822175, "n822175.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xFrame(getShape(1), uno::UNO_QUERY); // Was text::WrapTextMode_THROUGH, due to missing Surround handling in the exporter. CPPUNIT_ASSERT_EQUAL(text::WrapTextMode_PARALLEL, getProperty(xFrame, "Surround")); } DECLARE_OOXMLEXPORT_TEST(testFdo66688, "fdo66688.docx") { // The problem was that TextFrame imported and exported the wrong value for transparency // (was stored as 'FillTransparence' instead of 'BackColorTransparency' uno::Reference xFrame(getShape(2), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL( sal_Int32( 80 ), getProperty< sal_Int32 >( xFrame, "FillTransparence" ) ); } DECLARE_OOXMLEXPORT_TEST(testFdo66773, "fdo66773.docx") { // The problem was the line spacing was interpreted by Word as 'Multiple 1.08' if no default settings were written. // Now after the 'docDefaults' section is written in - there is no more problem. // (Word does not try to calculate some arbitrary value for line spacing). uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xParaEnumAccess(xTextDocument->getText(), uno::UNO_QUERY); uno::Reference xParaEnum = xParaEnumAccess->createEnumeration(); CPPUNIT_ASSERT(xParaEnum->hasMoreElements()); style::LineSpacing alineSpacing = getProperty(xParaEnum->nextElement(), "ParaLineSpacing"); CPPUNIT_ASSERT_EQUAL(style::LineSpacingMode::PROP, alineSpacing.Mode); CPPUNIT_ASSERT_EQUAL(static_cast(100), static_cast(alineSpacing.Height)); } DECLARE_OOXMLEXPORT_TEST(testFdo58577, "fdo58577.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The second frame was simply missing, so let's check if both frames were imported back. CPPUNIT_ASSERT_EQUAL(2, getShapes()); } DECLARE_OOXMLEXPORT_TEST(testBnc581614, "bnc581614.doc") { uno::Reference xFrame(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_NONE, getProperty(xFrame, "FillStyle")); } DECLARE_OOXMLEXPORT_TEST(testFdo66929, "fdo66929.docx") { // The problem was that the default 'inset' attribute of the 'textbox' node was exported incorrectly. // A node like '' was exported back as '' // This is wrong because the original node denotes a specific 'left' inset, and a default 'top','right','bottom' inset uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); if (xIndexAccess->getCount()) { // VML import -> TextFrame uno::Reference xFrame(xIndexAccess->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL( sal_Int32( 0 ) , getProperty< sal_Int32 >( xFrame, "LeftBorderDistance" ) ); CPPUNIT_ASSERT_EQUAL( sal_Int32( 127 ), getProperty< sal_Int32 >( xFrame, "TopBorderDistance" ) ); CPPUNIT_ASSERT_EQUAL( sal_Int32( 254 ), getProperty< sal_Int32 >( xFrame, "RightBorderDistance" ) ); CPPUNIT_ASSERT_EQUAL( sal_Int32( 127 ), getProperty< sal_Int32 >( xFrame, "BottomBorderDistance" ) ); } else { // drawingML import -> shape with TextBox uno::Reference xShape(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(0), getProperty(xShape, "TextLeftDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(127), getProperty(xShape, "TextUpperDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(254), getProperty(xShape, "TextRightDistance")); CPPUNIT_ASSERT_EQUAL(sal_Int32(127), getProperty(xShape, "TextLowerDistance")); } } DECLARE_OOXMLEXPORT_EXPORTONLY_TEST(testPageBorderSpacingExportCase2, "page-borders-export-case-2.docx") { // The problem was that the exporter didn't mirror the workaround of the // importer, regarding the page border's spacing : the // and the inner nodes like // // The exporter ALWAYS exported 'w:offsetFrom="text"' even when the spacing values where too large // for Word to handle (larger than 31 points) xmlDocUniquePtr pXmlDoc = parseExport(); // Assert the XPath expression - page borders assertXPath(pXmlDoc, "/w:document/w:body/w:sectPr/w:pgBorders", "offsetFrom", "page"); // Assert the XPath expression - 'left' border assertXPath(pXmlDoc, "/w:document/w:body/w:sectPr/w:pgBorders/w:left", "space", "24"); // Assert the XPath expression - 'right' border assertXPath(pXmlDoc, "/w:document/w:body/w:sectPr/w:pgBorders/w:right", "space", "24"); } DECLARE_OOXMLEXPORT_TEST(testFdo66145, "fdo66145.docx") { // The Writer ignored the 'First Is Shared' flag CPPUNIT_ASSERT_EQUAL(OUString("This is the FIRST page header."), parseDump("/root/page[1]/header/txt/text()")); CPPUNIT_ASSERT_EQUAL( OUString("This is the header for the REST OF THE FILE."), parseDump("/root/page[2]/header/txt/text()")); CPPUNIT_ASSERT_EQUAL( OUString("This is the header for the REST OF THE FILE."), parseDump("/root/page[3]/header/txt/text()")); } DECLARE_OOXMLEXPORT_EXPORTONLY_TEST(testGrabBag, "grabbag.docx") { // w:mirrorIndents was lost on roundtrip, now should be handled as a grab bag property xmlDocUniquePtr pXmlDoc = parseExport(); assertXPath(pXmlDoc, "/w:document/w:body/w:p/w:pPr/w:mirrorIndents"); } DECLARE_OOXMLEXPORT_TEST(testFdo66781, "fdo66781.docx") { // The problem was that bullets with level=0 were shown in LO as normal bullets, // and when saved back to DOCX were saved with level=1 (so hidden bullets became visible) uno::Reference xPropertySet(getStyles("NumberingStyles")->getByName("WWNum1"), uno::UNO_QUERY); uno::Reference xLevels(xPropertySet->getPropertyValue("NumberingRules"), uno::UNO_QUERY); uno::Sequence aProps; xLevels->getByIndex(0) >>= aProps; // 1st level for (beans::PropertyValue const & rProp : std::as_const(aProps)) { if (rProp.Name == "BulletChar") { CPPUNIT_ASSERT_EQUAL(OUString("\x0", 1, RTL_TEXTENCODING_ASCII_US), rProp.Value.get()); return; } } // Shouldn't reach here CPPUNIT_FAIL("Did not find bullet with level 0"); } DECLARE_OOXMLEXPORT_TEST(testFdo60990, "fdo60990.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // The shape had no background, no paragraph adjust and no font color. uno::Reference xShape(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x00CFE7F5), getProperty(xShape, "FillColor")); uno::Reference xText = uno::Reference(xShape, uno::UNO_QUERY_THROW)->getText(); uno::Reference xParagraph = getParagraphOfText(1, xText); CPPUNIT_ASSERT_EQUAL(style::ParagraphAdjust_CENTER, static_cast(getProperty(xParagraph, "ParaAdjust"))); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x00FF00), getProperty(getRun(xParagraph, 1), "CharColor")); } DECLARE_OOXMLEXPORT_TEST(testFdo65718, "fdo65718.docx") { // The problem was that the exporter always exported values of "0" for an images distance from text. // the actual attributes where 'distT', 'distB', 'distL', 'distR' uno::Reference xPropertySet(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(oox::drawingml::convertEmuToHmm(0), getProperty(xPropertySet, "TopMargin") ); CPPUNIT_ASSERT_EQUAL(oox::drawingml::convertEmuToHmm(0), getProperty(xPropertySet, "BottomMargin") ); // 'getProperty' return 318 (instead of 317.5) // I think this is because it returns an integer, instead of a float. // The actual exporting to DOCX exports the correct value (114300 = 317.5 * 360) // The exporting to DOCX uses the 'SvxLRSpacing' that stores the value in TWIPS (180 TWIPS) // However, the 'LeftMargin' property is an integer property that holds that value in 'MM100' (should hold 317.5, but it is 318) CPPUNIT_ASSERT_EQUAL(oox::drawingml::convertEmuToHmm(114300), getProperty(xPropertySet, "LeftMargin") ); CPPUNIT_ASSERT_EQUAL(oox::drawingml::convertEmuToHmm(114300), getProperty(xPropertySet, "RightMargin") ); } DECLARE_OOXMLEXPORT_TEST(testFdo64350, "fdo64350.docx") { // The problem was that page border shadows were not exported table::ShadowFormat aShadow = getProperty(getStyles("PageStyles")->getByName("Standard"), "ShadowFormat"); CPPUNIT_ASSERT_EQUAL(table::ShadowLocation_BOTTOM_RIGHT, aShadow.Location); } DECLARE_OOXMLEXPORT_TEST(testFdo67013, "fdo67013.docx") { /* * The problem was that borders inside headers \ footers were not exported * This was checked in xray using these commands: * * xHeaderText = ThisComponent.getStyleFamilies().getByName("PageStyles").getByName("Standard").HeaderText * xHeaderEnum = xHeaderText.createEnumeration() * xHeaderFirstParagraph = xHeaderEnum.nextElement() * xHeaderBottomBorder = xHeaderFirstParagraph.BottomBorder * * xFooterText = ThisComponent.getStyleFamilies().getByName("PageStyles").getByName("Standard").FooterText * xFooterEnum = xFooterText.createEnumeration() * xFooterFirstParagraph = xFooterEnum.nextElement() * xFooterTopBorder = xFooterFirstParagraph.TopBorder */ uno::Reference xHeaderText = getProperty< uno::Reference >(getStyles("PageStyles")->getByName("Standard"), "HeaderText"); uno::Reference< text::XTextRange > xHeaderParagraph = getParagraphOfText( 1, xHeaderText ); table::BorderLine2 aHeaderBottomBorder = getProperty(xHeaderParagraph, "BottomBorder"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x622423), aHeaderBottomBorder.Color); CPPUNIT_ASSERT_EQUAL(sal_Int16(106), aHeaderBottomBorder.InnerLineWidth); CPPUNIT_ASSERT_EQUAL(sal_Int16(26), aHeaderBottomBorder.LineDistance); CPPUNIT_ASSERT_EQUAL(sal_Int16(7), aHeaderBottomBorder.LineStyle); CPPUNIT_ASSERT_EQUAL(sal_uInt32(159), aHeaderBottomBorder.LineWidth); CPPUNIT_ASSERT_EQUAL(sal_Int16(26), aHeaderBottomBorder.OuterLineWidth); uno::Reference xFooterText = getProperty< uno::Reference >(getStyles("PageStyles")->getByName("Standard"), "FooterText"); uno::Reference< text::XTextRange > xFooterParagraph = getParagraphOfText( 1, xFooterText ); table::BorderLine2 aFooterTopBorder = getProperty(xFooterParagraph, "TopBorder"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x622423), aFooterTopBorder.Color); CPPUNIT_ASSERT_EQUAL(sal_Int16(26), aFooterTopBorder.InnerLineWidth); CPPUNIT_ASSERT_EQUAL(sal_Int16(26), aFooterTopBorder.LineDistance); CPPUNIT_ASSERT_EQUAL(sal_Int16(4), aFooterTopBorder.LineStyle); CPPUNIT_ASSERT_EQUAL(sal_uInt32(159), aFooterTopBorder.LineWidth); CPPUNIT_ASSERT_EQUAL(sal_Int16(106), aFooterTopBorder.OuterLineWidth); } DECLARE_OOXMLEXPORT_TEST(testParaShadow, "para-shadow.docx") { // The problem was that in w:pBdr, child elements had a w:shadow attribute, but that was ignored. table::ShadowFormat aShadow = getProperty(getParagraph(2), "ParaShadowFormat"); CPPUNIT_ASSERT_EQUAL(COL_BLACK, Color(aShadow.Color)); CPPUNIT_ASSERT_EQUAL(table::ShadowLocation_BOTTOM_RIGHT, aShadow.Location); // w:sz="48" is in eights of a point, 1 pt is 20 twips. CPPUNIT_ASSERT_EQUAL(sal_Int16(convertTwipToMm100(24/8*20)), aShadow.ShadowWidth); } DECLARE_OOXMLEXPORT_TEST(testTableFloating, "table-floating.docx") { // Both the size and the position of the table was incorrect. uno::Reference xTablesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xTables(xTablesSupplier->getTextTables(), uno::UNO_QUERY); // Second table was too wide: 16249, i.e. as wide as the first table. CPPUNIT_ASSERT_EQUAL(sal_Int32(11248), getProperty(xTables->getByIndex(1), "Width")); uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); if (xIndexAccess->getCount()) { // After import, table is inside a TextFrame. uno::Reference xFrame(xIndexAccess->getByIndex(0), uno::UNO_QUERY); // This was 0, should be the opposite of (left margin + half of the border width). CPPUNIT_ASSERT_EQUAL(sal_Int32(-199), getProperty(xFrame, "HoriOrientPosition")); // Was 0 as well, should be the right margin. CPPUNIT_ASSERT_EQUAL(sal_Int32(250), getProperty(xFrame, "RightMargin")); } else { // After import, table is inside a TextFrame. uno::Reference xShape(getShape(1), uno::UNO_QUERY); // This was 0, should be the opposite of (left margin + half of the border width). CPPUNIT_ASSERT_EQUAL(sal_Int32(-199), getProperty(xShape, "HoriOrientPosition")); // Was 0 as well, should be the right margin. CPPUNIT_ASSERT_EQUAL(sal_Int32(250), getProperty(xShape, "RightMargin")); } } DECLARE_OOXMLEXPORT_TEST(testFdo44689_start_page_0, "fdo44689_start_page_0.docx") { // The problem was that the import & export process did not analyze the 'start from page' attribute of a section uno::Reference xPara(getParagraph(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int16(0), getProperty(xPara, "PageNumberOffset")); } DECLARE_OOXMLEXPORT_TEST(testFdo44689_start_page_7, "fdo44689_start_page_7.docx") { // The problem was that the import & export process did not analyze the 'start from page' attribute of a section uno::Reference xPara(getParagraph(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int16(7), getProperty(xPara, "PageNumberOffset")); } DECLARE_OOXMLEXPORT_TEST(testFdo67737, "fdo67737.docx") { // The problem was that imported shapes did not import and render the 'flip:x' and 'flip:y' attributes uno::Reference xArrow = getShape(1); const uno::Sequence aProps = getProperty< uno::Sequence >(xArrow, "CustomShapeGeometry"); for (beans::PropertyValue const & rProp : aProps) { if (rProp.Name == "MirroredY") { CPPUNIT_ASSERT_EQUAL( true, rProp.Value.get() ); return; } } // Shouldn't reach here CPPUNIT_FAIL("Did not find MirroredY=true property"); } DECLARE_OOXMLEXPORT_TEST(testTransparentShadow, "transparent-shadow.docx") { uno::Reference xPicture = getShape(1); sal_Int32 nShadowColor = getProperty(xPicture, "ShadowColor"); sal_Int16 nShadowTransparence = getProperty(xPicture, "ShadowTransparence"); CPPUNIT_ASSERT_EQUAL(sal_Int32(0x808080), nShadowColor); CPPUNIT_ASSERT_EQUAL(sal_Int16(50), nShadowTransparence); } DECLARE_OOXMLEXPORT_EXPORTONLY_TEST(NoFillAttrInImagedata, "NoFillAttrInImagedata.docx") { //problem was that type and color2 which are v:fill attributes were written in 'v:imagedata' xmlDocUniquePtr pXmlDoc = parseExport("word/document.xml"); assertXPathNoAttribute(pXmlDoc, "/w:document/w:body/w:p/w:r/mc:AlternateContent[2]/mc:Fallback/w:pict/v:rect/v:imagedata", "type"); assertXPathNoAttribute(pXmlDoc, "/w:document/w:body/w:p/w:r/mc:AlternateContent[2]/mc:Fallback/w:pict/v:rect/v:imagedata", "color2"); } DECLARE_OOXMLEXPORT_TEST(testBnc837302, "bnc837302.docx") { // The problem was that text with empty author was not inserted as a redline uno::Reference xParagraph = getParagraph(1); // previously 'AAA' was not an own run getRun(xParagraph, 3, "AAA"); // interestingly the 'Insert' is set on the _previous_ run CPPUNIT_ASSERT_EQUAL(OUString("Insert"), getProperty(getRun(xParagraph, 2), "RedlineType")); // make sure we don't introduce a redlined delete in the 2nd paragraph xParagraph = getParagraph(2); CPPUNIT_ASSERT_EQUAL(false, hasProperty(getRun(xParagraph, 1), "RedlineType")); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */