/* -*- 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 constexpr OUStringLiteral DATA_DIRECTORY = u"/sw/qa/core/txtnode/data/"; /// Covers sw/source/core/txtnode/ fixes. class SwCoreTxtnodeTest : public SwModelTestBase { }; CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testBtlrCellChinese) { // Load a document with a table cell, with btlr cell direction. The cell has text which is // classified as vertical, i.e. the glyph has the same direction in both the lrtb ("Latin") and // tbrl ("Chinese") directions. Make sure that Chinese text is handled the same way in the btlr // case as it's handled in the Latin case. load(DATA_DIRECTORY, "btlr-cell-chinese.doc"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pShell = pTextDoc->GetDocShell(); std::shared_ptr xMetaFile = pShell->GetPreviewMetaFile(); MetafileXmlDump dumper; xmlDocUniquePtr pXmlDoc = dumpAndParse(dumper, *xMetaFile); assertXPath(pXmlDoc, "//font[1]", "orientation", "900"); // Without the accompanying fix in place, this test would have failed with: // - Expected: false // - Actual : true // i.e. the glyph was rotated further, so it was upside down. assertXPath(pXmlDoc, "//font[1]", "vertical", "false"); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testTextBoxCopyAnchor) { load(DATA_DIRECTORY, "textbox-copy-anchor.docx"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pShell = pTextDoc->GetDocShell(); SwWrtShell* pWrtShell = pShell->GetWrtShell(); SwDoc aClipboard; pWrtShell->SelAll(); pWrtShell->Copy(aClipboard); pWrtShell->SttEndDoc(/*bStart=*/false); pWrtShell->Paste(aClipboard); const SwFrameFormats& rFormats = *pShell->GetDoc()->GetSpzFrameFormats(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 4 // - Actual : 6 // i.e. 2 fly frames were copied twice. CPPUNIT_ASSERT_EQUAL(static_cast(4), rFormats.size()); SwPosition aDrawAnchor1 = *rFormats[0]->GetAnchor().GetContentAnchor(); SwPosition aFlyAnchor1 = *rFormats[1]->GetAnchor().GetContentAnchor(); CPPUNIT_ASSERT_EQUAL(aFlyAnchor1.nNode, aDrawAnchor1.nNode); SwPosition aDrawAnchor2 = *rFormats[2]->GetAnchor().GetContentAnchor(); SwPosition aFlyAnchor2 = *rFormats[3]->GetAnchor().GetContentAnchor(); // This also failed, aFlyAnchor2 was wrong, as it got out of sync with aDrawAnchor2. CPPUNIT_ASSERT_EQUAL(aFlyAnchor2.nNode, aDrawAnchor2.nNode); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testTextBoxNodeSplit) { load(DATA_DIRECTORY, "textbox-node-split.docx"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pShell = pTextDoc->GetDocShell(); SwWrtShell* pWrtShell = pShell->GetWrtShell(); pWrtShell->SttEndDoc(/*bStart=*/false); // Without the accompanying fix in place, this would have crashed in // SwFlyAtContentFrame::SwClientNotify(). pWrtShell->SplitNode(); } namespace { struct ViewCallback { int m_nInvalidations = 0; static void callback(int nType, const char* pPayload, void* pData); void callbackImpl(int nType, const char* pPayload); }; void ViewCallback::callback(int nType, const char* pPayload, void* pData) { static_cast(pData)->callbackImpl(nType, pPayload); } void ViewCallback::callbackImpl(int nType, const char* /*pPayload*/) { switch (nType) { case LOK_CALLBACK_INVALIDATE_TILES: { ++m_nInvalidations; } break; } } } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testTitleFieldInvalidate) { // Set up LOK to track invalidations. comphelper::LibreOfficeKit::setActive(true); // Given a document with a title field: load(DATA_DIRECTORY, "title-field-invalidate.fodt"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); pTextDoc->initializeForTiledRendering({}); SwDocShell* pShell = pTextDoc->GetDocShell(); SwDoc* pDoc = pShell->GetDoc(); SwWrtShell* pWrtShell = pShell->GetWrtShell(); pWrtShell->SttEndDoc(/*bStt=*/false); ViewCallback aCallback; TestLokCallbackWrapper aCallbackWrapper(&ViewCallback::callback, &aCallback); pWrtShell->GetSfxViewShell()->setLibreOfficeKitViewCallback(&aCallbackWrapper); aCallbackWrapper.setLOKViewId(SfxLokHelper::getView(pWrtShell->GetSfxViewShell())); Scheduler::ProcessEventsToIdle(); aCallback.m_nInvalidations = 0; // When typing to the document: pWrtShell->Insert("x"); pWrtShell->GetSfxViewShell()->flushPendingLOKInvalidateTiles(); // Then make sure that only the text frame at the cursor is invalidated: pDoc->getIDocumentStatistics().GetUpdatedDocStat(/*bCompleteAsync=*/true, /*bFields=*/false); // Without the accompanying fix in place, this test would have failed with: // - Expected: 1 // - Actual : 2 // i.e. the footer was also invalidated on each keypress. CPPUNIT_ASSERT_EQUAL(1, aCallback.m_nInvalidations); // Tear down LOK. pWrtShell->GetSfxViewShell()->setLibreOfficeKitViewCallback(nullptr); mxComponent->dispose(); mxComponent.clear(); comphelper::LibreOfficeKit::setActive(false); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testFlyAnchorUndo) { // Given a document with a fly frame, anchored after the last char of the document: load(DATA_DIRECTORY, "fly-anchor-undo.odt"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pShell = pTextDoc->GetDocShell(); SwDoc* pDoc = pShell->GetDoc(); const SwFrameFormats& rSpz = *pDoc->GetSpzFrameFormats(); sal_Int32 nExpected = rSpz[0]->GetAnchor().GetContentAnchor()->nContent.GetIndex(); // When deleting that last character and undoing it: SwWrtShell* pWrtShell = pShell->GetWrtShell(); pWrtShell->SttEndDoc(/*bStt=*/false); pWrtShell->DelLeft(); pWrtShell->Undo(); // Then make sure the anchor position after the undo is the same as the original: sal_Int32 nActual = rSpz[0]->GetAnchor().GetContentAnchor()->nContent.GetIndex(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 3 // - Actual : 2 // i.e. the anchor position was left unchanged by the undo. CPPUNIT_ASSERT_EQUAL(nExpected, nActual); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testSplitNodeSuperscriptCopy) { // Given a document with superscript text at the end of a paragraph: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->Insert("1st"); pWrtShell->Left(CRSR_SKIP_CHARS, /*bSelect=*/true, 2, /*bBasicCall=*/false); SfxItemSetFixed aSet(pWrtShell->GetAttrPool()); SvxEscapementItem aItem(SvxEscapement::Superscript, RES_CHRATR_ESCAPEMENT); aSet.Put(aItem); pWrtShell->SetAttrSet(aSet); // When hitting enter at the end of the paragraph: pWrtShell->SttEndDoc(/*bStt=*/false); pWrtShell->SplitNode(/*bAutoFormat=*/true); // Then make sure that the superscript formatting doesn't appear on the next paragraph: aSet.ClearItem(RES_CHRATR_ESCAPEMENT); pWrtShell->GetCurAttr(aSet); // Without the accompanying fix in place, this test would have failed, the unexpected // superscript appeared in the next paragraph. CPPUNIT_ASSERT(!aSet.HasItem(RES_CHRATR_ESCAPEMENT)); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testInsertDropDownContentControlTwice) { // Given an already selected dropdown content control: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->InsertContentControl(SwContentControlType::DROP_DOWN_LIST); // When trying to insert an inner one, make sure that we don't crash: pWrtShell->InsertContentControl(SwContentControlType::DROP_DOWN_LIST); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testCheckboxContentControlKeyboard) { // Given an already selected checkbox content control: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->InsertContentControl(SwContentControlType::CHECKBOX); SwEditWin& rEditWin = pWrtShell->GetView().GetEditWin(); // When pressing space on the keyboard: KeyEvent aKeyEvent(' ', KEY_SPACE); rEditWin.KeyInput(aKeyEvent); // Then make sure the state is toggled: SwTextNode* pTextNode = pWrtShell->GetCursor()->GetNode().GetTextNode(); SwTextAttr* pAttr = pTextNode->GetTextAttrForCharAt(0, RES_TXTATR_CONTENTCONTROL); auto pTextContentControl = static_txtattr_cast(pAttr); auto& rFormatContentControl = static_cast(pTextContentControl->GetAttr()); std::shared_ptr pContentControl = rFormatContentControl.GetContentControl(); // Without the accompanying fix in place, this test would have failed, because the state // remained unchanged. CPPUNIT_ASSERT(pContentControl->GetChecked()); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testDropdownContentControlKeyboard) { // Given an already selected dropdown content control: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->InsertContentControl(SwContentControlType::DROP_DOWN_LIST); // When checking if alt-down should open a popup: SwTextContentControl* pTextContentControl = pWrtShell->CursorInsideContentControl(); auto& rFormatContentControl = static_cast(pTextContentControl->GetAttr()); std::shared_ptr pContentControl = rFormatContentControl.GetContentControl(); vcl::KeyCode aKeyCode(KEY_DOWN, KEY_MOD2); bool bShouldOpen = pContentControl->ShouldOpenPopup(aKeyCode); // Then make sure that the answer is yes for dropdowns: // Without the accompanying fix in place, this test would have failed, the dropdown popup was // mouse-only. CPPUNIT_ASSERT(bShouldOpen); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testPictureContentControlKeyboard) { // Given an already selected picture content control: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->InsertContentControl(SwContentControlType::PICTURE); pWrtShell->GotoObj(/*bNext=*/true, GotoObjFlags::Any); // When checking if enter should trigger the file picker: const SwFrameFormat* pFlyFormat = pWrtShell->GetFlyFrameFormat(); const SwFormatAnchor& rFormatAnchor = pFlyFormat->GetAnchor(); const SwPosition* pAnchorPos = rFormatAnchor.GetContentAnchor(); SwTextNode* pTextNode = pAnchorPos->nNode.GetNode().GetTextNode(); SwTextAttr* pAttr = pTextNode->GetTextAttrAt(pAnchorPos->nContent.GetIndex(), RES_TXTATR_CONTENTCONTROL, SwTextNode::PARENT); auto pTextContentControl = static_txtattr_cast(pAttr); auto& rFormatContentControl = static_cast(pTextContentControl->GetAttr()); std::shared_ptr pContentControl = rFormatContentControl.GetContentControl(); bool bIsInteracting = pContentControl->IsInteractingCharacter('\r'); // Then make sure that the answer is yes for pictures: // Without the accompanying fix in place, this test would have failed, the picture replacement // file-picker was mouse-only. CPPUNIT_ASSERT(bIsInteracting); } CPPUNIT_TEST_FIXTURE(SwCoreTxtnodeTest, testDateContentControlKeyboard) { // Given an already selected date content control: SwDoc* pDoc = createSwDoc(); SwWrtShell* pWrtShell = pDoc->GetDocShell()->GetWrtShell(); pWrtShell->InsertContentControl(SwContentControlType::DATE); // When checking if alt-down should open a popup: SwTextContentControl* pTextContentControl = pWrtShell->CursorInsideContentControl(); auto& rFormatContentControl = static_cast(pTextContentControl->GetAttr()); std::shared_ptr pContentControl = rFormatContentControl.GetContentControl(); vcl::KeyCode aKeyCode(KEY_DOWN, KEY_MOD2); bool bShouldOpen = pContentControl->ShouldOpenPopup(aKeyCode); // Then make sure that the answer is yes for date: // Without the accompanying fix in place, this test would have failed, the date popup was // mouse-only. CPPUNIT_ASSERT(bShouldOpen); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */