/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace css; class Test : public SwModelTestBase { public: Test() : SwModelTestBase("/sw/qa/extras/rtfexport/data/", "Rich Text Format") { } bool mustTestImportOf(const char* filename) const override { // Don't test the first import of these, for some reason those tests fail const char* aDenylist[] = { "math-eqarray.rtf", "math-escaping.rtf", "math-lim.rtf", "math-mso2007.rtf", "math-nary.rtf", "math-rad.rtf", "math-vertical-stacks.rtf", "math-runs.rtf", }; std::vector vDenylist(aDenylist, aDenylist + SAL_N_ELEMENTS(aDenylist)); // If the testcase is stored in some other format, it's pointless to test. return (OString(filename).endsWith(".rtf") && std::find(vDenylist.begin(), vDenylist.end(), filename) == vDenylist.end()); } }; DECLARE_RTFEXPORT_TEST(testZoom, "zoom.rtf") { 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); } DECLARE_RTFEXPORT_TEST(testFdo38176, "fdo38176.rtf") { CPPUNIT_ASSERT_EQUAL(9, getLength()); } DECLARE_RTFEXPORT_TEST(testFdo49683, "fdo49683.rtf") { uno::Reference xDocumentPropertiesSupplier( mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties( xDocumentPropertiesSupplier->getDocumentProperties()); uno::Sequence aKeywords(xDocumentProperties->getKeywords()); CPPUNIT_ASSERT_EQUAL(sal_Int32(2), aKeywords.getLength()); CPPUNIT_ASSERT_EQUAL(OUString("one"), aKeywords[0]); CPPUNIT_ASSERT_EQUAL(OUString("two"), aKeywords[1]); } DECLARE_RTFEXPORT_TEST(testFdo44174, "fdo44174.rtf") { uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xPropertySet(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); OUString aValue; xPropertySet->getPropertyValue("PageStyleName") >>= aValue; CPPUNIT_ASSERT_EQUAL(OUString("First Page"), aValue); } DECLARE_RTFEXPORT_TEST(testFdo50087, "fdo50087.rtf") { uno::Reference xDocumentPropertiesSupplier( mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties( xDocumentPropertiesSupplier->getDocumentProperties()); CPPUNIT_ASSERT_EQUAL(OUString("Title"), xDocumentProperties->getTitle()); CPPUNIT_ASSERT_EQUAL(OUString("Subject"), xDocumentProperties->getSubject()); CPPUNIT_ASSERT_EQUAL(OUString("First line.\nSecond line."), xDocumentProperties->getDescription()); } DECLARE_RTFEXPORT_TEST(testFdo50831, "fdo50831.rtf") { uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xParaEnumAccess(xTextDocument->getText(), uno::UNO_QUERY); uno::Reference xParaEnum = xParaEnumAccess->createEnumeration(); uno::Reference xPropertySet(xParaEnum->nextElement(), uno::UNO_QUERY); float fValue = 0; xPropertySet->getPropertyValue("CharHeight") >>= fValue; CPPUNIT_ASSERT_EQUAL(10.f, fValue); } DECLARE_RTFEXPORT_TEST(testFdo48335, "fdo48335.odt") { CPPUNIT_ASSERT_EQUAL(3, getPages()); /* * The problem was that we exported a fake pagebreak, make sure it's just a soft one now. * * oParas = ThisComponent.Text.createEnumeration * oPara = oParas.nextElement * oPara = oParas.nextElement * oPara = oParas.nextElement * oRuns = oPara.createEnumeration * oRun = oRuns.nextElement * xray oRun.TextPortionType 'was Text, should be SoftPageBreak */ uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xParaEnumAccess(xTextDocument->getText(), uno::UNO_QUERY); uno::Reference xParaEnum = xParaEnumAccess->createEnumeration(); for (int i = 0; i < 2; i++) xParaEnum->nextElement(); uno::Reference xRunEnumAccess(xParaEnum->nextElement(), uno::UNO_QUERY); uno::Reference xRunEnum = xRunEnumAccess->createEnumeration(); uno::Reference xPropertySet(xRunEnum->nextElement(), uno::UNO_QUERY); OUString aValue; xPropertySet->getPropertyValue("TextPortionType") >>= aValue; CPPUNIT_ASSERT_EQUAL(OUString("SoftPageBreak"), aValue); } DECLARE_RTFEXPORT_TEST(testFdo38244, "fdo38244.rtf") { // See ooxmlexport's testFdo38244(). // Test comment range feature. 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")); // Test 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")); } DECLARE_RTFEXPORT_TEST(testCommentsNested, "comments-nested.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xOuter = getProperty>(getRun(getParagraph(1), 2), "TextField"); CPPUNIT_ASSERT_EQUAL(OUString("Outer"), getProperty(xOuter, "Content").trim()); uno::Reference xInner = getProperty>(getRun(getParagraph(1), 4), "TextField"); CPPUNIT_ASSERT_EQUAL(OUString("Inner"), getProperty(xInner, "Content").trim()); } DECLARE_RTFEXPORT_TEST(testMathAccents, "math-accents.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString("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}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathEqarray, "math-eqarray.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString("y = left lbrace stack { 0 , x < 0 # 1 , x = 0 # {x} ^ {2} , x > 0 } right none"), aActual); } DECLARE_RTFEXPORT_TEST(testMathD, "math-d.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString("left (x mline y mline z right ) left (1 right ) left [2 right ] left " "ldbracket 3 right rdbracket left lline 4 right rline left ldline 5 " "right rdline left langle 6 right rangle left langle a mline b right " "rangle left ({x} over {y} right )"), aActual); } DECLARE_RTFEXPORT_TEST(testMathEscaping, "math-escaping.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString(u"\u00E1 \\{"), aActual); } DECLARE_RTFEXPORT_TEST(testMathLim, "math-lim.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString(u"lim from {x \u2192 1} {x}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathMatrix, "math-matrix.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("left [matrix {1 # 2 ## 3 # 4} right ]"), aActual); } DECLARE_RTFEXPORT_TEST(testMathBox, "math-mbox.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("a"), aActual); } DECLARE_RTFEXPORT_TEST(testMathMso2007, "math-mso2007.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); OUString aExpected(u"A = \u03C0 {r} ^ {2}"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(2), 1)); aExpected = OUString(u"{left (x + a right )} ^ {n} = sum from {k = 0} to {n} {left (stack { n " u"# k } right ) {x} ^ {k} {a} ^ {n \u2212 k}}"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(3), 1)); aExpected = OUString(u"{left (1 + x right )} ^ {n} = 1 + {nx} over {1 !} + {n left (n \u2212 1 " u"right ) {x} ^ {2}} over {2 !} + \u2026"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(4), 1)); aExpected = OUString(u"f left (x right ) = {a} rsub {0} + sum from {n = 1} to {\u221E} {left " u"({a} rsub {n} cos {n\u03C0x} over {L} + {b} rsub {n} sin {n\u03C0x} " u"over {L} right )}"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(5), 1)); aExpected = "{a} ^ {2} + {b} ^ {2} = {c} ^ {2}"; CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(6), 1)); aExpected = OUString(u"x = {\u2212 b \u00B1 sqrt {{b} ^ {2} \u2212 4 ac}} over {2 a}"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(7), 1)); aExpected = OUString(u"{e} ^ {x} = 1 + {x} over {1 !} + {{x} ^ {2}} over {2 !} + {{x} ^ {3}} " u"over {3 !} + \u2026 , \u2212 \u221E < x < \u221E"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(8), 1)); aExpected = OUString(u"sin \u03B1 \u00B1 sin \u03B2 = 2 sin {1} over {2} left (\u03B1 \u00B1 " u"\u03B2 right ) cos {1} over {2} left (\u03B1 \u2213 \u03B2 right )"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); aActual = getFormula(getRun(getParagraph(9), 1)); aExpected = OUString(u"cos \u03B1 + cos \u03B2 = 2 cos {1} over {2} left (\u03B1 + \u03B2 " u"right ) cos {1} over {2} left (\u03B1 \u2212 \u03B2 right )"); CPPUNIT_ASSERT_EQUAL(aExpected, aActual); } DECLARE_RTFEXPORT_TEST(testMathNary, "math-nary.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString("lllint from {1} to {2} {x + 1} prod from {a} {b} sum to {2} {x}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathLimupp, "math-limupp.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("{abcd} overbrace {4}"), aActual); aActual = getFormula(getRun(getParagraph(2), 1)); CPPUNIT_ASSERT_EQUAL(OUString("{xyz} underbrace {3}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathStrikeh, "math-strikeh.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("overstrike {abc}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathPlaceholders, "math-placeholders.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("sum from to "), aActual); } DECLARE_RTFEXPORT_TEST(testMathRad, "math-rad.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("sqrt {4} nroot {3} {x + 1}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathSepchr, "math-sepchr.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL(OUString("AxByBzC"), aActual); } DECLARE_RTFEXPORT_TEST(testMathSubscripts, "math-subscripts.rtf") { OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString("{x} ^ {y} + {e} ^ {x} {x} ^ {b} {x} rsub {b} {a} rsub {c} rsup {b} " "{x} lsub {2} lsup {1} {{x csup {6} csub {3}} lsub {4} lsup {5}} rsub " "{2} rsup {1}"), aActual); } DECLARE_RTFEXPORT_TEST(testMathVerticalstacks, "math-vertical-stacks.rtf") { CPPUNIT_ASSERT_EQUAL(OUString("{a} over {b}"), getFormula(getRun(getParagraph(1), 1))); CPPUNIT_ASSERT_EQUAL(OUString("{a} / {b}"), getFormula(getRun(getParagraph(2), 1))); CPPUNIT_ASSERT_EQUAL(OUString("stack { a # b }"), getFormula(getRun(getParagraph(3), 1))); CPPUNIT_ASSERT_EQUAL(OUString("stack { a # stack { b # c } }"), getFormula(getRun(getParagraph(4), 1))); } DECLARE_RTFEXPORT_TEST(testTdf49073, "tdf49073.rtf") { // test case for Asian phontic guide (ruby text.) sal_Unicode aRuby[3] = { 0x304D, 0x3082, 0x3093 }; OUString sRuby(aRuby, SAL_N_ELEMENTS(aRuby)); CPPUNIT_ASSERT_EQUAL(sRuby, getProperty(getParagraph(1)->getStart(), "RubyText")); OUString sStyle = getProperty(getParagraph(1)->getStart(), "RubyCharStyleName"); uno::Reference xPropertySet( getStyles("CharacterStyles")->getByName(sStyle), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(5.f, getProperty(xPropertySet, "CharHeight")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyAdjust_CENTER), getProperty(getParagraph(2)->getStart(), "RubyAdjust")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyAdjust_BLOCK), getProperty(getParagraph(3)->getStart(), "RubyAdjust")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyAdjust_INDENT_BLOCK), getProperty(getParagraph(4)->getStart(), "RubyAdjust")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyAdjust_LEFT), getProperty(getParagraph(5)->getStart(), "RubyAdjust")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyAdjust_RIGHT), getProperty(getParagraph(6)->getStart(), "RubyAdjust")); CPPUNIT_ASSERT_EQUAL(sal_Int16(text::RubyPosition::INTER_CHARACTER), getProperty(getParagraph(7)->getStart(), "RubyPosition")); } DECLARE_RTFEXPORT_TEST(testMathRuns, "math-runs.rtf") { // was [](){}, i.e. first curly bracket had an incorrect position CPPUNIT_ASSERT_EQUAL(OUString("\\{ left [ right ] left ( right ) \\}"), getFormula(getRun(getParagraph(1), 1))); } DECLARE_RTFEXPORT_TEST(testFdo77979, "fdo77979.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // font name is encoded with \fcharset of font CPPUNIT_ASSERT_EQUAL(OUString(u"\u5FAE\u8F6F\u96C5\u9ED1"), getProperty(getRun(getParagraph(1), 1), "CharFontName")); } DECLARE_RTFEXPORT_TEST(testFdo53113, "fdo53113.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); /* * The problem was that a custom shape was missing its second (and all the other remaining) coordinates. * * oShape = ThisComponent.DrawPage(0) * oPathPropVec = oShape.CustomShapeGeometry(1).Value * oCoordinates = oPathPropVec(0).Value * xray oCoordinates(1).First.Value ' 535 * xray oCoordinates(1).Second.Value ' 102 */ const uno::Sequence aProps = getProperty>(getShape(1), "CustomShapeGeometry"); uno::Sequence aPathProps; for (beans::PropertyValue const& rProp : aProps) { if (rProp.Name == "Path") rProp.Value >>= aPathProps; } uno::Sequence aPairs; for (beans::PropertyValue const& rProp : std::as_const(aPathProps)) { if (rProp.Name == "Coordinates") rProp.Value >>= aPairs; } CPPUNIT_ASSERT_EQUAL(sal_Int32(16), aPairs.getLength()); CPPUNIT_ASSERT_EQUAL(sal_Int32(535), aPairs[1].First.Value.get()); CPPUNIT_ASSERT_EQUAL(sal_Int32(102), aPairs[1].Second.Value.get()); } DECLARE_RTFEXPORT_TEST(testFdo55939, "fdo55939.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The problem was that the exported RTF was invalid. // Also, the 'Footnote text.' had an additional newline at its end. uno::Reference xParagraph(getParagraph(1)); getRun(xParagraph, 1, "Main text before footnote."); // Why the tab has to be removed here? CPPUNIT_ASSERT_EQUAL(OUString("Footnote text."), getProperty>( getRun(xParagraph, 2), "Footnote") ->getText() ->getString() .replaceAll("\t", "")); getRun(xParagraph, 3, " Text after the footnote."); // However, this leading space is intentional and OK. } DECLARE_RTFEXPORT_TEST(testTextFrames, "textframes.odt") { CPPUNIT_ASSERT_EQUAL(3, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // The output was simply invalid, so let's check if all 3 frames were imported back. uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(3), xIndexAccess->getCount()); } DECLARE_RTFEXPORT_TEST(testFdo53604, "fdo53604.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // Invalid output on empty footnote. uno::Reference xFootnotesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xFootnotes = xFootnotesSupplier->getFootnotes(); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xFootnotes->getCount()); } DECLARE_RTFEXPORT_TEST(testFdo52286, "fdo52286.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // The problem was that font size wasn't reduced in sub/super script. CPPUNIT_ASSERT_EQUAL( sal_Int32(58), getProperty(getRun(getParagraph(1), 2), "CharEscapementHeight")); CPPUNIT_ASSERT_EQUAL( sal_Int32(58), getProperty(getRun(getParagraph(2), 2), "CharEscapementHeight")); } DECLARE_RTFEXPORT_TEST(testFdo61507, "fdo61507.rtf") { /* * Unicode-only characters in \title confused Wordpad. Once the exporter * was fixed to guard the problematic characters with \upr and \ud, the * importer didn't cope with these new keywords. */ uno::Reference xDocumentPropertiesSupplier( mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties( xDocumentPropertiesSupplier->getDocumentProperties()); CPPUNIT_ASSERT_EQUAL(OUString(u"\u00C9\u00C1\u0150\u0170\u222D"), xDocumentProperties->getTitle()); // Only "Hello.", no additional characters. CPPUNIT_ASSERT_EQUAL(6, getLength()); } DECLARE_RTFEXPORT_TEST(testFdo30983, "fdo30983.rtf") { // These were 'page text area', not 'entire page', i.e. both the horizontal // and vertical positions were incorrect. CPPUNIT_ASSERT_EQUAL(text::RelOrientation::PAGE_FRAME, getProperty(getShape(1), "HoriOrientRelation")); CPPUNIT_ASSERT_EQUAL(text::RelOrientation::PAGE_FRAME, getProperty(getShape(1), "VertOrientRelation")); } DECLARE_RTFEXPORT_TEST(testPlaceholder, "placeholder.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // Only the field text was exported, make sure we still have a field with the correct Hint text. uno::Reference xRun(getRun(getParagraph(1), 2)); CPPUNIT_ASSERT_EQUAL(OUString("TextField"), getProperty(xRun, "TextPortionType")); uno::Reference xField = getProperty>(xRun, "TextField"); CPPUNIT_ASSERT_EQUAL(OUString("place holder"), getProperty(xField, "Hint")); } DECLARE_RTFEXPORT_TEST(testMnor, "mnor.rtf") { // \mnor wasn't handled, leading to missing quotes around "divF" and so on. OUString aActual = getFormula(getRun(getParagraph(1), 1)); CPPUNIT_ASSERT_EQUAL( OUString( u"iiint from {V} to {\"divF\"} dV = llint from {S} to {\"F\" \u2219 \"n\" dS}"), aActual); } DECLARE_RTFEXPORT_TEST(testI120928, "i120928.rtf") { // \listpicture and \levelpicture0 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_RTFEXPORT_TEST(testBookmark, "bookmark.rtf") { uno::Reference xBookmarksSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xBookmark( xBookmarksSupplier->getBookmarks()->getByName("firstword"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("Hello"), xBookmark->getAnchor()->getString()); } DECLARE_RTFEXPORT_TEST(testHyperlink, "hyperlink.rtf") { CPPUNIT_ASSERT_EQUAL( OUString(), getProperty(getRun(getParagraph(1), 1, "Hello"), "HyperLinkURL")); CPPUNIT_ASSERT_EQUAL( OUString("http://en.wikipedia.org/wiki/World"), getProperty(getRun(getParagraph(1), 2, "world"), "HyperLinkURL")); CPPUNIT_ASSERT_EQUAL(OUString(), getProperty(getRun(getParagraph(1), 3, "!"), "HyperLinkURL")); } DECLARE_RTFEXPORT_TEST(testHyperlinkTdf100105, "hyperlink_empty.rtf") { // export of empty link was invalid, group was closed before it was opened uno::Reference xTextDoc(mxComponent, uno::UNO_QUERY); uno::Reference xCursor(xTextDoc->getText()->createTextCursor()); xCursor->gotoStart(false); CPPUNIT_ASSERT_EQUAL(OUString("http://example.net"), getProperty(xCursor, "HyperLinkURL")); // getRun doesn't provide a 0-length hyperlink CPPUNIT_ASSERT_EQUAL( OUString(), getProperty(getRun(getParagraph(1), 1, "foobar"), "HyperLinkURL")); } DECLARE_RTFEXPORT_TEST(test78758, "fdo78758.rtf") { CPPUNIT_ASSERT_EQUAL( OUString("#__RefHeading___Toc264438068"), getProperty(getRun(getParagraph(2), 1, "EE5E EeEEE5EE"), "HyperLinkURL")); CPPUNIT_ASSERT_EQUAL(OUString("#__RefHeading___Toc264438068"), getProperty(getRun(getParagraph(2), 2, "e"), "HyperLinkURL")); CPPUNIT_ASSERT_EQUAL(OUString("#__RefHeading___Toc264438068"), getProperty(getRun(getParagraph(2), 3, "\t46"), "HyperLinkURL")); } DECLARE_RTFEXPORT_TEST(testTextFrameBorders, "textframe-borders.rtf") { uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); uno::Reference xFrame(xIndexAccess->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(Color(0xD99594), Color(ColorTransparency, getProperty(xFrame, "BackColor"))); table::BorderLine2 aBorder = getProperty(xFrame, "TopBorder"); CPPUNIT_ASSERT_EQUAL(Color(0xC0504D), Color(ColorTransparency, 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(Color(0x622423), Color(ColorTransparency, aShadowFormat.Color)); } DECLARE_RTFEXPORT_TEST(testTextframeGradient, "textframe-gradient.rtf") { uno::Reference xTextFramesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xIndexAccess(xTextFramesSupplier->getTextFrames(), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(sal_Int32(2), xIndexAccess->getCount()); uno::Reference xFrame(xIndexAccess->getByIndex(0), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_GRADIENT, getProperty(xFrame, "FillStyle")); awt::Gradient aGradient = getProperty(xFrame, "FillGradient"); CPPUNIT_ASSERT_EQUAL(Color(0xC0504D), Color(ColorTransparency, aGradient.StartColor)); CPPUNIT_ASSERT_EQUAL(Color(0xD99594), Color(ColorTransparency, aGradient.EndColor)); CPPUNIT_ASSERT_EQUAL(awt::GradientStyle_AXIAL, aGradient.Style); xFrame.set(xIndexAccess->getByIndex(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_GRADIENT, getProperty(xFrame, "FillStyle")); aGradient = getProperty(xFrame, "FillGradient"); CPPUNIT_ASSERT_EQUAL(COL_BLACK, Color(ColorTransparency, aGradient.StartColor)); CPPUNIT_ASSERT_EQUAL(Color(0x666666), Color(ColorTransparency, aGradient.EndColor)); CPPUNIT_ASSERT_EQUAL(awt::GradientStyle_AXIAL, aGradient.Style); } DECLARE_RTFEXPORT_TEST(testRecordChanges, "record-changes.rtf") { // \revisions wasn't imported/exported. CPPUNIT_ASSERT_EQUAL(true, getProperty(mxComponent, "RecordChanges")); } DECLARE_RTFEXPORT_TEST(testTextframeTable, "textframe-table.rtf") { uno::Reference xTextRange(getShape(1), uno::UNO_QUERY); uno::Reference xText = xTextRange->getText(); CPPUNIT_ASSERT_EQUAL(OUString("First para."), getParagraphOfText(1, xText)->getString()); uno::Reference xTable(getParagraphOrTable(2, xText), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("A"), uno::Reference( xTable->getCellByName("A1"), uno::UNO_QUERY_THROW) ->getString()); CPPUNIT_ASSERT_EQUAL(OUString("B"), uno::Reference( xTable->getCellByName("B1"), uno::UNO_QUERY_THROW) ->getString()); CPPUNIT_ASSERT_EQUAL(OUString("Last para."), getParagraphOfText(3, xText)->getString()); } DECLARE_RTFEXPORT_TEST(testFdo66682, "fdo66682.rtf") { 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 OUString aListFormat; for (beans::PropertyValue const& rProp : std::as_const(aProps)) { if (rProp.Name == "ListFormat") aListFormat = rProp.Value.get(); } // Suffix was '\0' instead of ' '. CPPUNIT_ASSERT_EQUAL(OUString(" %1% "), aListFormat); } DECLARE_RTFEXPORT_TEST(testParaShadow, "para-shadow.rtf") { // The problem was that \brdrsh was ignored. table::ShadowFormat aShadow = getProperty(getParagraph(2), "ParaShadowFormat"); CPPUNIT_ASSERT_EQUAL(COL_BLACK, Color(ColorTransparency, aShadow.Color)); CPPUNIT_ASSERT_EQUAL(table::ShadowLocation_BOTTOM_RIGHT, aShadow.Location); CPPUNIT_ASSERT_EQUAL(sal_Int16(convertTwipToMm100(60)), aShadow.ShadowWidth); } DECLARE_RTFEXPORT_TEST(testCharacterBorder, "charborder.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xRun(getRun(getParagraph(1), 1), uno::UNO_QUERY); // RTF has just one border attribute (chbrdr) for text border so all side has // the same border with the same padding // Border { const table::BorderLine2 aTopBorder = getProperty(xRun, "CharTopBorder"); CPPUNIT_ASSERT_BORDER_EQUAL(table::BorderLine2(0xFF6600, 0, 318, 0, 0, 318), aTopBorder); CPPUNIT_ASSERT_BORDER_EQUAL(aTopBorder, getProperty(xRun, "CharLeftBorder")); CPPUNIT_ASSERT_BORDER_EQUAL(aTopBorder, getProperty(xRun, "CharBottomBorder")); CPPUNIT_ASSERT_BORDER_EQUAL(aTopBorder, getProperty(xRun, "CharRightBorder")); } // Padding (brsp) { const sal_Int32 nTopPadding = getProperty(xRun, "CharTopBorderDistance"); // In the original ODT file the padding is 150, but the unit conversion round it down. CPPUNIT_ASSERT_EQUAL(sal_Int32(141), nTopPadding); CPPUNIT_ASSERT_EQUAL(nTopPadding, getProperty(xRun, "CharLeftBorderDistance")); CPPUNIT_ASSERT_EQUAL(nTopPadding, getProperty(xRun, "CharBottomBorderDistance")); CPPUNIT_ASSERT_EQUAL(nTopPadding, getProperty(xRun, "CharRightBorderDistance")); } // Shadow (brdrsh) /* RTF use just one bool value for shadow so the next conversions are made during an export-import round color: any -> black location: any -> bottom-right width: any -> border width */ { const table::ShadowFormat aShadow = getProperty(xRun, "CharShadowFormat"); CPPUNIT_ASSERT_EQUAL(COL_BLACK, Color(ColorTransparency, aShadow.Color)); CPPUNIT_ASSERT_EQUAL(table::ShadowLocation_BOTTOM_RIGHT, aShadow.Location); CPPUNIT_ASSERT_EQUAL(sal_Int16(318), aShadow.ShadowWidth); } } DECLARE_RTFEXPORT_TEST(testFdo66743, "fdo66743.rtf") { uno::Reference xTable(getParagraphOrTable(1), uno::UNO_QUERY); uno::Reference xCell = xTable->getCellByName("A1"); // This was too dark, 0x7f7f7f. CPPUNIT_ASSERT_EQUAL(Color(0xd8d8d8), Color(ColorTransparency, getProperty(xCell, "BackColor"))); } DECLARE_RTFEXPORT_TEST(testFdo68787, "fdo68787.rtf") { uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); // This was 0, the 'lack of \chftnsep' <-> '0 line width' mapping was missing in the RTF tokenizer / exporter. CPPUNIT_ASSERT_EQUAL(sal_Int32(25), getProperty(xPageStyle, "FootnoteLineRelativeWidth")); } DECLARE_RTFEXPORT_TEST(testFdo74709, "fdo74709.rtf") { uno::Reference xCell = getCell(getParagraphOrTable(1), "B1"); // This was 0, as top/bottom/left/right padding wasn't imported. CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(360)), getProperty(xCell, "RightBorderDistance")); } DECLARE_RTFEXPORT_TEST(testTdf84832, "tdf84832.docx") { uno::Reference xCell = getCell(getParagraphOrTable(2), "A1"); // This was 0, as left padding wasn't exported. CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(108)), getProperty(xCell, "LeftBorderDistance")); } DECLARE_RTFEXPORT_TEST(testRelsize, "relsize.rtf") { uno::Reference xShape = getShape(1); CPPUNIT_ASSERT_EQUAL(sal_Int16(40), getProperty(xShape, "RelativeWidth")); CPPUNIT_ASSERT_EQUAL(text::RelOrientation::PAGE_FRAME, getProperty(xShape, "RelativeWidthRelation")); CPPUNIT_ASSERT_EQUAL(sal_Int16(20), getProperty(xShape, "RelativeHeight")); CPPUNIT_ASSERT_EQUAL(text::RelOrientation::FRAME, getProperty(xShape, "RelativeHeightRelation")); } DECLARE_RTFEXPORT_TEST(testLineNumbering, "linenumbering.rtf") { uno::Reference xLineNumberingProperties(mxComponent, uno::UNO_QUERY_THROW); uno::Reference xPropertySet = xLineNumberingProperties->getLineNumberingProperties(); CPPUNIT_ASSERT_EQUAL(true, getProperty(xPropertySet, "IsOn")); CPPUNIT_ASSERT_EQUAL(sal_Int32(5), getProperty(xPropertySet, "Interval")); } DECLARE_RTFEXPORT_TEST(testFdo77600, "fdo77600.rtf") { // This was 'Liberation Serif'. CPPUNIT_ASSERT_EQUAL(OUString("Arial"), getProperty(getRun(getParagraph(1), 3), "CharFontName")); } DECLARE_RTFEXPORT_TEST(testFdo80167, "fdo80167.rtf") { // Problem was that after export, the page break was missing, so this was 1. CPPUNIT_ASSERT_EQUAL(2, getPages()); } DECLARE_RTFEXPORT_TEST(testFdo32613, "fdo32613.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // This was AS_CHARACTER, RTF export did not support writing anchored pictures. CPPUNIT_ASSERT_EQUAL(text::TextContentAnchorType_AT_CHARACTER, getProperty(getShape(1), "AnchorType")); } DECLARE_RTFEXPORT_TEST(testPictureWrapPolygon, "picture-wrap-polygon.rtf") { // The problem was that the wrap polygon was ignored during import. drawing::PointSequenceSequence aSeqSeq = getProperty(getShape(1), "ContourPolyPolygon"); // This was 0: the polygon list was empty. CPPUNIT_ASSERT_EQUAL(sal_Int32(1), aSeqSeq.getLength()); drawing::PointSequence aSeq = aSeqSeq[0]; CPPUNIT_ASSERT_EQUAL(sal_Int32(11), aSeq.getLength()); // The shape also didn't have negative top / left coordinates. CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(-1177)), getProperty(getShape(1), "HoriOrientPosition")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(-67)), getProperty(getShape(1), "VertOrientPosition")); } DECLARE_RTFEXPORT_TEST(testTdf113408, "tdf113408.rtf") { // This was 0, left margin was not inherited from style properly. CPPUNIT_ASSERT_EQUAL(static_cast(1270), getProperty(getParagraph(1), "ParaLeftMargin")); } DECLARE_RTFEXPORT_TEST(testAbi10039, "abi10039.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // Make sure we don't just crash on export, and additionally the shape should not be inline (as it's at-page anchored originally). CPPUNIT_ASSERT(text::TextContentAnchorType_AS_CHARACTER != getProperty(getShape(1), "AnchorType")); } DECLARE_RTFEXPORT_TEST(testAbi10076, "abi10076.odt") { CPPUNIT_ASSERT_EQUAL(2, getPages()); // Just make sure that we don't crash after exporting a fully calculated layout. } DECLARE_RTFEXPORT_TEST(testEm, "em.rtf") { // Test all possible \acc* control words. CPPUNIT_ASSERT_EQUAL(text::FontEmphasis::NONE, getProperty(getRun(getParagraph(1), 1), "CharEmphasis")); CPPUNIT_ASSERT_EQUAL(text::FontEmphasis::DOT_ABOVE, getProperty(getRun(getParagraph(1), 2), "CharEmphasis")); CPPUNIT_ASSERT_EQUAL(text::FontEmphasis::ACCENT_ABOVE, getProperty(getRun(getParagraph(1), 3), "CharEmphasis")); // This was missing. CPPUNIT_ASSERT_EQUAL(text::FontEmphasis::CIRCLE_ABOVE, getProperty(getRun(getParagraph(1), 4), "CharEmphasis")); // This one, too. CPPUNIT_ASSERT_EQUAL(text::FontEmphasis::DOT_BELOW, getProperty(getRun(getParagraph(1), 5), "CharEmphasis")); } DECLARE_RTFEXPORT_TEST(testNumberingFont, "numbering-font.rtf") { uno::Reference xStyle( getStyles("CharacterStyles")->getByName("ListLabel 1"), uno::UNO_QUERY); // This was Liberation Serif, i.e. custom font of the numbering itself ("1.\t") was lost on import. CPPUNIT_ASSERT_EQUAL(OUString("Impact"), getProperty(xStyle, "CharFontName")); CPPUNIT_ASSERT_EQUAL(72.f, getProperty(xStyle, "CharHeight")); } DECLARE_RTFEXPORT_TEST(testFdo82860, "fdo82860.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); // The problem was that: // 1) The import tried to use fieldmarks for SHAPE fields // 2) The exporter did not handle "shape with textbox" text. uno::Reference xTextRange(getShape(1), uno::UNO_QUERY); uno::Reference xText = xTextRange->getText(); CPPUNIT_ASSERT_EQUAL(OUString("hello"), getParagraphOfText(1, xText)->getString()); } DECLARE_RTFEXPORT_TEST(testFdo82858, "fdo82858.docx") { // This was table::BorderLineStyle::SOLID, exporter failed to write explicit no line when line color was written. CPPUNIT_ASSERT_EQUAL(table::BorderLineStyle::NONE, getProperty(getShape(1), "TopBorder").LineStyle); } DECLARE_RTFEXPORT_TEST(testTdf104936, "tdf104936.rtf") { uno::Reference xShape1(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(static_cast(0), getProperty(xShape1, "ZOrder")); // This failed, the shape without text covered the shape with text. CPPUNIT_ASSERT(xShape1->getString().isEmpty()); uno::Reference xShape2(getShape(2), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(static_cast(1), getProperty(xShape2, "ZOrder")); CPPUNIT_ASSERT_EQUAL(OUString("Hello"), xShape2->getString()); } DECLARE_RTFEXPORT_TEST(testTableRtl, "table-rtl.rtf") { uno::Reference xTablesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xTables(xTablesSupplier->getTextTables(), uno::UNO_QUERY); uno::Reference xTable(xTables->getByIndex(0), uno::UNO_QUERY); // This was text::WritingMode2::LR_TB, i.e. direction of the table was ignored. CPPUNIT_ASSERT_EQUAL(text::WritingMode2::RL_TB, getProperty(xTable, "WritingMode")); } DECLARE_RTFEXPORT_TEST(testNumOverrideStart, "num-override-start.rtf") { // The numbering on the second level was "3.1", not "1.3". uno::Reference xRules = getProperty>( getStyles("NumberingStyles")->getByName("WWNum1"), "NumberingRules"); CPPUNIT_ASSERT_EQUAL( sal_Int16(1), comphelper::SequenceAsHashMap(xRules->getByIndex(0))["StartWith"].get()); CPPUNIT_ASSERT_EQUAL( sal_Int16(3), comphelper::SequenceAsHashMap(xRules->getByIndex(1))["StartWith"].get()); } DECLARE_RTFEXPORT_TEST(testFdo82006, "fdo82006.rtf") { // These were 176 (100 twips), as \sbauto and \sbbefore were ignored. CPPUNIT_ASSERT_EQUAL(sal_Int32(0), getProperty(getParagraph(1), "ParaTopMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(1), "ParaBottomMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(2), "ParaTopMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(2), "ParaBottomMargin")); } DECLARE_RTFEXPORT_TEST(testTdf104081, "tdf104081.rtf") { // These were 494 (280 twips), as \htmautsp was ignored. CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(100)), getProperty(getParagraph(1), "ParaTopMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(100)), getProperty(getParagraph(1), "ParaBottomMargin")); } DECLARE_RTFEXPORT_TEST(testTdf88583, "tdf88583.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); // This was FillStyle_NONE, as background color was missing from the color table during export. CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_SOLID, getProperty(getParagraph(1), "FillStyle")); CPPUNIT_ASSERT_EQUAL(static_cast(0x00cc00), getProperty(getParagraph(1), "FillColor")); } DECLARE_RTFEXPORT_TEST(testMargmirror, "margmirror.rtf") { // \margmirror was not handled, this was PageStyleLayout_ALL. uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(style::PageStyleLayout_MIRRORED, getProperty(xPageStyle, "PageStyleLayout")); } DECLARE_RTFEXPORT_TEST(testSautoupd, "sautoupd.rtf") { // \sautoupd was ignored during import and export. uno::Reference xHeading1( getStyles("ParagraphStyles")->getByName("Heading 1"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(true, getProperty(xHeading1, "IsAutoUpdate")); uno::Reference xHeading2( getStyles("ParagraphStyles")->getByName("Heading 2"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(false, getProperty(xHeading2, "IsAutoUpdate")); } DECLARE_RTFEXPORT_TEST(testHyphauto, "hyphauto.rtf") { CPPUNIT_ASSERT_EQUAL(true, getProperty(getParagraph(1), "ParaIsHyphenation")); } DECLARE_RTFEXPORT_TEST(testHyphpar, "hyphpar.rtf") { // Hyphenation was enabled for all 3 paragraphs, but it should be disabled for the 2nd one. CPPUNIT_ASSERT_EQUAL(false, getProperty(getParagraph(2), "ParaIsHyphenation")); } DECLARE_RTFEXPORT_TEST(testTdf108955, "tdf108955.rtf") { CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_SOLID, getProperty(getParagraph(1), "FillStyle")); // This was 0xffffff, i.e. non-white background was overwritten from the paragraph style. CPPUNIT_ASSERT_EQUAL(static_cast(0xffff99), getProperty(getParagraph(1), "FillColor")); } DECLARE_RTFEXPORT_TEST(testTdf80708, "tdf80708.rtf") { uno::Reference xTextTablesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xTables(xTextTablesSupplier->getTextTables(), uno::UNO_QUERY); uno::Reference xTable(xTables->getByIndex(1), uno::UNO_QUERY); uno::Reference xTableRows = xTable->getRows(); // This was 2, i.e. the second table had 3 cols, now 2 as expected. CPPUNIT_ASSERT_EQUAL(static_cast(1), getProperty>( xTableRows->getByIndex(0), "TableColumnSeparators") .getLength()); } CPPUNIT_TEST_FIXTURE(Test, testHyperlinkWithoutURL) { load(mpTestDocumentPath, "tdf90421.fodt"); // Change the hyperlink, so its URL is empty. uno::Reference xRun(getRun(getParagraph(1), 2), uno::UNO_QUERY); xRun->setPropertyValue("HyperLinkURL", uno::makeAny(OUString())); reload(mpFilter, "gutter-left.rtf"); SvMemoryStream aMemoryStream; SvFileStream aStream(maTempFile.GetURL(), StreamMode::READ); aStream.ReadStream(aMemoryStream); OString aData(static_cast(aMemoryStream.GetData()), aMemoryStream.GetSize()); // This was some positive number, i.e. we exported a hyperlink with an empty URL. CPPUNIT_ASSERT_EQUAL(static_cast(-1), aData.indexOf("HYPERLINK")); } DECLARE_RTFEXPORT_TEST(testTdf92521, "tdf92521.odt") { // There should be a page break that's in the middle of the document: right after the table. // But there wasn't, so this was 1. CPPUNIT_ASSERT_EQUAL(2, getPages()); } DECLARE_RTFEXPORT_TEST(testTdf114309, "tdf114309.rtf") { // Without the fix in place, this test would have failed with // - the property is of unexpected type or void: TextSection auto xTextSection = getProperty>(getParagraph(1), "TextSection"); auto xTextColumns = getProperty>(xTextSection, "TextColumns"); CPPUNIT_ASSERT_EQUAL(sal_Int16(2), xTextColumns->getColumnCount()); } DECLARE_RTFEXPORT_TEST(testTdf94043, "tdf94043.rtf") { auto xTextSection = getProperty>(getParagraph(2), "TextSection"); auto xTextColumns = getProperty>(xTextSection, "TextColumns"); // This was 0, the separator line was not visible due to 0 width. CPPUNIT_ASSERT_EQUAL(static_cast(2), getProperty(xTextColumns, "SeparatorLineWidth")); CPPUNIT_ASSERT_EQUAL(7, getParagraphs()); } DECLARE_RTFEXPORT_TEST(testTdf94377, "tdf94377.rtf") { uno::Reference xTextFieldsSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xFieldsAccess( xTextFieldsSupplier->getTextFields()); uno::Reference xFields(xFieldsAccess->createEnumeration()); uno::Reference xPropertySet(xFields->nextElement(), uno::UNO_QUERY); auto xText = getProperty>(xPropertySet, "TextRange"); // This failed, as: // 1) multiple paragraphs were not exported, so the text was "Asdf10asdf12". // 2) direct formatting of runs were not exported, so this was 12 (the document default). CPPUNIT_ASSERT_EQUAL( 10.f, getProperty(getRun(getParagraphOfText(1, xText, "Asdf10"), 1), "CharHeight")); CPPUNIT_ASSERT_EQUAL( 12.f, getProperty(getRun(getParagraphOfText(2, xText, "asdf12"), 1), "CharHeight")); } DECLARE_RTFEXPORT_TEST(testTdf104079, "tdf104079.rtf") { bool bFound = false; int nIndex = 0; while (!bFound) { uno::Reference xParagraph = getParagraph(++nIndex); if (!xParagraph->getString().startsWith("toc3")) continue; bFound = true; // This was 0, 3rd paragraph of ToC lost its bottom paragraph margin. CPPUNIT_ASSERT_EQUAL(static_cast(212), getProperty(xParagraph, "ParaBottomMargin")); } } DECLARE_RTFEXPORT_TEST(testPageBackground, "page-background.rtf") { // The problem was that \background was ignored. uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(Color(0x92D050), Color(ColorTransparency, getProperty(xPageStyle, "BackColor"))); } DECLARE_RTFEXPORT_TEST(testTdf96175, "tdf96175.rtf") { // The problem that a user defined property named "Company" was lost on export. uno::Reference xDocumentPropertiesSupplier( mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties = xDocumentPropertiesSupplier->getDocumentProperties(); uno::Reference xUserDefinedProperties = xDocumentProperties->getUserDefinedProperties(); // This resulted in a beans::UnknownPropertyException. CPPUNIT_ASSERT_EQUAL(OUString("foobar"), getProperty(xUserDefinedProperties, "Company")); } DECLARE_RTFEXPORT_TEST(testRedline, "redline.rtf") { CPPUNIT_ASSERT_EQUAL(OUString("Rebecca Lopez"), getProperty(getRun(getParagraph(1), 2), "RedlineAuthor")); CPPUNIT_ASSERT_EQUAL(OUString("Dorothy Jones"), getProperty(getRun(getParagraph(2), 2), "RedlineAuthor")); } DECLARE_RTFEXPORT_TEST(testCustomDocProps, "custom-doc-props.rtf") { // Custom document properties were not improved, this resulted in a beans::UnknownPropertyException. uno::Reference xDocumentPropertiesSupplier( mxComponent, uno::UNO_QUERY); uno::Reference xDocumentProperties = xDocumentPropertiesSupplier->getDocumentProperties(); uno::Reference xUserDefinedProperties = xDocumentProperties->getUserDefinedProperties(); CPPUNIT_ASSERT_EQUAL( OUString("2016-03-08T10:55:18,531376147"), getProperty(xUserDefinedProperties, "urn:bails:IntellectualProperty:Authorization:StartValidity")); CPPUNIT_ASSERT_EQUAL( OUString("None"), getProperty(xUserDefinedProperties, "urn:bails:IntellectualProperty:Authorization:StopValidity")); // Test roundtrip of numbers. This failed as getProperty() did not find "n". CPPUNIT_ASSERT_EQUAL(42.0, getProperty(xUserDefinedProperties, "n")); // Test boolean "yes". CPPUNIT_ASSERT(getProperty(xUserDefinedProperties, "by")); // Test boolean "no". CPPUNIT_ASSERT(!getProperty(xUserDefinedProperties, "bn")); // Test roundtrip of date in general, and year/month/day in particular. util::DateTime aDate = getProperty(xUserDefinedProperties, "d"); CPPUNIT_ASSERT_EQUAL(static_cast(2016), aDate.Year); CPPUNIT_ASSERT_EQUAL(o3tl::narrowing(1), aDate.Month); CPPUNIT_ASSERT_EQUAL(o3tl::narrowing(30), aDate.Day); // Test real number. CPPUNIT_ASSERT_EQUAL(3.14, getProperty(xUserDefinedProperties, "pi")); } DECLARE_RTFEXPORT_TEST(testTdf65642, "tdf65642.rtf") { uno::Reference xPageStyles = getStyles("PageStyles"); uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xCursor(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); xCursor->jumpToLastPage(); OUString pageStyleName = getProperty(xCursor, "PageStyleName"); // The second page's numbering type: this was style::NumberingType::ARABIC. CPPUNIT_ASSERT_EQUAL( style::NumberingType::CHARS_UPPER_LETTER_N, getProperty(xPageStyles->getByName(pageStyleName), "NumberingType")); // The second page's restart value: this was 0. CPPUNIT_ASSERT_EQUAL(static_cast(1), getProperty(getParagraph(2), "PageNumberOffset")); } DECLARE_RTFEXPORT_TEST(testPgnlcltr, "pgnlcltr.rtf") { uno::Reference xPageStyles = getStyles("PageStyles"); uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xCursor(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); xCursor->jumpToLastPage(); OUString pageStyleName = getProperty(xCursor, "PageStyleName"); // The second page's numbering type: this was style::NumberingType::ARABIC. CPPUNIT_ASSERT_EQUAL( style::NumberingType::CHARS_LOWER_LETTER_N, getProperty(xPageStyles->getByName(pageStyleName), "NumberingType")); } DECLARE_RTFEXPORT_TEST(testPgnucrm, "pgnucrm.rtf") { uno::Reference xPageStyles = getStyles("PageStyles"); uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xCursor(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); xCursor->jumpToLastPage(); OUString pageStyleName = getProperty(xCursor, "PageStyleName"); // The second page's numbering type: this was style::NumberingType::ARABIC. CPPUNIT_ASSERT_EQUAL( style::NumberingType::ROMAN_UPPER, getProperty(xPageStyles->getByName(pageStyleName), "NumberingType")); } DECLARE_RTFEXPORT_TEST(testPgnlcrm, "pgnlcrm.rtf") { uno::Reference xPageStyles = getStyles("PageStyles"); uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xCursor(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); xCursor->jumpToLastPage(); OUString pageStyleName = getProperty(xCursor, "PageStyleName"); // The second page's numbering type: this was style::NumberingType::ARABIC. CPPUNIT_ASSERT_EQUAL( style::NumberingType::ROMAN_LOWER, getProperty(xPageStyles->getByName(pageStyleName), "NumberingType")); } DECLARE_RTFEXPORT_TEST(testPgndec, "pgndec.rtf") { uno::Reference xPageStyles = getStyles("PageStyles"); uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xTextViewCursorSupplier( xModel->getCurrentController(), uno::UNO_QUERY); uno::Reference xCursor(xTextViewCursorSupplier->getViewCursor(), uno::UNO_QUERY); xCursor->jumpToLastPage(); OUString pageStyleName = getProperty(xCursor, "PageStyleName"); // The second page's numbering type: this was style::NumberingType::ROMAN_LOWER. CPPUNIT_ASSERT_EQUAL( style::NumberingType::ARABIC, getProperty(xPageStyles->getByName(pageStyleName), "NumberingType")); // tdf#82111 ensure a paragraph exists before a section break. // This was only two paragraphs, and both page number fields were in one para on page 2 ("11"). getParagraph(2, "1"); CPPUNIT_ASSERT_EQUAL(3, getParagraphs()); } DECLARE_RTFEXPORT_TEST(testTdf98806, "tdf98806.rtf") { uno::Reference xBookmarksSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xBookmark( xBookmarksSupplier->getBookmarks()->getByName("bookmark"), uno::UNO_QUERY); // This was empty, bookmark in table wasn't imported correctly. CPPUNIT_ASSERT_EQUAL(OUString("BBB"), xBookmark->getAnchor()->getString()); } DECLARE_RTFEXPORT_TEST(testTdf61901, "tdf61901.rtf") { // Test the file directly, as current RTF import gives the correct font name with and without the fix. if (mbExported) { SvStream* pStream = maTempFile.GetStream(StreamMode::READ); OString sLine; while (pStream->ReadLine(sLine)) { sal_Int32 nIndex = sLine.indexOf("\\loch\\loch"); if (nIndex != -1) { // Make sure that \hich is always written after a \loch\loch. OString sRemaining = sLine.copy(nIndex); CPPUNIT_ASSERT(sRemaining.indexOf("\\hich") != -1); } } } } DECLARE_RTFEXPORT_TEST(testTdf103925, "tdf103925.rtf") { // This was true, \animtext0 resulted in setting the blinking font effect. CPPUNIT_ASSERT_EQUAL(false, getProperty(getRun(getParagraph(1), 1), "CharFlash")); } DECLARE_RTFEXPORT_TEST(testTdf104228, "tdf104228.rtf") { uno::Reference xTable(getParagraphOrTable(2), uno::UNO_QUERY); uno::Reference xCell(xTable->getCellByName("C1"), uno::UNO_QUERY); uno::Reference xParagraph = getParagraphOfText(1, xCell->getText()); // This was 2103, implicit 0 as direct formatting was ignored on the // paragraph (and the style had this larger value). CPPUNIT_ASSERT_EQUAL(static_cast(0), getProperty(xParagraph, "ParaLeftMargin")); } DECLARE_RTFEXPORT_TEST(testTdf104085, "tdf104085.rtf") { uno::Reference xPara(getParagraph(1)); uno::Reference properties(xPara, uno::UNO_QUERY); uno::Reference xLevels(properties->getPropertyValue("NumberingRules"), uno::UNO_QUERY); uno::Sequence aProps; xLevels->getByIndex(0) >>= aProps; for (beans::PropertyValue const& prop : std::as_const(aProps)) { if (prop.Name == "BulletChar") return; } CPPUNIT_FAIL("no BulletChar property"); } DECLARE_RTFEXPORT_TEST(testTdf113550, "tdf113550.rtf") { uno::Reference xTable(getParagraphOrTable(1), uno::UNO_QUERY); uno::Reference xCell(xTable->getCellByName("A1"), uno::UNO_QUERY); uno::Reference xParagraph = getParagraphOfText(1, xCell->getText()); // This was 2501, 0 as direct formatting was ignored on the paragraph (and // the style had this larger value). CPPUNIT_ASSERT_EQUAL(static_cast(0), getProperty(xParagraph, "ParaLeftMargin")); } DECLARE_RTFEXPORT_TEST(testLeveljcCenter, "leveljc-center.rtf") { // Tests that \leveljc1 is mapped to Adjust=Center for a numbering rule. uno::Reference xPara(getParagraph(1)); uno::Reference properties(xPara, uno::UNO_QUERY); uno::Reference xLevels(properties->getPropertyValue("NumberingRules"), uno::UNO_QUERY); uno::Sequence aProps; xLevels->getByIndex(0) >>= aProps; for (beans::PropertyValue const& prop : std::as_const(aProps)) { if (prop.Name == "Adjust") { sal_Int16 nValue = 0; CPPUNIT_ASSERT(prop.Value >>= nValue); CPPUNIT_ASSERT_EQUAL(text::HoriOrientation::CENTER, nValue); return; } } CPPUNIT_FAIL("no Adjust property"); } DECLARE_RTFEXPORT_TEST(testHyperlinkTarget, "hyperlink-target.rtf") { // This was empty, hyperlink target was lost on import. CPPUNIT_ASSERT_EQUAL(OUString("_blank"), getProperty(getRun(getParagraph(1), 1), "HyperLinkTarget")); } DECLARE_RTFEXPORT_TEST(testTdf107620, "tdf107620.docx") { // This failed, RTF export didn't write the \htmautsp compat flag, the // original bugdoc resulting in 2 pages instead of 1. uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xSettings( xFactory->createInstance("com.sun.star.document.Settings"), uno::UNO_QUERY); bool bAddParaTableSpacing = true; xSettings->getPropertyValue("AddParaTableSpacing") >>= bAddParaTableSpacing; CPPUNIT_ASSERT(!bAddParaTableSpacing); } DECLARE_RTFEXPORT_TEST(testTdf104937, "tdf104937.rtf") { 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 xTableRows = xTable->getRows(); auto aSeparators = getProperty>( xTableRows->getByIndex(1), "TableColumnSeparators"); // First table's second row had 9 cells (so 8 separators). CPPUNIT_ASSERT_EQUAL(static_cast(8), aSeparators.getLength()); // This was 3174, i.e. last cell was wider than expected, while others were // narrower. CPPUNIT_ASSERT_GREATER(static_cast(4500), aSeparators[7].Position); } DECLARE_RTFEXPORT_TEST(testTdf112507, "tdf112507.rtf") { 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 xTableRows = xTable->getRows(); auto aSeparators = getProperty>( xTableRows->getByIndex(1), "TableColumnSeparators"); // First table's second row had 3 cells (so 2 separators). CPPUNIT_ASSERT_EQUAL(static_cast(2), aSeparators.getLength()); // This was 3333, i.e. the B2 cell was too narrow and the text needed 2 lines. CPPUNIT_ASSERT_GREATEREQUAL(5000, aSeparators[1].Position - aSeparators[0].Position); } DECLARE_RTFEXPORT_TEST(testTdf107480, "tdf107480.rtf") { // These were 176 (100 twips), as \htmautsp was parsed too late. CPPUNIT_ASSERT_EQUAL(sal_Int32(0), getProperty(getParagraph(1), "ParaTopMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(1), "ParaBottomMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(2), "ParaTopMargin")); CPPUNIT_ASSERT_EQUAL(sal_Int32(convertTwipToMm100(280)), getProperty(getParagraph(2), "ParaBottomMargin")); } DECLARE_RTFEXPORT_TEST(testWatermark, "watermark.rtf") { uno::Reference xShape(getShape(1), uno::UNO_QUERY); CPPUNIT_ASSERT_EQUAL(OUString("WatermarkRTF"), xShape->getString()); uno::Reference xPropertySet(xShape, uno::UNO_QUERY); OUString aFont; float nFontSize; // Check transparency CPPUNIT_ASSERT_EQUAL(sal_Int16(50), getProperty(xShape, "FillTransparence")); // 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(66), nFontSize); } DECLARE_RTFEXPORT_TEST(testTdf109790, "tdf109790.rtf") { uno::Reference xTable(getParagraphOrTable(2), uno::UNO_QUERY); uno::Reference xCell(xTable->getCellByName("A1"), uno::UNO_QUERY); // Style information was reset, which caused character height to be 22. CPPUNIT_ASSERT_EQUAL( 10.f, getProperty(getRun(getParagraphOfText(1, xCell->getText()), 1), "CharHeight")); } DECLARE_RTFEXPORT_TEST(testTdf112211, "tdf112211.rtf") { // This was 0, \fi in a list level definition was not imported. auto xRules = getProperty>( getStyles("NumberingStyles")->getByName("WWNum1"), "NumberingRules"); comphelper::SequenceAsHashMap aRule(xRules->getByIndex(0)); CPPUNIT_ASSERT_EQUAL(static_cast(-635), aRule["FirstLineIndent"].get()); } DECLARE_RTFEXPORT_TEST(testTdf113202, "tdf113202.rtf") { // This failed, contextual spacing in 4th paragraph was lost. CPPUNIT_ASSERT(getProperty(getParagraph(4), "ParaContextMargin")); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */