/* -*- 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 constexpr OUStringLiteral DATA_DIRECTORY = u"/sw/qa/core/frmedt/data/"; /// Covers sw/source/core/frmedt/ fixes. class SwCoreFrmedtTest : public SwModelTestBase { }; CPPUNIT_TEST_FIXTURE(SwCoreFrmedtTest, testTextboxReanchor) { // Load a document with a textframe and a textbox(shape+textframe). load(DATA_DIRECTORY, "textbox-reanchor.odt"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDoc* pDoc = pTextDoc->GetDocShell()->GetDoc(); SdrPage* pDrawPage = pDoc->getIDocumentDrawModelAccess().GetDrawModel()->GetPage(0); SdrObject* pDrawShape = pDrawPage->GetObj(1); CPPUNIT_ASSERT_EQUAL(OUString("draw shape"), pDrawShape->GetName()); // Select the shape of the textbox. Point aPoint; SwWrtShell* pShell = pDoc->GetDocShell()->GetWrtShell(); pShell->SelectObj(aPoint, /*nFlag=*/0, pDrawShape); // Anchor the shape of the textbox into its own textframe. SdrObject* pTextFrameObj = pDrawPage->GetObj(2); SwFrameFormat* pTextFrameFormat = FindFrameFormat(pTextFrameObj); CPPUNIT_ASSERT_EQUAL(OUString("Frame2"), pTextFrameFormat->GetName()); SwFrameFormat* pDrawShapeFormat = FindFrameFormat(pDrawShape); SwNodeOffset nOldAnchor = pDrawShapeFormat->GetAnchor().GetContentAnchor()->nNode.GetIndex(); pShell->FindAnchorPos(pTextFrameObj->GetLastBoundRect().Center(), true); SwNodeOffset nNewAnchor = pDrawShapeFormat->GetAnchor().GetContentAnchor()->nNode.GetIndex(); // Without the accompanying fix in place, this test would have failed with: // - Expected: 6 // - Actual : 9 // i.e. SwFEShell allowed to anchor the textframe of a textbox into itself. CPPUNIT_ASSERT_EQUAL(nOldAnchor, nNewAnchor); } CPPUNIT_TEST_FIXTURE(SwCoreFrmedtTest, testVertPosFromBottomBoundingBox) { // Insert a shape and anchor it vertically in a way, so its position is from the top of the page // bottom margin area. mxComponent = loadFromDesktop("private:factory/swriter", "com.sun.star.text.TextDocument"); uno::Reference xFactory(mxComponent, uno::UNO_QUERY); uno::Reference xShape( xFactory->createInstance("com.sun.star.drawing.RectangleShape"), uno::UNO_QUERY); xShape->setSize(awt::Size(10000, 10000)); uno::Reference xShapeProps(xShape, uno::UNO_QUERY); xShapeProps->setPropertyValue("AnchorType", uno::Any(text::TextContentAnchorType_AT_CHARACTER)); xShapeProps->setPropertyValue("VertOrient", uno::Any(text::VertOrientation::NONE)); xShapeProps->setPropertyValue("VertOrientRelation", uno::Any(text::RelOrientation::PAGE_PRINT_AREA_BOTTOM)); xShapeProps->setPropertyValue("VertOrientPosition", uno::Any(static_cast(-11000))); uno::Reference xDrawPageSupplier(mxComponent, uno::UNO_QUERY); xDrawPageSupplier->getDrawPage()->add(xShape); // Get the absolute position of the top of the page bottom margin area. xmlDocUniquePtr pXmlDoc = parseLayoutDump(); SwTwips nPagePrintAreaBottom = getXPath(pXmlDoc, "//page/infos/prtBounds", "bottom").toInt32(); // Calculate the allowed bounding box of the shape, e.g. the shape's position & size dialog uses // this to limit the vertical position to sensible values. SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwWrtShell* pWrtShell = pTextDoc->GetDocShell()->GetWrtShell(); SwRect aBoundRect; RndStdIds eAnchorType = RndStdIds::FLY_AT_CHAR; SwDoc* pDoc = pTextDoc->GetDocShell()->GetDoc(); const auto& rFrameFormats = *pDoc->GetFrameFormats(); const SwPosition* pContentPos = rFrameFormats[0]->GetAnchor().GetContentAnchor(); sal_Int16 eHoriRelOrient = text::RelOrientation::PAGE_FRAME; sal_Int16 eVertRelOrient = text::RelOrientation::PAGE_PRINT_AREA_BOTTOM; bool bFollowTextFlow = false; bool bMirror = false; Size aPercentSize; pWrtShell->CalcBoundRect(aBoundRect, eAnchorType, eHoriRelOrient, eVertRelOrient, pContentPos, bFollowTextFlow, bMirror, nullptr, &aPercentSize); // Without the accompanying fix in place, this test would have failed with: // - Expected: -14705 // - Actual : -1134 // i.e. UI did not allow anchoring a shape 10cm above the bottom of the page due to wrong // bounding box. CPPUNIT_ASSERT_EQUAL(-1 * nPagePrintAreaBottom, aBoundRect.Pos().getY()); } CPPUNIT_TEST_FIXTURE(SwCoreFrmedtTest, testPasteFlyInTextBox) { // Given a document that contains a textbox, which contains an sw image (fly frame) load(DATA_DIRECTORY, "paste-fly-in-textbox.docx"); SwXTextDocument* pTextDoc = dynamic_cast(mxComponent.get()); SwDocShell* pDocShell = pTextDoc->GetDocShell(); SwWrtShell* pWrtShell = pDocShell->GetWrtShell(); SwDoc* pDoc = pDocShell->GetDoc(); SdrPage* pPage = pDoc->getIDocumentDrawModelAccess().GetDrawModel()->GetPage(0); SdrObject* pObject = pPage->GetObj(0); pWrtShell->SelectObj(Point(), 0, pObject); CPPUNIT_ASSERT_EQUAL(static_cast(3), pDoc->GetSpzFrameFormats()->GetFormatCount()); rtl::Reference pTransfer = new SwTransferable(*pWrtShell); pTransfer->Cut(); CPPUNIT_ASSERT_EQUAL(static_cast(0), pDoc->GetSpzFrameFormats()->GetFormatCount()); TransferableDataHelper aHelper(pTransfer); // When pasting that to an empty document. SwTransferable::Paste(*pWrtShell, aHelper); // Then we should have the image only once: 3 formats (draw+fly formats for the textbox and a // fly format for the image). // Without the accompanying fix in place, this test would have failed with: // - Expected: 3 // - Actual : 4 // i.e. the image was pasted twice. CPPUNIT_ASSERT_EQUAL(static_cast(3), pDoc->GetSpzFrameFormats()->GetFormatCount()); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */