1
0
Fork 0
libreoffice/sw/qa/core/layout/fly.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

174 lines
7.1 KiB
C++

/* -*- 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 <swmodeltestbase.hxx>
#include <IDocumentLayoutAccess.hxx>
#include <anchoredobject.hxx>
#include <flyfrms.hxx>
#include <pagefrm.hxx>
#include <rootfrm.hxx>
#include <sortedobjs.hxx>
#include <docsh.hxx>
#include <wrtsh.hxx>
#include <bodyfrm.hxx>
#include <txtfrm.hxx>
namespace
{
/// Covers sw/source/core/layout/fly.cxx fixes, i.e. mostly SwFlyFrame.
class Test : public SwModelTestBase
{
public:
Test()
: SwModelTestBase(u"/sw/qa/core/layout/data/"_ustr)
{
}
};
CPPUNIT_TEST_FIXTURE(Test, testSplitFlyNegativeHeight)
{
// Given a document with complex enough content that a split fly frame temporarily moves below
// the bottom of the body frame on a page:
// When laying out the document, SwEditShell::CalcLayout() never returned:
createSwDoc("floattable-negative-height.docx");
SwWrtShell* pWrtShell = getSwDocShell()->GetWrtShell();
pWrtShell->Reformat();
// Make sure that all the pages have the expected content:
SwDoc* pDoc = getSwDoc();
SwRootFrame* pLayout = pDoc->getIDocumentLayoutAccess().GetCurrentLayout();
for (SwFrame* pFrame = pLayout->Lower(); pFrame; pFrame = pFrame->GetNext())
{
auto pPage = pFrame->DynCastPageFrame();
if (!pPage->GetPrev())
{
// First page: start of the split fly chain:
CPPUNIT_ASSERT(pPage->GetSortedObjs());
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), rPageObjs.size());
auto pFly = rPageObjs[0]->DynCastFlyFrame()->DynCastFlyAtContentFrame();
CPPUNIT_ASSERT(pFly);
CPPUNIT_ASSERT(!pFly->GetPrecede());
CPPUNIT_ASSERT(pFly->GetFollow());
}
else if (pPage->GetPrev() && pPage->GetNext() && pPage->GetNext()->GetNext())
{
// Middle pages: have a previous and a next fly:
CPPUNIT_ASSERT(pPage->GetSortedObjs());
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), rPageObjs.size());
auto pFly = rPageObjs[0]->DynCastFlyFrame()->DynCastFlyAtContentFrame();
CPPUNIT_ASSERT(pFly);
CPPUNIT_ASSERT(pFly->GetPrecede());
CPPUNIT_ASSERT(pFly->GetFollow());
}
else if (pPage->GetPrev() && pPage->GetNext() && !pPage->GetNext()->GetNext())
{
// Page last but one: end of the fly chain:
CPPUNIT_ASSERT(pPage->GetSortedObjs());
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), rPageObjs.size());
auto pFly = rPageObjs[0]->DynCastFlyFrame()->DynCastFlyAtContentFrame();
CPPUNIT_ASSERT(pFly);
CPPUNIT_ASSERT(pFly->GetPrecede());
CPPUNIT_ASSERT(!pFly->GetFollow());
}
else if (pPage->GetPrev() && !pPage->GetNext())
{
// Last page: no flys.
CPPUNIT_ASSERT(!pPage->GetSortedObjs());
}
}
}
CPPUNIT_TEST_FIXTURE(Test, testFlyRelWithRounding)
{
// Given a document where page width is 21.001cm (11906 twips), and the image width is 48% of
// the page width:
createSwDoc("fly-rel-width-rounding.odt");
// When laying out that document:
SwDoc* pDoc = getSwDoc();
SwRootFrame* pLayout = pDoc->getIDocumentLayoutAccess().GetCurrentLayout();
// Then make sure that we calculate the width of the fly correctly:
auto pPage = pLayout->GetLower()->DynCastPageFrame();
CPPUNIT_ASSERT(pPage->GetSortedObjs());
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), rPageObjs.size());
auto pFly = rPageObjs[0]->DynCastFlyFrame()->DynCastFlyAtContentFrame();
CPPUNIT_ASSERT(pFly);
tools::Long nFlyWidth = pFly->getFrameArea().Width();
// Without the accompanying fix in place, this test would have failed with:
// - Expected: 5715
// - Actual : 5714
// i.e. 5714.88 was truncated, not rounded.
CPPUNIT_ASSERT_EQUAL(static_cast<tools::Long>(5715), nFlyWidth);
}
CPPUNIT_TEST_FIXTURE(Test, testShapeLeftPaddingOffPage)
{
// Given a document with a shape that is 1cm off the page:
createSwDoc("shape-left-padding-off-page.docx");
// When laying out that document:
SwDoc* pDoc = getSwDoc();
SwRootFrame* pLayout = pDoc->getIDocumentLayoutAccess().GetCurrentLayout();
// Then make sure that the 2.5cm page margin + 2.5cm para margin and the 5cm shape left padding
// line up (with 1px tolerance):
auto pPage = pLayout->GetLower()->DynCastPageFrame();
auto pBody = static_cast<SwBodyFrame*>(pPage->GetLower());
auto pTextFrame = pBody->GetLower()->DynCastTextFrame();
SwTwips nBodyLeft = pTextFrame->getFrameArea().Left() + pTextFrame->getFramePrintArea().Left();
CPPUNIT_ASSERT(pPage->GetSortedObjs());
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), rPageObjs.size());
auto pFly = rPageObjs[1]->DynCastFlyFrame()->DynCastFlyAtContentFrame();
CPPUNIT_ASSERT(pFly);
SwTwips nFlyLeft = pFly->getFrameArea().Left() + pFly->getFramePrintArea().Left();
// Without the accompanying fix in place, this test would have failed with:
// - Expected greater or equal than: 3119
// - Actual : 2574
// i.e. the shape text had ~4cm left padding (visually) instead of 5cm.
CPPUNIT_ASSERT_GREATEREQUAL(nBodyLeft - MINFLY, nFlyLeft);
}
CPPUNIT_TEST_FIXTURE(Test, testShapeLeftPaddingWrapThroughOffPage)
{
// Given a document with a shape that is off the page (partially clipped on the left):
createSwDoc("shape-left-padding-wrap-through-off-page.docx");
// When laying out that document:
SwDoc* pDoc = getSwDoc();
SwRootFrame* pLayout = pDoc->getIDocumentLayoutAccess().GetCurrentLayout();
// Then make sure that the text is also clipped on the left, left margin is only the
// lIns="91440" (in EMUs) from the file:
auto pPage = pLayout->GetLower()->DynCastPageFrame();
SwSortedObjs& rPageObjs = *pPage->GetSortedObjs();
CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), rPageObjs.size());
auto it = std::find_if(rPageObjs.begin(), rPageObjs.end(), [](SwAnchoredObject* pObj) -> bool {
return pObj->DynCastFlyFrame() != nullptr;
});
CPPUNIT_ASSERT(it != rPageObjs.end());
auto pFly = (*it)->DynCastFlyFrame();
CPPUNIT_ASSERT(pFly);
SwTwips nFlyLeftMargin = pFly->getFramePrintArea().Left();
// Without the accompanying fix in place, this test would have failed with:
// - Expected: 144
// - Actual : 3031
// i.e. there was a large left margin, the text was clipped on the right.
CPPUNIT_ASSERT_EQUAL(static_cast<SwTwips>(144), nFlyLeftMargin);
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */