/* -*- 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 /// Covers sw/source/core/doc/ fixes. class SwCoreDocTest : public SwModelTestBase { public: SwCoreDocTest() : SwModelTestBase(u"/sw/qa/core/doc/data/"_ustr) { } }; CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testMathInsertAnchorType) { // Given an empty document. createSwDoc(); SwDoc* pDoc = getSwDoc(); // When inserting an a math object. SwWrtShell* pShell = getSwDocShell()->GetWrtShell(); SvGlobalName aGlobalName(SO3_SM_CLASSID); pShell->InsertObject(svt::EmbeddedObjectRef(), &aGlobalName); // Then the anchor type should be as-char. sw::SpzFrameFormats& rFormats = *pDoc->GetSpzFrameFormats(); CPPUNIT_ASSERT_EQUAL(static_cast(1), rFormats.size()); const SwFrameFormat& rFormat = *rFormats[0]; const SwFormatAnchor& rAnchor = rFormat.GetAnchor(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 4 // i.e. the anchor type was at-char, not as-char. CPPUNIT_ASSERT_EQUAL(RndStdIds::FLY_AS_CHAR, rAnchor.GetAnchorId()); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testTextboxTextRotateAngle) { // Check the writing direction of the only TextFrame in the document. createSwDoc("textbox-textrotateangle.odt"); SwDoc* pDoc = getSwDoc(); sw::SpzFrameFormats& rFrameFormats = *pDoc->GetSpzFrameFormats(); CPPUNIT_ASSERT_EQUAL(static_cast(2), rFrameFormats.size()); CPPUNIT_ASSERT_EQUAL(o3tl::narrowing(RES_DRAWFRMFMT), rFrameFormats[0]->Which()); CPPUNIT_ASSERT_EQUAL(o3tl::narrowing(RES_FLYFRMFMT), rFrameFormats[1]->Which()); SvxFrameDirection eActual = rFrameFormats[1]->GetAttrSet().GetItem(RES_FRAMEDIR)->GetValue(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 5 (btlr) // - Actual : 0 (lrtb) // i.e. the writing direction was in the ODT file, but it was lost on import in the textbox // case. CPPUNIT_ASSERT_EQUAL(SvxFrameDirection::Vertical_LR_BT, eActual); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testNumDownIndent) { createSwDoc("num-down-indent.docx"); SwDocShell* pDocShell = getSwDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); pWrtShell->Down(/*bSelect=*/false); SwEditWin& rEditWin = pDocShell->GetView()->GetEditWin(); KeyEvent aKeyEvent(0, KEY_TAB); rEditWin.KeyInput(aKeyEvent); SwTextNode* pTextNode = pWrtShell->GetCursor()->GetPointNode().GetTextNode(); // Without the accompanying fix in place, this test would have failed with: // - Expected: \tB // - Actual : B // i.e. pressing at the start of the paragraph did not change the layout. CPPUNIT_ASSERT_EQUAL(u"\tB"_ustr, pTextNode->GetText()); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testBulletsOnSpaceOff) { SvxAutoCorrect* pAutoCorrect = SvxAutoCorrCfg::Get().GetAutoCorrect(); pAutoCorrect->GetSwFlags().bSetNumRule = false; pAutoCorrect->GetSwFlags().bSetNumRuleAfterSpace = false; createSwDoc(); SwDocShell* pDocShell = getSwDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); pWrtShell->Down(/*bSelect=*/false); SwEditWin& rEditWin = pDocShell->GetView()->GetEditWin(); KeyEvent aKeyEvent('-', 0); rEditWin.KeyInput(aKeyEvent); KeyEvent aKeyEvent2(' ', KEY_SPACE); rEditWin.KeyInput(aKeyEvent2); KeyEvent aKeyEvent3('a', 0); rEditWin.KeyInput(aKeyEvent3); SwTextNode* pTextNode = pWrtShell->GetCursor()->GetPointNode().GetTextNode(); CPPUNIT_ASSERT_EQUAL(u"- a"_ustr, pTextNode->GetText()); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testBulletsOnSpace) { SvxAutoCorrect* pAutoCorrect = SvxAutoCorrCfg::Get().GetAutoCorrect(); pAutoCorrect->GetSwFlags().bSetNumRule = true; pAutoCorrect->GetSwFlags().bSetNumRuleAfterSpace = true; createSwDoc(); SwDocShell* pDocShell = getSwDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); pWrtShell->Down(/*bSelect=*/false); SwEditWin& rEditWin = pDocShell->GetView()->GetEditWin(); KeyEvent aKeyEvent('-', 0); rEditWin.KeyInput(aKeyEvent); KeyEvent aKeyEvent2(' ', KEY_SPACE); rEditWin.KeyInput(aKeyEvent2); KeyEvent aKeyEvent3('a', 0); rEditWin.KeyInput(aKeyEvent3); SwTextNode* pTextNode = pWrtShell->GetCursor()->GetPointNode().GetTextNode(); // '- ' was converted into bullet CPPUNIT_ASSERT_EQUAL(u"a"_ustr, pTextNode->GetText()); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testLocaleIndependentTemplate) { createSwDoc("locale-independent-template.odt"); SwDocShell* pDocShell = getSwDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); SfxItemSet aSet(pWrtShell->GetAttrPool(), svl::Items); pWrtShell->GetCurAttr(aSet); const SvxLanguageItem* pItem = aSet.GetItem(RES_CHRATR_LANGUAGE); CPPUNIT_ASSERT(pItem); LanguageType eLang = pItem->GetValue(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1033 (LANGUAGE_ENGLISH_US) // - Actual : 1023 (LANGUAGE_DONTKNOW) // i.e. the status bar and the format -> character dialog didn't fall back to the UI locale when // an explicit language was not set for the document. CPPUNIT_ASSERT_EQUAL(LANGUAGE_ENGLISH_US, eLang); ErrorRegistry::Reset(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testTextBoxZOrder) { createSwDoc("textbox-zorder.docx"); SwDoc* pDoc = getSwDoc(); sw::SpzFrameFormats& rFormats = *pDoc->GetSpzFrameFormats(); CPPUNIT_ASSERT_EQUAL(static_cast(3), rFormats.size()); const sw::SpzFrameFormat* pEllipse = rFormats[2]; const SdrObject* pEllipseShape = pEllipse->FindRealSdrObject(); // Make sure we test the right shape. CPPUNIT_ASSERT_EQUAL(u"Shape3"_ustr, pEllipseShape->GetName()); // Without the accompanying fix in place, this test would have failed with: // - Expected: 2 // - Actual : 1 // i.e. the ellipse was under the frame of the shape-frame pair, not on top of it. CPPUNIT_ASSERT_EQUAL(static_cast(2), pEllipseShape->GetOrdNum()); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testTextBoxMakeFlyFrame) { // Given a document with an as-char textbox (as-char draw format + at-char fly format): createSwDoc("textbox-makeflyframe.docx"); // When cutting the textbox and pasting it to a new document: selectShape(1); SwDocShell* pDocShell = getSwDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); rtl::Reference pTransfer = new SwTransferable(*pWrtShell); pTransfer->Cut(); TransferableDataHelper aHelper(pTransfer); mxComponent2 = loadFromDesktop(u"private:factory/swriter"_ustr, u"com.sun.star.text.TextDocument"_ustr, {}); SwXTextDocument* pTextDoc2 = dynamic_cast(mxComponent2.get()); SwDocShell* pDocShell2 = pTextDoc2->GetDocShell(); SwWrtShell* pWrtShell2 = pDocShell2->GetWrtShell(); SwTransferable::Paste(*pWrtShell2, aHelper); // Then make sure its fly frame is created. xmlDocUniquePtr pLayout = parseLayoutDump(mxComponent2); // Without the accompanying fix in place, this test would have failed, because the first text // frame in the body frame had an SwAnchoredDrawObject anchored to it, but not a fly frame, so // a blank square was painted, not the image. assertXPath(pLayout, "/root/page/body/txt/anchored/fly", 1); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testIMEGrouping) { // TODO figure out why the ext text input in this test code reaches the wrong window on // non-headless. #if !defined MACOSX && !defined _WIN32 // Given an empty document: createSwDoc(); SwDoc* pDoc = getSwDoc(); // Make sure no idle is in action, so the ExtTextInput events go to SwEditWin. Scheduler::ProcessEventsToIdle(); // When pressing two keys via IME: SwDocShell* pDocShell = getSwDocShell(); SwEditWin& rEditWin = pDocShell->GetView()->GetEditWin(); rEditWin.PostExtTextInputEvent(VclEventId::ExtTextInput, u"a"_ustr); rEditWin.PostExtTextInputEvent(VclEventId::EndExtTextInput, u""_ustr); rEditWin.PostExtTextInputEvent(VclEventId::ExtTextInput, u"b"_ustr); rEditWin.PostExtTextInputEvent(VclEventId::EndExtTextInput, u""_ustr); // Then make sure that gets grouped together to a single undo action: SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); SwTextNode* pTextNode = pWrtShell->GetCursor()->GetPointNode().GetTextNode(); CPPUNIT_ASSERT_EQUAL(u"ab"_ustr, pTextNode->GetText()); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. 2 subsequent IME events got their own undo actions. CPPUNIT_ASSERT_EQUAL(static_cast(1), pDoc->GetUndoManager().GetUndoActionCount()); #endif } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testImageHyperlinkStyle) { // Given a document with an image with a hyperlink: createSwDoc(); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xDocument(mxComponent, uno::UNO_QUERY); uno::Reference xText = xDocument->getText(); uno::Reference xCursor = xText->createTextCursor(); uno::Reference xImage( xFactory->createInstance(u"com.sun.star.text.TextGraphicObject"_ustr), uno::UNO_QUERY); xText->insertTextContent(xCursor, xImage, /*bAbsorb=*/false); uno::Reference xImageProps(xImage, uno::UNO_QUERY); OUString aExpected = u"http://www.example.com"_ustr; xImageProps->setPropertyValue(u"HyperLinkURL"_ustr, uno::Any(aExpected)); // When applying a frame style on it: xImageProps->setPropertyValue(u"FrameStyleName"_ustr, uno::Any(u"Frame"_ustr)); // Then make sure that the hyperlink is not lost: auto aActual = getProperty(xImageProps, u"HyperLinkURL"_ustr); // Without the accompanying fix in place, this test would have failed with: // - Expected: http://www.example.com // - Actual : // i.e. the link was lost, even if the frame style dialog doesn't allow specifying a link on // frames. CPPUNIT_ASSERT_EQUAL(aExpected, aActual); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testContentControlDelete) { // Given a document with a content control: createSwDoc(); uno::Reference xMSF(mxComponent, uno::UNO_QUERY); uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xText = xTextDocument->getText(); uno::Reference xCursor = xText->createTextCursor(); xText->insertString(xCursor, u"test"_ustr, /*bAbsorb=*/false); xCursor->gotoStart(/*bExpand=*/false); xCursor->gotoEnd(/*bExpand=*/true); uno::Reference xContentControl( xMSF->createInstance(u"com.sun.star.text.ContentControl"_ustr), uno::UNO_QUERY); xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true); // When deleting the dummy character at the end of the content control: SwWrtShell* pWrtShell = getSwDocShell()->GetWrtShell(); pWrtShell->SttEndDoc(/*bStt=*/false); pWrtShell->DelLeft(); // Then make sure that we only enter the content control, to be consistent with the start dummy // character: SwTextNode* pTextNode = pWrtShell->GetCursor()->GetMark()->GetNode().GetTextNode(); // Without the accompanying fix in place, this test would have failed with: // - Expected: ^Atest^A // - Actual : ^Atest // i.e. the end dummy character got deleted, but not the first one, which is inconsistent. CPPUNIT_ASSERT_EQUAL(u"\x0001test\x0001"_ustr, pTextNode->GetText()); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testCopyBookmarks) { // Given a document with a bookmark in a header that is linked later: createSwDoc("copy-bookmarks.docx"); SwDoc* pDoc = getSwDoc(); // When checking the # of non-copy bookmarks in the resulting doc model: sal_Int32 nActual = 0; for (auto it = pDoc->getIDocumentMarkAccess()->getBookmarksBegin(); it != pDoc->getIDocumentMarkAccess()->getBookmarksEnd(); ++it) { if ((*it)->GetName().indexOf("Copy") == -1) { ++nActual; } } // Then make sure we have a single non-copy bookmark, with no duplications: // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. the 2nd header had a duplicated bookmark without "Copy" in its name. CPPUNIT_ASSERT_EQUAL(static_cast(1), nActual); // Also, when checking the # of non-copy images in the resulting doc model: nActual = 0; for (auto pSpz : *pDoc->GetSpzFrameFormats()) { if (pSpz->GetName().indexOf("Copy") == -1) { ++nActual; } } // Then make sure we have a single non-copy image, with no duplications: // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. the 2nd header had a duplicated image without "Copy" in its name. CPPUNIT_ASSERT_EQUAL(static_cast(1), nActual); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testLinkedStyleDelete) { // Given a document with linked styles: myparastyle is linked to mycharstyle and vica versa: createSwDoc(); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xParaStyle( xFactory->createInstance(u"com.sun.star.style.ParagraphStyle"_ustr), uno::UNO_QUERY); uno::Reference xCharStyle( xFactory->createInstance(u"com.sun.star.style.CharacterStyle"_ustr), uno::UNO_QUERY); uno::Reference xParaStyles(getStyles(u"ParagraphStyles"_ustr), uno::UNO_QUERY); xParaStyles->insertByName(u"myparastyle"_ustr, uno::Any(xParaStyle)); uno::Reference xCharStyles(getStyles(u"CharacterStyles"_ustr), uno::UNO_QUERY); xCharStyles->insertByName(u"mycharstyle"_ustr, uno::Any(xCharStyle)); xParaStyle->setPropertyValue(u"LinkStyle"_ustr, uno::Any(u"mycharstyle"_ustr)); xCharStyle->setPropertyValue(u"LinkStyle"_ustr, uno::Any(u"myparastyle"_ustr)); // When deleting the paragraph style (and only that): xParaStyles->removeByName(u"myparastyle"_ustr); // Then make sure we don't crash on save: uno::Reference xStorable(mxComponent, uno::UNO_QUERY); uno::Sequence aArgs = { comphelper::makePropertyValue(u"FilterName"_ustr, u"writer8"_ustr), }; xStorable->storeAsURL(maTempFile.GetURL(), aArgs); } namespace { /// This selection listener calls getAnchor() on selection change, which creates UNO cursors and is /// invoked in the middle of a bookmark deletion. struct SelectionChangeListener : public cppu::WeakImplHelper { uno::Reference m_xBookmarks; std::vector> m_aAnchors; public: SelectionChangeListener(const uno::Reference& xBookmarks); // view::XSelectionChangeListener void SAL_CALL selectionChanged(const lang::EventObject& rEvent) override; // lang::XEventListener void SAL_CALL disposing(const lang::EventObject& rSource) override; }; } SelectionChangeListener::SelectionChangeListener( const uno::Reference& xBookmarks) : m_xBookmarks(xBookmarks) { } void SelectionChangeListener::selectionChanged(const lang::EventObject& /*rEvent*/) { uno::Sequence aElementNames = m_xBookmarks->getElementNames(); for (const auto& rName : aElementNames) { uno::Reference xTextContent(m_xBookmarks->getByName(rName), uno::UNO_QUERY); m_aAnchors.push_back(xTextContent->getAnchor()); } } void SelectionChangeListener::disposing(const lang::EventObject& /*rSource*/) {} CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testBookmarkDeleteListeners) { // Given a document with 2 bookmarks: createSwDoc(); uno::Reference xTextDocument(mxComponent, uno::UNO_QUERY); uno::Reference xText = xTextDocument->getText(); uno::Reference xCursor = xText->createTextCursor(); { xText->insertString(xCursor, u"test"_ustr, /*bAbsorb=*/false); xCursor->gotoStart(/*bExpand=*/false); xCursor->gotoEnd(/*bExpand=*/true); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xBookmark( xFactory->createInstance(u"com.sun.star.text.Bookmark"_ustr), uno::UNO_QUERY); uno::Reference xBookmarkNamed(xBookmark, uno::UNO_QUERY); xBookmarkNamed->setName(u"mybookmark"_ustr); xText->insertTextContent(xCursor, xBookmark, /*bAbsorb=*/true); } { xCursor->gotoEnd(/*bExpand=*/false); xText->insertString(xCursor, u"test2"_ustr, /*bAbsorb=*/false); xCursor->goLeft(4, /*bExpand=*/true); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xBookmark( xFactory->createInstance(u"com.sun.star.text.Bookmark"_ustr), uno::UNO_QUERY); uno::Reference xBookmarkNamed(xBookmark, uno::UNO_QUERY); xBookmarkNamed->setName(u"mybookmark2"_ustr); xText->insertTextContent(xCursor, xBookmark, /*bAbsorb=*/true); } uno::Reference xBookmarksSupplier(mxComponent, uno::UNO_QUERY); uno::Reference xBookmarks = xBookmarksSupplier->getBookmarks(); // When registering a selection listener that creates uno marks: uno::Reference xModel(mxComponent, uno::UNO_QUERY); uno::Reference xController(xModel->getCurrentController(), uno::UNO_QUERY); xController->addSelectionChangeListener(new SelectionChangeListener(xBookmarks)); // Then make sure that deleting a bookmark doesn't crash: uno::Reference xBookmark(xBookmarks->getByName(u"mybookmark2"_ustr), uno::UNO_QUERY); // Without the accompanying fix in place, this test would have crashed, an invalidated iterator // was used with erase(). xBookmark->dispose(); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testBookmarkDeleteRedline) { // Given a document with redlines, a mark (annotation mark) inside a redline: createSwDoc("bookmark-delete-redline.doc"); SwDoc* pDoc = getSwDoc(); // When hiding deletions / showing only inserts, make sure we don't crash: // Without the accompanying fix in place, this test would have crashed, equal_range() was used // on an unsorted container. pDoc->getIDocumentRedlineAccess().SetRedlineFlags(RedlineFlags::ShowInsert); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testHeaderFooterDelete) { // Given a document with bookmarks in header/footers: // When importing that document: // Then make sure that we don't crash: // Without the accompanying fix in place, this test would have crashed, an invalidated iterator // was used in sw::mark::MarkManager::deleteMarks(). createSwDoc("header-footer-delete.docx"); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testSplitFlyChain) { // Given a document with 2 fly frames, first is allowed to split, second is not: createSwDoc(); SwWrtShell* pWrtShell = getSwDocShell()->GetWrtShell(); SwFlyFrameAttrMgr aMgr(true, pWrtShell, Frmmgr_Type::TEXT, nullptr); RndStdIds eAnchor = RndStdIds::FLY_AT_PARA; aMgr.InsertFlyFrame(eAnchor, aMgr.GetPos(), aMgr.GetSize()); SwDoc* pDoc = getSwDoc(); auto& rFlys = *pDoc->GetSpzFrameFormats(); { pWrtShell->StartAllAction(); auto pFly = rFlys[0]; SwAttrSet aSet(pFly->GetAttrSet()); aSet.Put(SwFormatFlySplit(true)); pDoc->SetAttr(aSet, *pFly); pWrtShell->EndAllAction(); } pWrtShell->UnSelectFrame(); pWrtShell->SttEndDoc(/*bStart=*/false); aMgr.InsertFlyFrame(eAnchor, aMgr.GetPos(), aMgr.GetSize()); auto pFly = rFlys[0]; auto pFly2 = rFlys[1]; // When checking if chaining is allowed: SwChainRet eActual = pDoc->Chainable(*pFly, *pFly2); // Then make sure the source is rejected if it is a split fly: // Without the accompanying fix in place, this test would have failed with: // - Expected: 5 (SwChainRet::SOURCE_CHAINED) // - Actual : 0 (SwChainRet::OK) // i.e. the UI allowed chaining for floating tables, which doesn't make sense. CPPUNIT_ASSERT_EQUAL(SwChainRet::SOURCE_CHAINED, eActual); // Also test the other way around, that should not be OK, either. eActual = pDoc->Chainable(*pFly2, *pFly); CPPUNIT_ASSERT_EQUAL(SwChainRet::IS_IN_CHAIN, eActual); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testSplitExpandGlossary) { // Given a document with a split fly (2 pages) and a 'dt' at the end: createSwDoc("floating-table-dummy-text.docx"); SwWrtShell* pWrtShell = getSwDocShell()->GetWrtShell(); pWrtShell->SttEndDoc(/*bStt=*/false); // When expanding 'dt' to an actual dummy text: dispatchCommand(mxComponent, u".uno:ExpandGlossary"_ustr, {}); // Then make sure the 2 fly frames stay on the 2 pages: SwDoc* pDoc = getSwDoc(); SwRootFrame* pLayout = pDoc->getIDocumentLayoutAccess().GetCurrentLayout(); auto pPage1 = dynamic_cast(pLayout->Lower()); CPPUNIT_ASSERT(pPage1); CPPUNIT_ASSERT(pPage1->GetSortedObjs()); const SwSortedObjs& rPage1Objs = *pPage1->GetSortedObjs(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. both parts of the split fly chain were on page 1. CPPUNIT_ASSERT_EQUAL(static_cast(1), rPage1Objs.size()); auto pPage2 = dynamic_cast(pPage1->GetNext()); CPPUNIT_ASSERT(pPage2); CPPUNIT_ASSERT(pPage2->GetSortedObjs()); const SwSortedObjs& rPage2Objs = *pPage2->GetSortedObjs(); CPPUNIT_ASSERT_EQUAL(static_cast(1), rPage2Objs.size()); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testSplitFlyInsertUndo) { // Given a document with an inline table, which is then turned into a floating one: createSwDoc(); SwDoc* pDoc = getSwDocShell()->GetDoc(); sw::FrameFormats& rFlyFormats = *pDoc->GetSpzFrameFormats(); CPPUNIT_ASSERT(rFlyFormats.empty()); // Insert a table: SwWrtShell* pWrtShell = getSwDocShell()->GetWrtShell(); SwInsertTableOptions aTableOptions(SwInsertTableFlags::DefaultBorder, 0); pWrtShell->InsertTable(aTableOptions, /*nRows=*/2, /*nCols=*/1); pWrtShell->MoveTable(GotoPrevTable, fnTableStart); pWrtShell->GoPrevCell(); pWrtShell->Insert(u"A1"_ustr); pWrtShell->GoNextCell(); pWrtShell->Insert(u"A2"_ustr); // Select cell: pWrtShell->SelAll(); // Select table: pWrtShell->SelAll(); // Wrap the table in a text frame: SwFlyFrameAttrMgr aMgr(true, pWrtShell, Frmmgr_Type::TEXT, nullptr); pWrtShell->StartAllAction(); aMgr.InsertFlyFrame(RndStdIds::FLY_AT_PARA, aMgr.GetPos(), aMgr.GetSize()); pWrtShell->EndAllAction(); CPPUNIT_ASSERT(!rFlyFormats.empty()); // When undoing the conversion to floating table: pDoc->GetUndoManager().Undo(); // Then make sure we don't have a frame anymore: // Without the accompanying fix in place, this test would have failed, there was no undo action // to execute. CPPUNIT_ASSERT(rFlyFormats.empty()); // And when redoing the conversion: pDoc->GetUndoManager().Redo(); // Then make sure we again have a frame: CPPUNIT_ASSERT(!rFlyFormats.empty()); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testVirtPageNumReset) { createSwDoc("tdf160843.odt"); auto pWrtShell = getSwDocShell()->GetWrtShell(); sal_uInt16 nPhys = 0; sal_uInt16 nVirt = 0; pWrtShell->GotoPage(1, false); pWrtShell->GetPageNum(nPhys, nVirt, true, false); CPPUNIT_ASSERT_EQUAL(static_cast(1), nPhys); CPPUNIT_ASSERT_EQUAL(static_cast(1), nVirt); pWrtShell->GotoPage(3, false); pWrtShell->GetPageNum(nPhys, nVirt, true, false); CPPUNIT_ASSERT_EQUAL(static_cast(3), nPhys); CPPUNIT_ASSERT_EQUAL(static_cast(1), nVirt); pWrtShell->GotoPage(5, false); pWrtShell->GetPageNum(nPhys, nVirt, true, false); CPPUNIT_ASSERT_EQUAL(static_cast(5), nPhys); CPPUNIT_ASSERT_EQUAL(static_cast(1), nVirt); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testDefaultPageDescUsed) { createSwDoc(); auto pDoc = getSwDocShell()->GetDoc(); CPPUNIT_ASSERT_EQUAL(static_cast(1), pDoc->GetPageDescCnt()); const SwPageDesc& rPageDesc = pDoc->GetPageDesc(0); CPPUNIT_ASSERT(rPageDesc.IsUsed()); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testTextBoxWordWrap) { // Given a document with a shape in the header that extends horizontally when there is enough // content: createSwDoc("text-box-word-wrap.docx"); // When checking the layout size of the shape: xmlDocUniquePtr pXmlDoc = parseLayoutDump(); auto nFlyHeight = getXPath(pXmlDoc, "//anchored/fly/infos/bounds", "height").toInt32(); // Then make sure it has a small height, hosting just one line: // Without the accompanying fix in place, this test would have failed with: // - Expected: 773 // - Actual : 5183 // i.e. the shape had new lines for each character instead of 1 line. CPPUNIT_ASSERT_LESS(static_cast(1000), nFlyHeight); } CPPUNIT_TEST_FIXTURE(SwCoreDocTest, testAtCharImageCopy) { // Given a document with 2 pages, one draw object on both pages: createSwDoc("at-char-image-copy.odt"); SwWrtShell* pWrtShell1 = getSwDocShell()->GetWrtShell(); pWrtShell1->SelAll(); rtl::Reference xTransfer = new SwTransferable(*pWrtShell1); xTransfer->Copy(); // Don't use createSwDoc(), UnoApiTest::loadWithParams() would dispose the first document. mxComponent2 = loadFromDesktop(u"private:factory/swriter"_ustr); // When copying the body text from that document to a new one: auto pXTextDocument2 = dynamic_cast(mxComponent2.get()); SwDocShell* pDocShell2 = pXTextDocument2->GetDocShell(); SwWrtShell* pWrtShell2 = pDocShell2->GetWrtShell(); TransferableDataHelper aHelper(xTransfer); SwTransferable::Paste(*pWrtShell2, aHelper); // Then make sure the new document also has the 2 images on the 2 pages: SwDoc* pDoc2 = pDocShell2->GetDoc(); SwRootFrame* pLayout2 = pDoc2->getIDocumentLayoutAccess().GetCurrentLayout(); auto pPage1 = pLayout2->GetLower()->DynCastPageFrame(); CPPUNIT_ASSERT(pPage1); CPPUNIT_ASSERT(pPage1->GetSortedObjs()); const SwSortedObjs& rPage1Objs = *pPage1->GetSortedObjs(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. both images went to page 1. CPPUNIT_ASSERT_EQUAL(static_cast(1), rPage1Objs.size()); auto pPage2 = pPage1->GetNext()->DynCastPageFrame(); CPPUNIT_ASSERT(pPage2); CPPUNIT_ASSERT(pPage2->GetSortedObjs()); const SwSortedObjs& rPage2Objs = *pPage2->GetSortedObjs(); CPPUNIT_ASSERT_EQUAL(static_cast(1), rPage2Objs.size()); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */