/* -*- 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 using namespace css; class Test : public SwModelTestBase { public: Test() : SwModelTestBase("/sw/qa/extras/rtfexport/data/", "Rich Text Format") { } }; DECLARE_RTFEXPORT_TEST(testTdf100961_fixedDateTime, "tdf100961_fixedDateTime.rtf") { // This should be a fixed date/time field, not the current time. getParagraph(1, "05.01.19 04:06:08"); uno::Reference xTFS(mxComponent, uno::UNO_QUERY); uno::Reference xFields(xTFS->getTextFields()->createEnumeration()); CPPUNIT_ASSERT_MESSAGE("constant time", getProperty(xFields->nextElement(), "IsFixed")); } DECLARE_RTFEXPORT_TEST(testTdf108949, "tdf108949_footnoteCharFormat.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); CPPUNIT_ASSERT_EQUAL_MESSAGE("Paragraph Numbering style", OUString(), getProperty(getParagraph(2), "NumberingStyleName")); uno::Reference xFootnotesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xFootnotes = xFootnotesSupplier->getFootnotes(); uno::Reference xFootnoteText; xFootnotes->getByIndex(0) >>= xFootnoteText; // This was green (0x00A800), the character property of the footnote character, not the footnote text CPPUNIT_ASSERT_EQUAL_MESSAGE( "Footnote Text color", COL_AUTO, getProperty(getRun(getParagraphOfText(1, xFootnoteText), 1), "CharColor")); } DECLARE_RTFEXPORT_TEST(testTdf141964_numId0, "tdf141964_numId0.rtf") { // Unit test added where numId is zero - which is only possible in RTF. uno::Reference xPara(getParagraph(3, "Geschichte"), uno::UNO_QUERY); // Pre-emptive test: ensure that paragraph 3 remains numbered and numId0 doesn't mean no numbering. CPPUNIT_ASSERT(!getProperty(xPara, "NumberingStyleName").isEmpty()); } DECLARE_RTFEXPORT_TEST(testTdf108949_footnote, "tdf108949_footnote.rtf") { CPPUNIT_ASSERT_EQUAL_MESSAGE("Paragraph Numbering style", OUString(), getProperty(getParagraph(2), "NumberingStyleName")); uno::Reference xFootnotesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xFootnotes = xFootnotesSupplier->getFootnotes(); uno::Reference xFootnote; xFootnotes->getByIndex(0) >>= xFootnote; // The color of the footnote anchor was black (0x000000) CPPUNIT_ASSERT_EQUAL_MESSAGE("Footnote Character color", Color(0xFF0000), getProperty(xFootnote->getAnchor(), "CharColor")); } DECLARE_RTFEXPORT_TEST(testTdf130817, "tdf130817.rtf") { uno::Reference xEndnotesSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xEndnotes = xEndnotesSupplier->getEndnotes(); uno::Reference xEndnote1; xEndnotes->getByIndex(0) >>= xEndnote1; uno::Reference xEndnoteText1; xEndnotes->getByIndex(0) >>= xEndnoteText1; CPPUNIT_ASSERT_EQUAL(OUString("Titolo 1"), xEndnoteText1->getString().trim()); CPPUNIT_ASSERT_EQUAL(OUString("$"), xEndnote1->getAnchor()->getString()); uno::Reference xEndnote2; xEndnotes->getByIndex(1) >>= xEndnote2; uno::Reference xEndnoteText2; xEndnotes->getByIndex(1) >>= xEndnoteText2; CPPUNIT_ASSERT_EQUAL(OUString("Titolo 2"), xEndnoteText2->getString().trim()); CPPUNIT_ASSERT_EQUAL(OUString("$"), xEndnote1->getAnchor()->getString()); } DECLARE_RTFEXPORT_TEST(testTdf137683_charHighlightNone, "tdf137683_charHighlightNone.rtf") { uno::Reference xRun(getRun(getParagraph(1), 1), uno::UNO_QUERY_THROW); // This test was failing with a brown charHighlight of 8421376 (0x808000), instead of COL_TRANSPARENT (0xFFFFFFFF) CPPUNIT_ASSERT_EQUAL(COL_AUTO, getProperty(xRun, "CharHighlight")); } DECLARE_RTFEXPORT_TEST(testTdf116436_tableBackground, "tdf116436_tableBackground.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); 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"); if (mbExported) CPPUNIT_ASSERT_EQUAL(Color(0xF8DF7C), getProperty(xCell, "BackColor")); xCell.set(xTable->getCellByName("A6")); CPPUNIT_ASSERT_EQUAL(Color(0x81D41A), getProperty(xCell, "BackColor")); xCell.set(xTable->getCellByName("B6")); if (mbExported) CPPUNIT_ASSERT_EQUAL(Color(0xFFFBCC), getProperty(xCell, "BackColor")); } DECLARE_RTFEXPORT_TEST(testTdf122589_firstSection, "tdf122589_firstSection.odt") { CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xPageStyle(getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); uno::Reference xHeaderText = getProperty>(xPageStyle, "HeaderText"); CPPUNIT_ASSERT_EQUAL(OUString("My header"), xHeaderText->getString()); CPPUNIT_ASSERT_EQUAL_MESSAGE("# of paragraphs", 2, getParagraphs()); } DECLARE_RTFEXPORT_TEST(testTdf104035, "tdf104035.rtf") { auto aTabStops = getProperty>(getParagraph(1), "ParaTabStops"); CPPUNIT_ASSERT(aTabStops.hasElements()); // This was 3330 twips instead, as tabs were assumed to be relative. CPPUNIT_ASSERT_EQUAL(static_cast(convertTwipToMm100(450)), aTabStops[0].Position); } DECLARE_RTFEXPORT_TEST(testGraphicObjectFliph, "graphic-object-fliph.rtf") { CPPUNIT_ASSERT(getProperty(getShape(1), "HoriMirroredOnEvenPages")); CPPUNIT_ASSERT(getProperty(getShape(1), "HoriMirroredOnOddPages")); } DECLARE_RTFEXPORT_TEST(testTdf114333, "tdf114333.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); // Check the distance from left CPPUNIT_ASSERT_EQUAL(sal_Int32(8502), getProperty(xTable, "LeftMargin")); // This was 17000 = 8502 + 8498 on import, 15240 on export and following import CPPUNIT_ASSERT_EQUAL(sal_Int32(8498), getProperty(xTable, "Width")); } DECLARE_RTFEXPORT_TEST(testTdf115180, "tdf115180.docx") { // On export to RTF, column separator positions were written without taking base width // into account and then arrived huge, ~64000, which resulted in wrong table and cell widths sal_Int32 rowWidth = parseDump("/root/page/body/tab/row/infos/bounds", "width").toInt32(); CPPUNIT_ASSERT_EQUAL_MESSAGE("Row width", sal_Int32(9360), rowWidth); sal_Int32 cell1Width = parseDump("/root/page/body/tab/row/cell[1]/infos/bounds", "width").toInt32(); CPPUNIT_ASSERT_MESSAGE("First cell width", cell1Width >= 9140); CPPUNIT_ASSERT_MESSAGE("First cell width", cell1Width <= 9142); sal_Int32 cell2Width = parseDump("/root/page/body/tab/row/cell[2]/infos/bounds", "width").toInt32(); CPPUNIT_ASSERT_MESSAGE("Second cell width", cell2Width >= 218); CPPUNIT_ASSERT_MESSAGE("Second cell width", cell2Width <= 220); } DECLARE_ODFEXPORT_TEST(testArabicZeroNumbering, "arabic-zero-numbering.rtf") { auto xNumberingRules = getProperty>(getParagraph(1), "NumberingRules"); comphelper::SequenceAsHashMap aMap(xNumberingRules->getByIndex(0)); // Without the accompanying fix in place, this test would have failed with: // - Expected: 64 // - Actual : 4 // i.e. numbering type was ARABIC, not ARABIC_ZERO. CPPUNIT_ASSERT_EQUAL(o3tl::narrowing(style::NumberingType::ARABIC_ZERO), aMap["NumberingType"].get()); } DECLARE_RTFEXPORT_TEST(testTdf116841, "tdf116841.rtf") { // This was 0, left margin was ignored as we assumed the default is already // fine for us. CPPUNIT_ASSERT_EQUAL(static_cast(1270), getProperty(getParagraph(1), "ParaLeftMargin")); } DECLARE_RTFEXPORT_TEST(testTdf117268, "tdf117268.rtf") { // Here we check that we correctly mimic Word's treatment of erroneous \itap0 inside tables. // Previously, the first table was import as text, and second top-level one only imported // last row with nested table (first row was also imported as text). uno::Reference xTablesSupplier(mxComponent, uno::UNO_QUERY_THROW); uno::Reference xTables(xTablesSupplier->getTextTables(), uno::UNO_QUERY_THROW); // First (simple) table uno::Reference xTable(xTables->getByIndex(0), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTable->getRows()->getCount()); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTable->getColumns()->getCount()); uno::Reference xCell(xTable->getCellByName("A1"), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(OUString("Text 1"), xCell->getString()); // Nested table xTable.set(xTables->getByIndex(1), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTable->getRows()->getCount()); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTable->getColumns()->getCount()); xCell.set(xTable->getCellByName("A1"), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(OUString("Text 3"), xCell->getString()); uno::Reference xNestedAnchor(xTable->getAnchor(), uno::UNO_QUERY_THROW); uno::Reference xAnchorCell(xNestedAnchor->getPropertyValue("Cell"), uno::UNO_QUERY_THROW); // Outer table xTable.set(xTables->getByIndex(2), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(sal_Int32(2), xTable->getRows()->getCount()); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), xTable->getColumns()->getCount()); xCell.set(xTable->getCellByName("A1"), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(OUString("Text 2"), xCell->getString()); xCell.set(xTable->getCellByName("A2"), uno::UNO_QUERY_THROW); CPPUNIT_ASSERT_EQUAL(xCell, xAnchorCell); } DECLARE_RTFEXPORT_TEST(testTdf117505, "tdf117505.odt") { CPPUNIT_ASSERT_EQUAL(1, getShapes()); CPPUNIT_ASSERT_EQUAL(1, getPages()); uno::Reference xPageStyles(getStyles("PageStyles")); uno::Reference xFirstPage(xPageStyles->getByName("First Page"), uno::UNO_QUERY); // This was 499, small header height resulted in visible whitespace from // remaining top margin -> header content moved down. CPPUNIT_ASSERT_EQUAL(static_cast(1499), getProperty(xFirstPage, "HeaderHeight")); } DECLARE_RTFEXPORT_TEST(testTdf112520, "tdf112520.docx") { if (!mbExported) return; // Assert that the white shape is on top of the yellow one. CPPUNIT_ASSERT_EQUAL(Color(0xffff00), getProperty(getShape(2), "FillColor")); CPPUNIT_ASSERT_EQUAL(text::TextContentAnchorType_AT_CHARACTER, getProperty(getShape(2), "AnchorType")); CPPUNIT_ASSERT_EQUAL(COL_WHITE, getProperty(getShape(3), "FillColor")); // Without the accompanying fix in place, this test would have failed with // 'expected: 4, actual: 2'. // This means the draw page was 0/at-char/white, 1/at-char/yellow, 2/at-page/white, // instead of the good 0/at-page/white, 1/at-char/yellow, 2/at-char/white. CPPUNIT_ASSERT_EQUAL(text::TextContentAnchorType_AT_CHARACTER, getProperty(getShape(3), "AnchorType")); } CPPUNIT_TEST_FIXTURE(Test, testNestedHyperlink) { // Given a hyperlink contains a footnote which contains a hyperlink: { createSwDoc(); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xFootnote( xFactory->createInstance("com.sun.star.text.Footnote"), uno::UNO_QUERY); uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xText = xTextDocument->getText(); uno::Reference xCursor = xText->createTextCursor(); xText->insertString(xCursor, "a", /*bAbsorb=*/false); xText->insertTextContent(xCursor, xFootnote, /*bAbsorb=*/false); xText->insertString(xCursor, "b", /*bAbsorb=*/false); xCursor->gotoStart(/*bExpand=*/false); xCursor->gotoEnd(/*bExpand=*/true); uno::Reference xCursorProps(xCursor, uno::UNO_QUERY); xCursorProps->setPropertyValue("HyperLinkURL", uno::Any(OUString("http://body.com/"))); uno::Reference xFootnoteText(xFootnote, uno::UNO_QUERY); xCursor = xFootnoteText->createTextCursor(); xFootnoteText->insertString(xCursor, "x", /*bAbsorb=*/false); xCursor->gotoStart(/*bExpand=*/false); xCursor->gotoEnd(/*bExpand=*/true); xCursorProps.set(xCursor, uno::UNO_QUERY); xCursorProps->setPropertyValue("HyperLinkURL", uno::Any(OUString("http://footnote.com/"))); } // When exporting to RTF: // Without the accompanying fix in place, this test would have failed with: // assertion failed // - Expression: xComponent.is() // i.e. the RTF output was not well-formed, loading failed. reload(mpFilter, "nested-hyperlink.rtf"); // Then make sure both hyperlinks are have the correct URLs. uno::Reference xParagraph = getParagraph(1); uno::Reference xPortion = getRun(xParagraph, 1); CPPUNIT_ASSERT_EQUAL(OUString("http://body.com/"), getProperty(xPortion, "HyperLinkURL")); auto xFootnote = getProperty>(getRun(xParagraph, 2), "Footnote"); uno::Reference xFootnotePortion = getRun(getParagraphOfText(1, xFootnote), 1); CPPUNIT_ASSERT_EQUAL(OUString("http://footnote.com/"), getProperty(xFootnotePortion, "HyperLinkURL")); } DECLARE_RTFEXPORT_TEST(testTdf121623, "tdf121623.rtf") { // This was 2, multicolumn section was ignored at the table. CPPUNIT_ASSERT_EQUAL(1, getPages()); } DECLARE_RTFEXPORT_TEST(testTdf66543, "tdf66543.rtf") { // Without the accompanying fix in place, this test would have failed with // 'Expected: 2; Actual : 3' after import (off-by-one), then with // 'Expected: 2; Actual : 0' (export not implemented). CPPUNIT_ASSERT_EQUAL(sal_Int32(2), getProperty(getParagraph(1), "ParaLineNumberStartValue")); } DECLARE_RTFEXPORT_TEST(testUlw, "ulw.rtf") { // Test underlying in individual words mode. CPPUNIT_ASSERT(getProperty(getRun(getParagraph(1), 1), "CharWordMode")); } DECLARE_RTFEXPORT_TEST(testTdf122455, "tdf122455.rtf") { // Without the accompanying fix in place, this test would have failed with // 'Expected: 16; Actual : 32', the font size from a list definition // leaked into the first run's character properties. CPPUNIT_ASSERT_EQUAL(16.0, getProperty(getRun(getParagraph(1), 1), "CharHeight")); } DECLARE_RTFEXPORT_TEST(testTdf125719_case_1, "tdf125719_case_1.rtf") { CPPUNIT_ASSERT_EQUAL(awt::FontWeight::NORMAL, getProperty(getRun(getParagraph(1), 1), "CharWeight")); CPPUNIT_ASSERT_EQUAL(awt::FontWeight::NORMAL, getProperty(getRun(getParagraph(3), 1), "CharWeight")); } DECLARE_RTFEXPORT_TEST(testTdf125719_case_2, "tdf125719_case_2.rtf") { CPPUNIT_ASSERT_EQUAL(awt::FontWeight::BOLD, getProperty(getRun(getParagraph(1), 1), "CharWeight")); CPPUNIT_ASSERT_EQUAL(awt::FontWeight::BOLD, getProperty(getRun(getParagraph(3), 1), "CharWeight")); CPPUNIT_ASSERT_EQUAL(awt::FontWeight::NORMAL, getProperty(getRun(getParagraph(5), 1), "CharWeight")); } DECLARE_RTFEXPORT_TEST(testTabs, "tabs.rtf") { // Test tab alignment in decimal mode. auto aTabStops = getProperty>(getParagraph(1), "ParaTabStops"); CPPUNIT_ASSERT_EQUAL(static_cast(1), aTabStops.getLength()); const style::TabStop& rTabStop = aTabStops[0]; CPPUNIT_ASSERT_EQUAL(style::TabAlign_DECIMAL, rTabStop.Alignment); } DECLARE_RTFEXPORT_TEST(testTdf123703, "tdf123703.rtf") { #if !defined(MACOSX) // This was 1, because of normal space character width in consecutive spaces CPPUNIT_ASSERT_EQUAL(2, getPages()); #else // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); #endif } DECLARE_RTFEXPORT_TEST(testTdf123703_stshfdbch, "tdf123703_stshfdbch.rtf") { // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); } DECLARE_RTFEXPORT_TEST(testTdf123703_compatible, "tdf123703_compatible.rtf") { #if !defined(MACOSX) // in the case of compatibility font id 31505 CPPUNIT_ASSERT_EQUAL(2, getPages()); #else // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); #endif } DECLARE_RTFEXPORT_TEST(testTdf128428_monospaced, "tdf128428_monospaced.rtf") { // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); } DECLARE_RTFEXPORT_TEST(testTdf128428_compatible_monospaced, "tdf128428_compatible_monospaced.rtf") { // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); } DECLARE_RTFEXPORT_TEST(testTdf128428_dntblnsbdb, "tdf128428_dntblnsbdb.rtf") { // still 1 here CPPUNIT_ASSERT_EQUAL(1, getPages()); } CPPUNIT_TEST_FIXTURE(Test, testRtlGutter) { auto verify = [this]() { uno::Reference xStandard( getStyles("PageStyles")->getByName("Standard"), uno::UNO_QUERY); CPPUNIT_ASSERT(getProperty(xStandard, "RtlGutter")); }; // Given a document with RTL gutter, when loading it: load(mpTestDocumentPath, "rtl-gutter.rtf"); // Then make sure the section's gutter is still RTL: // Without the accompanying fix in place, this test would have failed as \rtlgutter was missing. verify(); reload(mpFilter, "rtl-gutter.rtf"); verify(); } CPPUNIT_TEST_FIXTURE(Test, testNegativePageBorder) { { // Given a document with a top margin and a border which has more spacing than the margin on // its 2nd page: createSwDoc(); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pDocShell = pTextDoc->GetDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); pWrtShell->Insert("first"); pWrtShell->SplitNode(); pWrtShell->Insert("second"); SwPageDesc* pDesc = pWrtShell->FindPageDescByName("Left Page", true); SwPaM aPaM(*pWrtShell->GetCursor()->GetPoint()); SwFormatPageDesc aFormatPageDesc(pDesc); pDocShell->GetDoc()->getIDocumentContentOperations().InsertPoolItem(aPaM, aFormatPageDesc); uno::Reference xPageStyle( getStyles("PageStyles")->getByName("Left Page"), uno::UNO_QUERY); xPageStyle->setPropertyValue("TopMargin", uno::Any(static_cast(501))); table::BorderLine2 aBorder; aBorder.LineWidth = 159; aBorder.OuterLineWidth = 159; xPageStyle->setPropertyValue("TopBorder", uno::Any(aBorder)); sal_Int32 nTopBorderDistance = -646; xPageStyle->setPropertyValue("TopBorderDistance", uno::Any(nTopBorderDistance)); pDocShell->GetDoc()->dumpAsXml(); } // When saving that document to RTF: reload(mpFilter, "negative-page-border.rtf"); // Then make sure that the border distance is negative, so the first line of body text appears // on top of the page border: SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pDocShell = pTextDoc->GetDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); pWrtShell->Down(/*bSelect=*/false); OUString aPageStyle = pWrtShell->GetCurPageStyle(); uno::Reference xPageStyle(getStyles("PageStyles")->getByName(aPageStyle), uno::UNO_QUERY); auto nTopMargin = xPageStyle->getPropertyValue("TopMargin").get(); CPPUNIT_ASSERT_EQUAL(static_cast(501), nTopMargin); auto aTopBorder = xPageStyle->getPropertyValue("TopBorder").get(); CPPUNIT_ASSERT_EQUAL(static_cast(159), aTopBorder.LineWidth); auto nTopBorderDistance = xPageStyle->getPropertyValue("TopBorderDistance").get(); // Without the accompanying fix in place, this test would have failed with: // - Expected: -646 // - Actual : 0 // i.e. the border negative distance was lost. CPPUNIT_ASSERT_EQUAL(static_cast(-646), nTopBorderDistance); } CPPUNIT_TEST_FIXTURE(Test, testTdf127806) { load(mpTestDocumentPath, "tdf127806.rtf"); CPPUNIT_ASSERT_EQUAL(2, getShapes()); CPPUNIT_ASSERT_EQUAL(OUString("com.sun.star.drawing.GroupShape"), getShape(1)->getShapeType()); auto xImage = getShape(2); CPPUNIT_ASSERT_EQUAL(OUString("FrameShape"), xImage->getShapeType()); awt::Size aSize(xImage->getSize()); CPPUNIT_ASSERT_EQUAL(static_cast(600), aSize.Height); CPPUNIT_ASSERT_EQUAL(static_cast(635), aSize.Width); reload(mpFilter, "tdf127806.rtf"); CPPUNIT_ASSERT_EQUAL(1, getShapes()); // FIXME: We lost one shape on export, that's sucks xImage = getShape(1); CPPUNIT_ASSERT_EQUAL(OUString("FrameShape"), xImage->getShapeType()); aSize = xImage->getSize(); CPPUNIT_ASSERT_EQUAL(static_cast(600), aSize.Height); CPPUNIT_ASSERT_EQUAL(static_cast(635), aSize.Width); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */