diff options
Diffstat (limited to 'sw/qa/core/uwriter.cxx')
-rw-r--r-- | sw/qa/core/uwriter.cxx | 1933 |
1 files changed, 1933 insertions, 0 deletions
diff --git a/sw/qa/core/uwriter.cxx b/sw/qa/core/uwriter.cxx new file mode 100644 index 000000000..ae59144b9 --- /dev/null +++ b/sw/qa/core/uwriter.cxx @@ -0,0 +1,1933 @@ +/* -*- 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 <sal/config.h> +#include <test/bootstrapfixture.hxx> + +#include <com/sun/star/i18n/WordType.hpp> + +#include <comphelper/processfactory.hxx> +#include <i18nutil/transliteration.hxx> +#include <editeng/adjustitem.hxx> +#include <editeng/boxitem.hxx> +#include <editeng/brushitem.hxx> +#include <editeng/colritem.hxx> +#include <editeng/contouritem.hxx> +#include <editeng/crossedoutitem.hxx> +#include <editeng/fhgtitem.hxx> +#include <editeng/fontitem.hxx> +#include <editeng/justifyitem.hxx> +#include <editeng/lineitem.hxx> +#include <editeng/postitem.hxx> +#include <editeng/shdditem.hxx> +#include <editeng/udlnitem.hxx> +#include <editeng/wghtitem.hxx> +#include <svl/intitem.hxx> +#include <svx/algitem.hxx> +#include <svx/rotmodit.hxx> +#include <tools/urlobj.hxx> +#include <unotools/tempfile.hxx> +#include <unotools/transliterationwrapper.hxx> + +#include <editeng/langitem.hxx> +#include <editeng/charhiddenitem.hxx> + +#include <sfx2/docfilt.hxx> +#include <sfx2/docfile.hxx> + +#include <xmloff/odffields.hxx> + +#include <breakit.hxx> +#include <doc.hxx> +#include <IDocumentRedlineAccess.hxx> +#include <IDocumentFieldsAccess.hxx> +#include <IDocumentStatistics.hxx> +#include <cellfml.hxx> +#include <docsh.hxx> +#include <docstat.hxx> +#include <docufld.hxx> +#include <fmtanchr.hxx> +#include <ndtxt.hxx> +#include <shellres.hxx> +#include <swscanner.hxx> +#include <swdll.hxx> +#include <swtypes.hxx> +#include <fmtftn.hxx> +#include <fmtrfmrk.hxx> +#include <fmtinfmt.hxx> +#include <fchrfmt.hxx> +#include <fmtfld.hxx> +#include <redline.hxx> +#include <docary.hxx> +#include <modeltoviewhelper.hxx> +#include <IMark.hxx> +#include <ring.hxx> +#include <calbck.hxx> +#include <pagedesc.hxx> +#include <calc.hxx> + +#include <tblafmt.hxx> +#include <unotbl.hxx> +#include <IDocumentMarkAccess.hxx> +#include <itabenum.hxx> + +typedef tools::SvRef<SwDocShell> SwDocShellRef; + +using namespace ::com::sun::star; + +/* Implementation of Swdoc-Test class */ + +class SwDocTest : public test::BootstrapFixture +{ +public: + SwDocTest() + : m_pDoc(nullptr) + { + } + + virtual void setUp() override; + virtual void tearDown() override; + + void testTableAutoFormats(); + void testPageDescName(); + void testFileNameFields(); + void testDocStat(); + void testModelToViewHelperPassthrough(); + void testModelToViewHelperExpandFieldsExpandFootnote(); + void testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode(); + void testModelToViewHelperExpandFields(); + void testModelToViewHelperExpandFieldsReplaceMode(); + void testModelToViewHelperExpandFieldsHideInvisible(); + void testModelToViewHelperExpandFieldsHideRedlined(); + void testModelToViewHelperExpandFieldsHideInvisibleExpandFootnote(); + void testModelToViewHelperExpandFieldsHideInvisibleExpandFootnoteReplaceMode(); + void testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnote(); + void testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnoteReplaceMode(); + void testModelToViewHelperHideInvisibleHideRedlined(); + void testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnote(); + void testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnoteReplaceMode(); + void testModelToViewHelperExpandFieldsExpandFootnote2(); + void testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode2(); + void testSwScanner(); + void testUserPerceivedCharCount(); + void testMergePortionsDeleteNotSorted(); + void testGraphicAnchorDeletion(); + void testTransliterate(); + void testMarkMove(); + void testFormulas(); + void testIntrusiveRing(); + void testClientModify(); + void testBroadcastingModify(); + void testWriterMultiListener(); + void test64kPageDescs(); + void testTdf92308(); + void testTableCellComparison(); + + CPPUNIT_TEST_SUITE(SwDocTest); + + CPPUNIT_TEST(testTransliterate); + CPPUNIT_TEST(testTableAutoFormats); + CPPUNIT_TEST(testPageDescName); + CPPUNIT_TEST(testFileNameFields); + CPPUNIT_TEST(testDocStat); + CPPUNIT_TEST(testModelToViewHelperPassthrough); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsExpandFootnote); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode); + CPPUNIT_TEST(testModelToViewHelperExpandFields); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsReplaceMode); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideInvisible); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideRedlined); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideInvisibleExpandFootnote); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideInvisibleExpandFootnoteReplaceMode); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnote); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnoteReplaceMode); + CPPUNIT_TEST(testModelToViewHelperHideInvisibleHideRedlined); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnote); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnoteReplaceMode); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsExpandFootnote2); + CPPUNIT_TEST(testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode2); + CPPUNIT_TEST(testSwScanner); + CPPUNIT_TEST(testUserPerceivedCharCount); + CPPUNIT_TEST(testMergePortionsDeleteNotSorted); + CPPUNIT_TEST(testGraphicAnchorDeletion); + CPPUNIT_TEST(testMarkMove); + CPPUNIT_TEST(testFormulas); + CPPUNIT_TEST(testIntrusiveRing); + CPPUNIT_TEST(testClientModify); + CPPUNIT_TEST(testBroadcastingModify); + CPPUNIT_TEST(testWriterMultiListener); + CPPUNIT_TEST(test64kPageDescs); + CPPUNIT_TEST(testTdf92308); + CPPUNIT_TEST(testTableCellComparison); + CPPUNIT_TEST_SUITE_END(); + +private: + SwDoc *m_pDoc; + SwDocShellRef m_xDocShRef; +}; + +void SwDocTest::testPageDescName() +{ + ShellResource aShellResources; + + std::vector<OUString> aResults; + + //These names must be unique for each different combination, otherwise + //duplicate page description names may exist, which will causes lookup + //by name to be incorrect, and so the corresponding export to .odt + aResults.push_back(aShellResources.GetPageDescName(1, ShellResource::NORMAL_PAGE)); + aResults.push_back(aShellResources.GetPageDescName(1, ShellResource::FIRST_PAGE)); + aResults.push_back(aShellResources.GetPageDescName(1, ShellResource::FOLLOW_PAGE)); + + std::sort(aResults.begin(), aResults.end()); + aResults.erase(std::unique(aResults.begin(), aResults.end()), aResults.end()); + + CPPUNIT_ASSERT_EQUAL_MESSAGE("GetPageDescName results must be unique", static_cast<size_t>(3), aResults.size()); +} + +//See https://bugs.libreoffice.org/show_bug.cgi?id=32463 +void SwDocTest::testFileNameFields() +{ + //Here's a file name with some chars in it that will be %% encoded, when expanding + //SwFileNameFields we want to restore the original readable filename + utl::TempFile aTempFile(u"demo [name]"); + aTempFile.EnableKillingFile(); + + INetURLObject aTempFileURL(aTempFile.GetURL()); + OUString sFileURL = aTempFileURL.GetMainURL(INetURLObject::DecodeMechanism::NONE); + SfxMedium aDstMed(sFileURL, StreamMode::STD_READWRITE); + + auto pFilter = std::make_shared<SfxFilter>( + "Text", + OUString(), SfxFilterFlags::NONE, SotClipboardFormatId::NONE, OUString(), OUString(), + "TEXT", OUString() ); + aDstMed.SetFilter(pFilter); + + m_xDocShRef->DoSaveAs(aDstMed); + m_xDocShRef->DoSaveCompleted(&aDstMed); + + const INetURLObject &rUrlObj = m_xDocShRef->GetMedium()->GetURLObject(); + + SwFileNameFieldType aNameField(*m_pDoc); + + { + OUString sResult(aNameField.Expand(FF_NAME)); + OUString sExpected(rUrlObj.getName(INetURLObject::LAST_SEGMENT, + true,INetURLObject::DecodeMechanism::WithCharset)); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected Readable FileName", sExpected, sResult); + } + + { + OUString sResult(aNameField.Expand(FF_PATHNAME)); + OUString sExpected(rUrlObj.GetFull()); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected Readable FileName", sExpected, sResult); + } + + { + OUString sResult(aNameField.Expand(FF_PATH)); + INetURLObject aTemp(rUrlObj); + aTemp.removeSegment(); + OUString sExpected(aTemp.PathToFileName()); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected Readable FileName", sExpected, sResult); + } + + { + OUString sResult(aNameField.Expand(FF_NAME_NOEXT)); + OUString sExpected(rUrlObj.getName(INetURLObject::LAST_SEGMENT, + true,INetURLObject::DecodeMechanism::WithCharset)); + //Chop off .tmp + sExpected = sExpected.copy(0, sExpected.getLength() - 4); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected Readable FileName", sExpected, sResult); + } + + m_xDocShRef->DoInitNew(); +} + +//See http://lists.freedesktop.org/archives/libreoffice/2011-August/016666.html +//Remove unnecessary parameter to IDocumentStatistics::UpdateDocStat for +//motivation +void SwDocTest::testDocStat() +{ + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected initial 0 count", static_cast<sal_uLong>(0), m_pDoc->getIDocumentStatistics().GetDocStat().nChar); + + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + + OUString sText("Hello World"); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sText); + + CPPUNIT_ASSERT_EQUAL_MESSAGE("Should still be non-updated 0 count", static_cast<sal_uLong>(0), m_pDoc->getIDocumentStatistics().GetDocStat().nChar); + + SwDocStat aDocStat = m_pDoc->getIDocumentStatistics().GetUpdatedDocStat( false, true ); + sal_uLong nLen = static_cast<sal_uLong>(sText.getLength()); + + CPPUNIT_ASSERT_EQUAL_MESSAGE("Should now have updated count", nLen, aDocStat.nChar); + + CPPUNIT_ASSERT_EQUAL_MESSAGE("And cache is updated too", nLen, m_pDoc->getIDocumentStatistics().GetDocStat().nChar); +} + +//For UI character counts we should follow UAX#29 and display the user +//perceived characters, not the number of codepoints, nor the number of code +//units http://unicode.org/reports/tr29/ +void SwDocTest::testUserPerceivedCharCount() +{ + SwBreakIt *pBreakIter = SwBreakIt::Get(); + + //Grapheme example, two different unicode code-points perceived by the user as a single + //glyph + static constexpr OUStringLiteral sALEF_QAMATS = u"\u05D0\u05B8"; + sal_Int32 nGraphemeCount = pBreakIter->getGraphemeCount(sALEF_QAMATS); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Grapheme Count should be 1", static_cast<sal_Int32>(1), nGraphemeCount); + + //Surrogate pair example, one single unicode code-point (U+1D11E) + //represented as two code units in UTF-16 + static constexpr OUStringLiteral sGCLEF = u"\U0001D11E"; + sal_Int32 nCount = pBreakIter->getGraphemeCount(sGCLEF); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Surrogate Pair should be counted as single character", static_cast<sal_Int32>(1), nCount); +} + +static SwTextNode* getModelToViewTestDocument(SwDoc *pDoc) +{ + SwNodeIndex aIdx(pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + + SwFormatFootnote aFootnote; + aFootnote.SetNumStr("foo"); + + pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + pDoc->getIDocumentContentOperations().InsertString(aPaM, "AAAAA BBBBB "); + SwTextNode* pTextNode = aPaM.GetNode().GetTextNode(); + sal_Int32 nPos = aPaM.GetPoint()->nContent.GetIndex(); + pTextNode->InsertItem(aFootnote, nPos, nPos); + pDoc->getIDocumentContentOperations().InsertString(aPaM, " CCCCC "); + nPos = aPaM.GetPoint()->nContent.GetIndex(); + pTextNode->InsertItem(aFootnote, nPos, nPos); + pDoc->getIDocumentContentOperations().InsertString(aPaM, " DDDDD"); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>((4*5) + 5 + 2), pTextNode->GetText().getLength()); + + //set start of selection to first B + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 6); + aPaM.SetMark(); + //set end of selection to last C + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 14); + //set character attribute hidden on range + SvxCharHiddenItem aHidden(true, RES_CHRATR_HIDDEN); + pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, aHidden ); + aPaM.DeleteMark(); + + //turn on red-lining and show changes + pDoc->getIDocumentRedlineAccess().SetRedlineFlags(RedlineFlags::On | RedlineFlags::ShowDelete|RedlineFlags::ShowInsert); + CPPUNIT_ASSERT_MESSAGE("redlining should be on", pDoc->getIDocumentRedlineAccess().IsRedlineOn()); + CPPUNIT_ASSERT_MESSAGE("redlines should be visible", IDocumentRedlineAccess::IsShowChanges(pDoc->getIDocumentRedlineAccess().GetRedlineFlags())); + + //set start of selection to last A + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 4); + aPaM.SetMark(); + //set end of selection to second last B + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 9); + pDoc->getIDocumentContentOperations().DeleteAndJoin(aPaM); //redline-aware deletion api + aPaM.DeleteMark(); + + return pTextNode; +} + +static SwTextNode* getModelToViewTestDocument2(SwDoc *pDoc) +{ + getModelToViewTestDocument(pDoc); + + SwNodeIndex aIdx(pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + + pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + pDoc->getIDocumentContentOperations().InsertString(aPaM, "AAAAA"); + IDocumentMarkAccess* pMarksAccess = pDoc->getIDocumentMarkAccess(); + sw::mark::IFieldmark *pFieldmark = + pMarksAccess->makeNoTextFieldBookmark(aPaM, "test", ODF_FORMDROPDOWN); + CPPUNIT_ASSERT(pFieldmark); + uno::Sequence< OUString > vListEntries { "BBBBB" }; + (*pFieldmark->GetParameters())[ODF_FORMDROPDOWN_LISTENTRY] <<= vListEntries; + (*pFieldmark->GetParameters())[ODF_FORMDROPDOWN_RESULT] <<= sal_Int32(0); + pDoc->getIDocumentContentOperations().InsertString(aPaM, "CCCCC"); + SwTextNode* pTextNode = aPaM.GetNode().GetTextNode(); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(11), + pTextNode->GetText().getLength()); + + return pTextNode; +} + +void SwDocTest::testModelToViewHelperPassthrough() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, ExpandMode::PassThrough); + OUString sViewText = aModelToViewHelper.getViewText(); + OUString sModelText = pTextNode->GetText(); + CPPUNIT_ASSERT_EQUAL(sModelText, sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsExpandFootnote() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::ExpandFootnote); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA BBBBB foo CCCCC foo DDDDD"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::ExpandFootnote | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA BBBBB " + OUStringChar(CHAR_ZWSP) + " CCCCC " + OUStringChar(CHAR_ZWSP) + " DDDDD"), + sViewText); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(12), + aModelToViewHelper.getFootnotePositions()[0]); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(20), + aModelToViewHelper.getFootnotePositions()[1]); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFieldPositions().size()); +} + +void SwDocTest::testModelToViewHelperExpandFields() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, ExpandMode::ExpandFields); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA BBBBB CCCCC DDDDD"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsReplaceMode() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL(OUString("AAAAA BBBBB CCCCC DDDDD"), + sViewText); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFieldPositions().size()); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideInvisible() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, ExpandMode::HideInvisible); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA CCCCC " + OUStringChar(CH_TXTATR_BREAKWORD) + " DDDDD"), + sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideRedlined() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, ExpandMode::HideDeletions); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAABB " + OUStringChar(CH_TXTATR_BREAKWORD) + " CCCCC " + OUStringChar(CH_TXTATR_BREAKWORD) + " DDDDD"), + sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideInvisibleExpandFootnote() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideInvisible | ExpandMode::ExpandFootnote); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL(OUString("AAAAA CCCCC foo DDDDD"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideInvisibleExpandFootnoteReplaceMode() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideInvisible | ExpandMode::ExpandFootnote | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA CCCCC " + OUStringChar(CHAR_ZWSP) + " DDDDD"), + sViewText); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(12), + aModelToViewHelper.getFootnotePositions()[0]); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFieldPositions().size()); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnote() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideDeletions | ExpandMode::ExpandFootnote); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAABB foo CCCCC foo DDDDD"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideHideRedlinedExpandFootnoteReplaceMode() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideDeletions | ExpandMode::ExpandFootnote | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAABB " + OUStringChar(CHAR_ZWSP) + " CCCCC " + OUStringChar(CHAR_ZWSP) + " DDDDD"), + sViewText); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(7), + aModelToViewHelper.getFootnotePositions()[0]); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(15), + aModelToViewHelper.getFootnotePositions()[1]); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFieldPositions().size()); +} + +void SwDocTest::testModelToViewHelperHideInvisibleHideRedlined() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::HideInvisible | ExpandMode::HideDeletions); + OUString sViewText = aModelToViewHelper.getViewText(); + OUString aBuffer = "AAAACCCCC " + + OUStringChar(CH_TXTATR_BREAKWORD) + + " DDDDD"; + CPPUNIT_ASSERT_EQUAL(aBuffer, sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnote() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideInvisible | ExpandMode::HideDeletions | ExpandMode::ExpandFootnote); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL(OUString("AAAACCCCC foo DDDDD"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsHideInvisibleHideRedlinedExpandFootnoteReplaceMode() +{ + SwTextNode* pTextNode = getModelToViewTestDocument(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::HideInvisible | ExpandMode::HideDeletions | ExpandMode::ExpandFootnote | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL(sViewText, + OUString("AAAACCCCC " + OUStringChar(CHAR_ZWSP) + " DDDDD")); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(10), + aModelToViewHelper.getFootnotePositions()[0]); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFieldPositions().size()); +} + +void SwDocTest::testModelToViewHelperExpandFieldsExpandFootnote2() +{ + SwTextNode* pTextNode = getModelToViewTestDocument2(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::ExpandFootnote); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL(OUString("AAAAABBBBBCCCCC"), sViewText); +} + +void SwDocTest::testModelToViewHelperExpandFieldsExpandFootnoteReplaceMode2() +{ + SwTextNode* pTextNode = getModelToViewTestDocument2(m_pDoc); + + ModelToViewHelper aModelToViewHelper(*pTextNode, nullptr, + ExpandMode::ExpandFields | ExpandMode::ExpandFootnote | ExpandMode::ReplaceMode); + OUString sViewText = aModelToViewHelper.getViewText(); + CPPUNIT_ASSERT_EQUAL( + OUString("AAAAA" + OUStringChar(CHAR_ZWSP) + "CCCCC"), + sViewText); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(0), + aModelToViewHelper.getFootnotePositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), + aModelToViewHelper.getFieldPositions().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(5), + aModelToViewHelper.getFieldPositions()[0]); +} + +void SwDocTest::testSwScanner() +{ + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + + SwTextNode* pTextNode = aPaM.GetNode().GetTextNode(); + + CPPUNIT_ASSERT_MESSAGE("Has Text Node", pTextNode); + + //See https://bugs.libreoffice.org/show_bug.cgi?id=40449 + //See https://bugs.libreoffice.org/show_bug.cgi?id=39365 + //Use a temporary OUString as the arg, as that's the trouble behind + //fdo#40449 and fdo#39365 + { + SwScanner aScanner(*pTextNode, + "Hello World", + nullptr, ModelToViewHelper(), i18n::WordType::DICTIONARY_WORD, 0, + RTL_CONSTASCII_LENGTH("Hello World")); + + bool bFirstOk = aScanner.NextWord(); + CPPUNIT_ASSERT_MESSAGE("First Token", bFirstOk); + const OUString &rHello = aScanner.GetWord(); + CPPUNIT_ASSERT_EQUAL(OUString("Hello"), rHello); + + bool bSecondOk = aScanner.NextWord(); + CPPUNIT_ASSERT_MESSAGE("Second Token", bSecondOk); + const OUString &rWorld = aScanner.GetWord(); + CPPUNIT_ASSERT_EQUAL(OUString("World"), rWorld); + } + + //See https://www.libreoffice.org/bugzilla/show_bug.cgi?id=45271 + { + static constexpr OUStringLiteral IDEOGRAPHICFULLSTOP_D = u"\u3002D"; + + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, IDEOGRAPHICFULLSTOP_D); + + SvxLanguageItem aCJKLangItem( LANGUAGE_CHINESE_SIMPLIFIED, RES_CHRATR_CJK_LANGUAGE ); + SvxLanguageItem aWestLangItem( LANGUAGE_ENGLISH_US, RES_CHRATR_LANGUAGE ); + m_pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, aCJKLangItem ); + m_pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, aWestLangItem ); + + SwDocStat aDocStat; + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, IDEOGRAPHICFULLSTOP_D.getLength()); + + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(2), aDocStat.nChar); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(2), aDocStat.nCharExcludingSpaces); + } + { + static constexpr OUStringLiteral test = + u"\u3053\u306E\u65E5\u672C\u8A9E\u306F\u6B63\u3057" + "\u304F\u6570\u3048\u3089\u308C\u308B\u3067\u3057" + "\u3087\u3046\u304B\u3002And " + "let's th" + "row some" + " English" + " in to m" + "ake it i" + "nteresti" + "ng. \u305D\u3057\u3066" + "\u3001\u307E\u305F\u65E5\u672C\u8A9E\u3000\u3000" + "\u3067\u3082\u4ECA\u56DE\u306F\u7A7A\u767D\u3092" + "\u3000\u3000\u5165\u308C\u307E\u3057\u305F\u3002" + " So how" + " does th" + "is do? "; + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, test); + + SvxLanguageItem aCJKLangItem( LANGUAGE_JAPANESE, RES_CHRATR_CJK_LANGUAGE ); + SvxLanguageItem aWestLangItem( LANGUAGE_ENGLISH_US, RES_CHRATR_LANGUAGE ); + m_pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, aCJKLangItem ); + m_pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, aWestLangItem ); + + SwDocStat aDocStat; + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, test.getLength()); + CPPUNIT_ASSERT_EQUAL_MESSAGE("words", static_cast<sal_uLong>(58), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL_MESSAGE("Asian characters and Korean syllables", static_cast<sal_uLong>(43), aDocStat.nAsianWord); + CPPUNIT_ASSERT_EQUAL_MESSAGE("non-whitespace chars", static_cast<sal_uLong>(105), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL_MESSAGE("characters", static_cast<sal_uLong>(128), aDocStat.nChar); + } + + //See https://bz.apache.org/ooo/show_bug.cgi?id=89042 + //See https://bugs.libreoffice.org/show_bug.cgi?id=53399 + { + SwDocStat aDocStat; + + static constexpr OUStringLiteral aShouldBeThree = + u"Should " + "\u2018be thr" + "ee\u2019"; + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, aShouldBeThree); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, aShouldBeThree.getLength()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(3), aDocStat.nWord); + + static constexpr OUStringLiteral aShouldBeFive = + u"french " + // << nbsp + "\u00AB\u00A0savoi" + // nnbsp + "r\u202fcalcu" + // idspace >> + "ler\u3000\u00BB"; + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, aShouldBeFive); + pTextNode = aPaM.GetNode().GetTextNode(); + aDocStat.Reset(); + pTextNode->CountWords(aDocStat, 0, aShouldBeFive.getLength()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(5), aDocStat.nWord); + } + + //See https://bugs.libreoffice.org/show_bug.cgi?id=49629 + { + SwDocStat aDocStat; + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Apple"); + pTextNode = aPaM.GetNode().GetTextNode(); + sal_Int32 nPos = aPaM.GetPoint()->nContent.GetIndex(); + SwFormatFootnote aFootnote; + aFootnote.SetNumStr("banana"); + SwTextAttr* pTA = pTextNode->InsertItem(aFootnote, nPos, nPos); + CPPUNIT_ASSERT(pTA); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(6), pTextNode->Len()); //Apple + 0x02 + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(1), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL_MESSAGE("footnote should be expanded", static_cast<sal_uLong>(11), aDocStat.nChar); + + const sal_Int32 nNextPos = aPaM.GetPoint()->nContent.GetIndex(); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(nPos+1), nNextPos); + SwFormatRefMark aRef("refmark"); + pTA = pTextNode->InsertItem(aRef, nNextPos, nNextPos); + CPPUNIT_ASSERT(pTA); + + aDocStat.Reset(); + pTextNode->SetWordCountDirty(true); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(1), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL_MESSAGE("refmark anchor should not be counted", static_cast<sal_uLong>(11), aDocStat.nChar); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Apple"); + + DateTime aDate(DateTime::SYSTEM); + SwPostItField aPostIt( + static_cast<SwPostItFieldType*>(m_pDoc->getIDocumentFieldsAccess().GetSysFieldType(SwFieldIds::Postit)), "An Author", + "Some Text", "Initials", "Name", aDate ); + m_pDoc->getIDocumentContentOperations().InsertPoolItem(aPaM, SwFormatField(aPostIt)); + + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Apple"); + pTextNode = aPaM.GetNode().GetTextNode(); + aDocStat.Reset(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(1), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL_MESSAGE("postit anchor should effectively not exist", static_cast<sal_uLong>(10), aDocStat.nChar); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(11), pTextNode->Len()); + + aDocStat.Reset(); + } + + //See https://bugs.libreoffice.org/show_bug.cgi?id=46757 + { + SwDocStat aDocStat; + + static constexpr OUStringLiteral aString = u"Lorem ipsum"; + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, aString); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(2), aDocStat.nWord); + + //turn on red-lining and show changes + m_pDoc->getIDocumentRedlineAccess().SetRedlineFlags(RedlineFlags::On | RedlineFlags::ShowDelete|RedlineFlags::ShowInsert); + CPPUNIT_ASSERT_MESSAGE("redlining should be on", m_pDoc->getIDocumentRedlineAccess().IsRedlineOn()); + CPPUNIT_ASSERT_MESSAGE("redlines should be visible", IDocumentRedlineAccess::IsShowChanges(m_pDoc->getIDocumentRedlineAccess().GetRedlineFlags())); + + //delete everything except the first word + aPaM.SetMark(); //set start of selection to current pos + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 5); //set end of selection to fifth char of current node + m_pDoc->getIDocumentContentOperations().DeleteAndJoin(aPaM); //redline-aware deletion api + //"real underlying text should be the same" + CPPUNIT_ASSERT_EQUAL(pTextNode->GetText(), OUString(aString)); + + aDocStat.Reset(); + pTextNode->SetWordCountDirty(true); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); //but word-counting the text should only count the non-deleted text + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(1), aDocStat.nWord); + + pTextNode->SetWordCountDirty(true); + + //keep red-lining on but hide changes + m_pDoc->getIDocumentRedlineAccess().SetRedlineFlags(RedlineFlags::On); + CPPUNIT_ASSERT_MESSAGE("redlining should be still on", m_pDoc->getIDocumentRedlineAccess().IsRedlineOn()); + CPPUNIT_ASSERT_MESSAGE("redlines should be invisible", !IDocumentRedlineAccess::IsShowChanges(m_pDoc->getIDocumentRedlineAccess().GetRedlineFlags())); + + aDocStat.Reset(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); //but word-counting the text should only count the non-deleted text + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(1), aDocStat.nWord); + + OUString sLorem = pTextNode->GetText(); + CPPUNIT_ASSERT_EQUAL(OUString("Lorem"), sLorem); + + const SwRedlineTable& rTable = m_pDoc->getIDocumentRedlineAccess().GetRedlineTable(); + + SwNodes& rNds = m_pDoc->GetNodes(); + CPPUNIT_ASSERT_EQUAL(static_cast<SwRedlineTable::size_type>(1), rTable.size()); + + SwNodeIndex* pNodeIdx = rTable[0]->GetContentIdx(); + CPPUNIT_ASSERT(pNodeIdx); + + pTextNode = rNds[ pNodeIdx->GetIndex() + 1 ]->GetTextNode(); //first deleted txtnode + CPPUNIT_ASSERT(pTextNode); + + OUString sIpsum = pTextNode->GetText(); + CPPUNIT_ASSERT_EQUAL(OUString(" ipsum"), sIpsum); + + aDocStat.Reset(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); //word-counting the text should only count the non-deleted text, and this whole chunk should be ignored + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(0), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(0), aDocStat.nChar); + + // https://bugs.libreoffice.org/show_bug.cgi?id=68347 we do want to count + // redline *added* text though + m_pDoc->getIDocumentRedlineAccess().SetRedlineFlags(RedlineFlags::On | RedlineFlags::ShowDelete|RedlineFlags::ShowInsert); + aPaM.DeleteMark(); + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), 0); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "redline-new-text "); + aDocStat.Reset(); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->SetWordCountDirty(true); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(2), aDocStat.nWord); + //redline-new-text Lorem ipsum + //+++++++++++++++++ ------ + //select start of original text and part of deleted text + aDocStat.Reset(); + pTextNode->CountWords(aDocStat, 17, 25); + CPPUNIT_ASSERT_EQUAL(static_cast<sal_uLong>(5), aDocStat.nChar); + } + + //See https://bugs.libreoffice.org/show_bug.cgi?id=38983 + { + SwDocStat aDocStat; + + OUString sTemplate("ThisXis a test."); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', ' ')); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(4), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(12), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replaceAll("X", " = ")); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(5), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(17), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replaceAll("X", " _ ")); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(5), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(17), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replaceAll("X", " -- ")); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(5), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(14), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(18), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', '_')); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(3), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', '-')); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(3), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', 0x2012)); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(3), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', 0x2015)); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(3), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + //But default configuration should, msword-alike treat emdash + //and endash as word separators for word-counting + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', 0x2013)); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(4), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replace('X', 0x2014)); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(4), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(15), aDocStat.nChar); + aDocStat.Reset(); + + static constexpr OUStringLiteral sChunk = u" \u2013 "; + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, sTemplate.replaceAll("X", sChunk)); + pTextNode = aPaM.GetNode().GetTextNode(); + pTextNode->CountWords(aDocStat, 0, pTextNode->Len()); + CPPUNIT_ASSERT_EQUAL(sal_uLong(4), aDocStat.nWord); + CPPUNIT_ASSERT_EQUAL(sal_uLong(13), aDocStat.nCharExcludingSpaces); + CPPUNIT_ASSERT_EQUAL(sal_uLong(17), aDocStat.nChar); + aDocStat.Reset(); + } +} + +void SwDocTest::testMergePortionsDeleteNotSorted() +{ + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, " AABBCC"); + + SwCharFormat *const pCharFormat(m_pDoc->MakeCharFormat("foo", nullptr)); + SwFormatCharFormat const charFormat(pCharFormat); + + SwFormatINetFormat const inetFormat("http://example.com", ""); + + IDocumentContentOperations & rIDCO(m_pDoc->getIDocumentContentOperations()); + aPaM.SetMark(); + aPaM.GetPoint()->nContent = 2; + aPaM.GetMark()->nContent = 4; + rIDCO.InsertPoolItem(aPaM, charFormat); + aPaM.GetPoint()->nContent = 2; + aPaM.GetMark()->nContent = 5; + rIDCO.InsertPoolItem(aPaM, inetFormat); + aPaM.GetPoint()->nContent = 6; + aPaM.GetMark()->nContent = 8; + rIDCO.InsertPoolItem(aPaM, charFormat); + aPaM.GetPoint()->nContent = 4; + aPaM.GetMark()->nContent = 6; + // this triggered an STL assert in SwpHints::MergePortions() + rIDCO.InsertPoolItem(aPaM, charFormat); +} + +//See https://bugs.libreoffice.org/show_bug.cgi?id=40599 +void SwDocTest::testGraphicAnchorDeletion() +{ + CPPUNIT_ASSERT_EQUAL_MESSAGE("Expected initial 0 count", static_cast<sal_uLong>(0), m_pDoc->getIDocumentStatistics().GetDocStat().nChar); + + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Paragraph 1"); + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "graphic anchor>><<graphic anchor"); + SwNodeIndex nPara2 = aPaM.GetPoint()->nNode; + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Paragraph 3"); + + aPaM.GetPoint()->nNode = nPara2; + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), RTL_CONSTASCII_LENGTH("graphic anchor>>")); + + //Insert a graphic at X of >>X<< in paragraph 2 + SfxItemSet aFlySet(m_pDoc->GetAttrPool(), svl::Items<RES_FRMATR_BEGIN, RES_FRMATR_END-1>); + SwFormatAnchor aAnchor(RndStdIds::FLY_AS_CHAR); + aAnchor.SetAnchor(aPaM.GetPoint()); + aFlySet.Put(aAnchor); + SwFlyFrameFormat *pFrame = m_pDoc->getIDocumentContentOperations().InsertGraphic(aPaM, OUString(), OUString(), nullptr, &aFlySet, nullptr, nullptr); + CPPUNIT_ASSERT_MESSAGE("Expected frame", pFrame != nullptr); + + CPPUNIT_ASSERT_EQUAL_MESSAGE("Should be 1 graphic", static_cast<size_t>(1), m_pDoc->GetFlyCount(FLYCNTTYPE_GRF)); + + //Delete >X< + aPaM.GetPoint()->nNode = nPara2; + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), + RTL_CONSTASCII_LENGTH("graphic anchor>><")+1); + aPaM.SetMark(); + aPaM.GetPoint()->nNode = nPara2; + aPaM.GetPoint()->nContent.Assign(aPaM.GetContentNode(), RTL_CONSTASCII_LENGTH("graphic anchor>")); + m_pDoc->getIDocumentContentOperations().DeleteRange(aPaM); + +#ifdef DEBUG_AS_HTML + { + SvFileStream aPasteDebug(OUString("cppunitDEBUG.html"), StreamMode::WRITE|StreamMode::TRUNC); + WriterRef xWrt; + GetHTMLWriter( String(), String(), xWrt ); + SwWriter aDbgWrt( aPasteDebug, *m_pDoc ); + aDbgWrt.Write( xWrt ); + } +#endif + + CPPUNIT_ASSERT_EQUAL_MESSAGE("Should be 0 graphics", static_cast<size_t>(0), m_pDoc->GetFlyCount(FLYCNTTYPE_GRF)); + + //Now, if instead we swap RndStdIds::FLY_AS_CHAR (inline graphic) to RndStdIds::FLY_AT_CHAR (anchored to character) + //and repeat the above, graphic is *not* deleted, i.e. it belongs to the paragraph, not the + //range to which its anchored, which is annoying. +} + +void SwDocTest::testTableAutoFormats() +{ + SwGlobals::ensure(); + + //create new AutoFormatTable + SwTableAutoFormatTable aTableAFT; + + //check the style size - default is expected + CPPUNIT_ASSERT_EQUAL( size_t(1), aTableAFT.size() ); + + //create new style + SwTableAutoFormat aTableAF( "TestItemStyle" ); + + //create new AutoFormat + SwBoxAutoFormat aBoxAF; + + //SetFont + SvxFontItem aFont( RES_CHRATR_FONT ); + aFont.SetFamily( FontFamily::FAMILY_DECORATIVE ); + aFont.SetPitch( FontPitch::PITCH_VARIABLE ); + aFont.SetCharSet( RTL_TEXTENCODING_MS_1251 ); + aBoxAF.SetFont( aFont ); + //SetHeight + SvxFontHeightItem aHeight( 280, 120, RES_CHRATR_FONTSIZE ); + aBoxAF.SetHeight( aHeight ); + //SetWeight + SvxWeightItem aWeight( FontWeight::WEIGHT_BOLD, RES_CHRATR_WEIGHT ); + aBoxAF.SetWeight( aWeight ); + //SetPosture + SvxPostureItem aPosture( FontItalic::ITALIC_NORMAL, RES_CHRATR_POSTURE ); + aBoxAF.SetPosture( aPosture ); + //SetCJKFont + SvxFontItem aCJKFont( RES_CHRATR_FONT ); + aCJKFont.SetFamily( FontFamily::FAMILY_MODERN ); + aCJKFont.SetPitch( FontPitch::PITCH_FIXED ); + aCJKFont.SetCharSet( RTL_TEXTENCODING_MS_1251 ); + aBoxAF.SetCJKFont( aCJKFont ); + //SetCJKHeight + SvxFontHeightItem aCJKHeight( 230, 110, RES_CHRATR_FONTSIZE ); + aBoxAF.SetCJKHeight( aCJKHeight ); + //SetCJKWeight + SvxWeightItem aCJKWeight( FontWeight::WEIGHT_SEMIBOLD, RES_CHRATR_WEIGHT ); + aBoxAF.SetCJKWeight( aCJKWeight ); + //SetCJKPosture + SvxPostureItem aCJKPosture( FontItalic::ITALIC_OBLIQUE, RES_CHRATR_POSTURE ); + aBoxAF.SetCJKPosture( aCJKPosture ); + //SetCTLFont + SvxFontItem aCTLFont( RES_CHRATR_FONT ); + aCTLFont.SetFamily( FontFamily::FAMILY_ROMAN ); + aCTLFont.SetPitch( FontPitch::PITCH_FIXED ); + aCTLFont.SetCharSet( RTL_TEXTENCODING_MS_1251 ); + aBoxAF.SetCTLFont( aCTLFont ); + //SetCTLHeight + SvxFontHeightItem aCTLHeight( 215, 105, RES_CHRATR_FONTSIZE ); + aBoxAF.SetCTLHeight( aCTLHeight ); + //SetCTLWeight + SvxWeightItem aCTLWeight( FontWeight::WEIGHT_ULTRABOLD, RES_CHRATR_WEIGHT ); + aBoxAF.SetCTLWeight( aCTLWeight ); + //SetCTLPosture + SvxPostureItem aCTLPosture( FontItalic::ITALIC_OBLIQUE, RES_CHRATR_POSTURE ); + aBoxAF.SetCTLPosture( aCTLPosture ); + //SetUnderline + SvxUnderlineItem aUnderline( FontLineStyle::LINESTYLE_DOTTED, RES_CHRATR_UNDERLINE ); + aBoxAF.SetUnderline( aUnderline ); + //SetOverline + SvxOverlineItem aOverline( FontLineStyle::LINESTYLE_DASH, RES_CHRATR_OVERLINE ); + aBoxAF.SetOverline( aOverline ); + //SetCrossedOut + SvxCrossedOutItem aCrossedOut( FontStrikeout::STRIKEOUT_BOLD, RES_CHRATR_CROSSEDOUT ); + aBoxAF.SetCrossedOut( aCrossedOut ); + //SetContour + SvxContourItem aContour( true, RES_CHRATR_CONTOUR ); + aBoxAF.SetContour( aContour ); + //SetShadowed + SvxShadowedItem aShadowed( false, RES_CHRATR_SHADOWED ); + aBoxAF.SetShadowed( aShadowed ); + //SetColor + SvxColorItem aColor( Color(0xFF23FF), RES_CHRATR_COLOR ); + aBoxAF.SetColor( aColor ); + //SetAdjust + SvxAdjustItem aAdjust( SvxAdjust::Center, RES_PARATR_ADJUST ); + aBoxAF.SetAdjust( aAdjust ); + //SetTextOrientation + SvxFrameDirectionItem aTOrientation( SvxFrameDirection::Vertical_RL_TB, RES_FRAMEDIR ); + aBoxAF.SetTextOrientation( aTOrientation ); + //SetVerticalAlignment + SwFormatVertOrient aVAlignment( 3, css::text::VertOrientation::CENTER, css::text::RelOrientation::PAGE_LEFT ); + aBoxAF.SetVerticalAlignment( aVAlignment ); + //SetBox + SvxBoxItem aBox( RES_BOX ); + aBox.SetAllDistances( 5 ); + aBoxAF.SetBox( aBox ); + //SetBackground + SvxBrushItem aBackground( Color(0xFF11FF), RES_BACKGROUND ); + aBoxAF.SetBackground( aBackground ); + //Set m_aTLBR + SvxLineItem aTLBRLine(0); aTLBRLine.ScaleMetrics( 11,12 ); + aBoxAF.SetTLBR(aTLBRLine); + //Set m_aBLTR + SvxLineItem aBLTRLine(0); aBLTRLine.ScaleMetrics( 13,14 ); + aBoxAF.SetBLTR(aBLTRLine); + //Set m_aHorJustify + SvxHorJustifyItem aHJustify( SvxCellHorJustify::Center, 0 ); + aBoxAF.SetHorJustify(aHJustify); + //Set m_aVerJustify + SvxVerJustifyItem aVJustify( SvxCellVerJustify::Center , 0 ); + aBoxAF.SetVerJustify(aVJustify); + //Set m_aStacked + SfxBoolItem aStacked(0, true); + aBoxAF.SetStacked(aStacked); + //Set m_aMargin + SvxMarginItem aSvxMarginItem(sal_Int16(4), sal_Int16(2), sal_Int16(3), sal_Int16(3), TypedWhichId<SvxMarginItem>(0)); + aBoxAF.SetMargin(aSvxMarginItem); + //Set m_aLinebreak + SfxBoolItem aLBreak(0, true); + aBoxAF.SetLinebreak(aLBreak); + //Set m_aRotateAngle + SfxInt32Item aRAngle(sal_Int32(5)); + aBoxAF.SetRotateAngle(aRAngle); + //Set m_aRotateMode + SvxRotateModeItem aSvxRotateModeItem(SVX_ROTATE_MODE_CENTER, TypedWhichId<SvxRotateModeItem>(0)); + aBoxAF.SetRotateMode(aSvxRotateModeItem); + //Set m_sNumFormatString + OUString aNFString = "UnitTestFormat"; + aBoxAF.SetNumFormatString(aNFString); + //Set m_eSysLanguage + LanguageType aSLang( LANGUAGE_ENGLISH_INDIA ); + aBoxAF.SetSysLanguage(aSLang); + //Set m_eNumFormatLanguage + LanguageType aNFLang( LANGUAGE_GERMAN ); + aBoxAF.SetNumFormatLanguage(aNFLang); + //Set m_aKeepWithNextPara + SvxFormatKeepItem aKWNPara( true, 0 ); + aTableAF.SetKeepWithNextPara(aKWNPara); + //Set m_aRepeatHeading + sal_uInt16 aRHeading = 3; + aTableAF.m_aRepeatHeading = aRHeading; + //Set m_bLayoutSplit + bool aLSplit = false; + aTableAF.m_bLayoutSplit = aLSplit; + //Set m_bRowSplit + bool aRSplit = false; + aTableAF.m_bRowSplit = aRSplit; + //Set m_bCollapsingBorders + bool aCBorders = false; + aTableAF.m_bCollapsingBorders = aCBorders; + //Set m_aShadow + SvxShadowItem aShadow( 0, nullptr, 103, SvxShadowLocation::BottomLeft ); + aTableAF.SetShadow(aShadow); + //Set bInclFont + bool aIFont = false; + aTableAF.m_bInclFont = aIFont; + //Set bInclJustify + bool aIJustify = false; + aTableAF.m_bInclJustify = aIJustify; + //Set bInclFrame + bool aIFrame = false; + aTableAF.m_bInclFrame = aIFrame; + //Set bInclBackground + bool aIBackground = false; + aTableAF.m_bInclBackground = aIBackground; + //Set bInclValueFormat + bool aIVFormat = false; + aTableAF.m_bInclValueFormat = aIVFormat; + + //set the box format to AutoFormat + aTableAF.SetBoxFormat( aBoxAF, sal_uInt8(0) ); + //add AutoFormat to AutoFormatTable + aTableAFT.AddAutoFormat( aTableAF ); + + //check the style size + CPPUNIT_ASSERT_EQUAL( size_t(2), aTableAFT.size() ); + + //save the bInclFontstyles + aTableAFT.Save(); + + //check the style size after save + CPPUNIT_ASSERT_EQUAL( size_t(2), aTableAFT.size() ); + + //create new AutoFormatTable + SwTableAutoFormatTable aLoadTAFT; + + //check the style size + CPPUNIT_ASSERT_EQUAL( size_t(1), aLoadTAFT.size() ); + + //load the saved styles + aLoadTAFT.Load(); + + //check the style size after load + CPPUNIT_ASSERT_EQUAL( size_t(2), aLoadTAFT.size() ); + + //assert the values + SwTableAutoFormat* pLoadAF = aLoadTAFT.FindAutoFormat( u"TestItemStyle" ); + CPPUNIT_ASSERT( pLoadAF ); + //GetFont + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetFont() == aFont ) ); + //GetHeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetHeight() == aHeight ) ); + //GetWeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetWeight() == aWeight ) ); + //GetPosture + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetPosture() == aPosture ) ); + //GetCJKFont + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCJKFont() == aCJKFont ) ); + //GetCJKHeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCJKHeight() == aCJKHeight ) ); + //GetCJKWeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCJKWeight() == aCJKWeight ) ); + //GetCJKPosture + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCJKPosture() == aCJKPosture ) ); + //GetCTLFont + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCTLFont() == aCTLFont ) ); + //GetCTLHeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCTLHeight() == aCTLHeight ) ); + //GetCTLWeight + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCTLWeight() == aCTLWeight ) ); + //GetCTLPosture + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCTLPosture() == aCTLPosture ) ); + //GetUnderline + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetUnderline() == aUnderline ) ); + //GetOverline + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetOverline() == aOverline ) ); + //GetCrossedOut + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetCrossedOut() == aCrossedOut ) ); + //GetContour + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetContour() == aContour ) ); + //GetShadowed + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetShadowed() == aShadowed ) ); + //GetColor + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetColor() == aColor) ); + //GetAdjust + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetAdjust() == aAdjust ) ); + //GetTextOrientation + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetTextOrientation() == aTOrientation ) ); + //GetVerticalAlignment + CPPUNIT_ASSERT (bool( pLoadAF->GetBoxFormat(0).GetVerticalAlignment() == aVAlignment ) ); + //GetBox + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetBox() == aBox ) ); + //GetBackground + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetBackground() == aBackground ) ); + //Get m_aTLBR + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetTLBR() == aTLBRLine ) ); + //Get m_aBLTR + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetBLTR() == aBLTRLine ) ); + //Get m_aHorJustify + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetHorJustify() == aHJustify ) ); + //Get m_aVerJustify + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetVerJustify() == aVJustify ) ); + //Get m_aStacked + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetStacked() == aStacked ) ); + //Get m_aMargin + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetMargin() == aSvxMarginItem ) ); + //Get m_aLinebreak + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetLinebreak() == aLBreak ) ); + //Get m_aRotateAngle + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetRotateAngle() == aRAngle ) ); + //Get m_aRotateMode + //SvxRotateModeItem aRMode = aBoxAF.m_aRotateMode;GetRotateMode + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetRotateMode() == aSvxRotateModeItem ) ); + //Get m_sNumFormatString + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetNumFormatString() == aNFString ) ); + //Get m_eSysLanguage + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetSysLanguage() == aSLang ) ); + //Get m_eNumFormatLanguage + CPPUNIT_ASSERT( bool( pLoadAF->GetBoxFormat(0).GetNumFormatLanguage() == aNFLang ) ); + //Get m_aKeepWithNextPara + CPPUNIT_ASSERT( bool( pLoadAF->GetKeepWithNextPara() == aKWNPara ) ); + //Get m_aRepeatHeading + CPPUNIT_ASSERT( bool( pLoadAF->m_aRepeatHeading == aRHeading ) ); + //Get m_bLayoutSplit + CPPUNIT_ASSERT( bool( pLoadAF->m_bLayoutSplit == aLSplit ) ); + //Get m_bRowSplit + CPPUNIT_ASSERT( bool( pLoadAF->m_bRowSplit == aRSplit ) ); + //Get m_bCollapsingBorders + CPPUNIT_ASSERT( bool( pLoadAF->m_bCollapsingBorders == aCBorders ) ); + //Get m_aShadow + CPPUNIT_ASSERT( bool( pLoadAF->GetShadow() == aShadow ) ); + //Get bInclFont + CPPUNIT_ASSERT( bool( pLoadAF->m_bInclFont == aIFont ) ); + //Get bInclJustify + CPPUNIT_ASSERT( bool( pLoadAF->m_bInclJustify == aIJustify ) ); + //Get bInclFrame + CPPUNIT_ASSERT( bool( pLoadAF->m_bInclFrame == aIFrame ) ); + //Get bInclBackground + CPPUNIT_ASSERT( bool( pLoadAF->m_bInclBackground == aIBackground ) ); + //Get bInclValueFormat + CPPUNIT_ASSERT( bool( pLoadAF->m_bInclValueFormat == aIVFormat ) ); +} + +static OUString +translitTest(SwDoc & rDoc, const SwPaM & rPaM, TransliterationFlags const nType) +{ + utl::TransliterationWrapper aTrans( + ::comphelper::getProcessComponentContext(), nType); + rDoc.getIDocumentContentOperations().TransliterateText(rPaM, aTrans); + return rPaM.GetText(); +} + +void SwDocTest::testTransliterate() +{ + // just some simple test to see if it's totally broken + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "foobar"); + aPaM.SetMark(); + aPaM.GetPoint()->nContent = 0; + CPPUNIT_ASSERT_EQUAL(OUString("foobar"), aPaM.GetText()); + + CPPUNIT_ASSERT_EQUAL(OUString("FOOBAR"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::LOWERCASE_UPPERCASE)); + CPPUNIT_ASSERT_EQUAL(OUString("Foobar"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::TITLE_CASE)); + CPPUNIT_ASSERT_EQUAL(OUString("fOOBAR"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::TOGGLE_CASE)); + CPPUNIT_ASSERT_EQUAL(OUString("foobar"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::UPPERCASE_LOWERCASE)); + CPPUNIT_ASSERT_EQUAL(OUString("Foobar"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::SENTENCE_CASE)); + CPPUNIT_ASSERT_EQUAL(OUString("Foobar"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::HIRAGANA_KATAKANA)); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "one (two) three"); + aPaM.SetMark(); + aPaM.GetMark()->nContent = 0; + CPPUNIT_ASSERT_EQUAL(OUString("One (Two) Three"), + translitTest(*m_pDoc, aPaM, + TransliterationFlags::TITLE_CASE)); +} + +namespace +{ + class SwTableFormulaTest : public SwTableFormula + { + SwTableNode *m_pNode; + public: + SwTableFormulaTest(const OUString &rStr, SwTableNode *pNode) + : SwTableFormula(rStr) + , m_pNode(pNode) + { + m_eNmType = INTRNL_NAME; + } + virtual const SwNode* GetNodeOfFormula() const override + { + return m_pNode; + } + }; +} + +//tdf#66353 Expression is faulty +void SwDocTest::testFormulas() +{ + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPosition aPos(aIdx); + + const SwTable *pTable = m_pDoc->InsertTable( + SwInsertTableOptions(SwInsertTableFlags::HeadlineNoBorder, 0), aPos, 1, 3, 0); + SwTableNode* pTableNode = pTable->GetTableNode(); + SwTableFormulaTest aFormula("<\x12-1,0>+<Table1.A1>", pTableNode); + + aFormula.PtrToBoxNm(pTable); + + CPPUNIT_ASSERT_EQUAL(OUString("<?>+<Table1.?>"), aFormula.GetFormula()); + + // tdf#61228: Evaluating non-defined function should return an error + SwCalc aCalc(*m_pDoc); + SwSbxValue val = aCalc.Calculate("foobar()"); + CPPUNIT_ASSERT(aCalc.IsCalcError()); + CPPUNIT_ASSERT(val.IsVoidValue()); + CPPUNIT_ASSERT(val.IsDouble()); + CPPUNIT_ASSERT_EQUAL(DBL_MAX, val.GetDouble()); + // Evaluating non-defined variable should return 0 without an error + val = aCalc.Calculate("foobar"); + CPPUNIT_ASSERT(!aCalc.IsCalcError()); + CPPUNIT_ASSERT(val.IsVoidValue()); + CPPUNIT_ASSERT(val.IsLong()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(0), val.GetLong()); +} + +void SwDocTest::testMarkMove() +{ + IDocumentMarkAccess* pMarksAccess = m_pDoc->getIDocumentMarkAccess(); + + { + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Paragraph 1"); + aPaM.SetMark(); + aPaM.GetMark()->nContent -= aPaM.GetMark()->nContent.GetIndex(); + pMarksAccess->makeMark(aPaM, "Para1", + IDocumentMarkAccess::MarkType::BOOKMARK, sw::mark::InsertMode::New); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Paragraph 2"); + aPaM.SetMark(); + aPaM.GetMark()->nContent -= aPaM.GetMark()->nContent.GetIndex(); + pMarksAccess->makeMark(aPaM, "Para2", + IDocumentMarkAccess::MarkType::BOOKMARK, sw::mark::InsertMode::New); + + m_pDoc->getIDocumentContentOperations().AppendTextNode(*aPaM.GetPoint()); + m_pDoc->getIDocumentContentOperations().InsertString(aPaM, "Paragraph 3"); + aPaM.SetMark(); + aPaM.GetMark()->nContent -= aPaM.GetMark()->nContent.GetIndex(); + pMarksAccess->makeMark(aPaM, "Para3", + IDocumentMarkAccess::MarkType::BOOKMARK, sw::mark::InsertMode::New); + } + + // join paragraph 2 and 3 and check + { + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -2); + SwTextNode& rParaNode2 = dynamic_cast<SwTextNode&>(aIdx.GetNode()); + rParaNode2.JoinNext(); + } + ::sw::mark::IMark* pBM1 = *pMarksAccess->findMark("Para1"); + ::sw::mark::IMark* pBM2 = *pMarksAccess->findMark("Para2"); + ::sw::mark::IMark* pBM3 = *pMarksAccess->findMark("Para3"); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(0) , pBM1->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(11), pBM1->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex(), + pBM1->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(0) , pBM2->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(11), pBM2->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkStart().nNode.GetIndex(), + pBM2->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(11), pBM3->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(22), pBM3->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM3->GetMarkStart().nNode.GetIndex(), + pBM3->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex()+1, + pBM2->GetMarkStart().nNode.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkStart().nNode.GetIndex(), + pBM3->GetMarkStart().nNode.GetIndex()); + + // cut some text + { + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPaM aPaM(aIdx, aIdx, SwNodeOffset(-1)); + aPaM.GetPoint()->nContent += 5; + aPaM.GetMark()->nContent += 6; + m_pDoc->getIDocumentContentOperations().DeleteAndJoin(aPaM); + } + pBM1 = *pMarksAccess->findMark("Para1"); + pBM2 = *pMarksAccess->findMark("Para2"); + pBM3 = *pMarksAccess->findMark("Para3"); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(0), pBM1->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(6), pBM1->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex(), + pBM1->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(6), pBM2->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(12), pBM2->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkStart().nNode.GetIndex(), + pBM2->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(12), pBM3->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(23), pBM3->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM3->GetMarkStart().nNode.GetIndex(), + pBM3->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex(), + pBM2->GetMarkStart().nNode.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkStart().nNode.GetIndex(), + pBM3->GetMarkStart().nNode.GetIndex()); + + // split the paragraph + { + SwNodeIndex aIdx(m_pDoc->GetNodes().GetEndOfContent(), -1); + SwPosition aPos(aIdx); + aPos.nContent += 8; + m_pDoc->getIDocumentContentOperations().SplitNode(aPos, false); + } + pBM1 = *pMarksAccess->findMark("Para1"); + pBM2 = *pMarksAccess->findMark("Para2"); + pBM3 = *pMarksAccess->findMark("Para3"); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(0), pBM1->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(6), pBM1->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex(), + pBM1->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(6), pBM2->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(4), pBM2->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkStart().nNode.GetIndex()+1, + pBM2->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL(sal_Int32(4), pBM3->GetMarkStart().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL(sal_Int32(15), pBM3->GetMarkEnd().nContent.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM3->GetMarkStart().nNode.GetIndex(), + pBM3->GetMarkEnd().nNode.GetIndex()); + + CPPUNIT_ASSERT_EQUAL( + pBM1->GetMarkStart().nNode.GetIndex(), + pBM2->GetMarkStart().nNode.GetIndex()); + CPPUNIT_ASSERT_EQUAL( + pBM2->GetMarkEnd().nNode.GetIndex(), + pBM3->GetMarkEnd().nNode.GetIndex()); +} + +namespace +{ + struct TestRing : public sw::Ring<TestRing> + { + TestRing() : sw::Ring<TestRing>() {}; + TestRing* GetNext() + { return GetNextInRing(); } + TestRing* GetPrev() + { return GetPrevInRing(); } + bool lonely() const + { return unique(); } + }; +} + +void SwDocTest::testIntrusiveRing() +{ + TestRing aRing1, aRing2, aRing3, aRing4, aRing5; + std::vector<TestRing*> vRings + { + &aRing1, + &aRing2, + &aRing3, + &aRing4, + &aRing5 + }; + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(1), aRing1.GetRingContainer().size()); + CPPUNIT_ASSERT(aRing1.lonely()); + CPPUNIT_ASSERT(aRing2.lonely()); + CPPUNIT_ASSERT(aRing3.lonely()); + aRing2.MoveTo(&aRing1); + aRing3.MoveTo(&aRing1); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(3), aRing1.GetRingContainer().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(3), aRing2.GetRingContainer().size()); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(3), aRing3.GetRingContainer().size()); + CPPUNIT_ASSERT(!aRing1.lonely()); + CPPUNIT_ASSERT(!aRing2.lonely()); + CPPUNIT_ASSERT(!aRing3.lonely()); + aRing5.MoveTo(&aRing4); + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(2), aRing4.GetRingContainer().size()); + aRing4.GetRingContainer().merge(aRing1.GetRingContainer()); + for(TestRing* pRing : vRings) + { + CPPUNIT_ASSERT_EQUAL(static_cast<size_t>(5), pRing->GetRingContainer().size()); + } + for(std::vector<TestRing*>::iterator ppRing = vRings.begin(); ppRing != vRings.end(); ++ppRing) + { + std::vector<TestRing*>::iterator ppNext = ppRing+1; + if(ppNext==vRings.end()) + ppNext = vRings.begin(); + CPPUNIT_ASSERT_EQUAL((*ppRing)->GetNext(), *ppNext); + CPPUNIT_ASSERT_EQUAL((*ppNext)->GetPrev(), *ppRing); + } + for(TestRing& r: aRing1.GetRingContainer()) + { + TestRing* pRing = &r; + CPPUNIT_ASSERT(pRing); + } + const TestRing* pConstRing = &aRing1; + for(const TestRing& r: pConstRing->GetRingContainer()) // this should fail without r being const + { + const TestRing* pRing = &r; + CPPUNIT_ASSERT(pRing); + } + TestRing foo, bar; + foo.MoveTo(&bar); + CPPUNIT_ASSERT_EQUAL(&foo, bar.GetNext()); + CPPUNIT_ASSERT_EQUAL(&foo, bar.GetPrev()); + CPPUNIT_ASSERT_EQUAL(&bar, foo.GetNext()); + CPPUNIT_ASSERT_EQUAL(&bar, foo.GetPrev()); + foo.MoveTo(&foo); + CPPUNIT_ASSERT_EQUAL(&bar, bar.GetNext()); + CPPUNIT_ASSERT_EQUAL(&bar, bar.GetPrev()); + CPPUNIT_ASSERT_EQUAL(&foo, foo.GetNext()); + CPPUNIT_ASSERT_EQUAL(&foo, foo.GetPrev()); +} + +namespace +{ + struct TestHint final : SfxHint {}; + struct TestModify : sw::BroadcastingModify + { + }; + struct TestClient : SwClient + { + int m_nModifyCount; + int m_nNotifyCount; + int m_nModifyChangedCount; + const SwModify* m_pLastChangedModify; + TestClient() : m_nModifyCount(0), m_nNotifyCount(0), m_nModifyChangedCount(0), m_pLastChangedModify(nullptr) {}; + virtual void SwClientNotify(const SwModify&, const SfxHint& rHint) override + { + if(typeid(TestHint) == typeid(rHint)) + ++m_nNotifyCount; + else if(dynamic_cast<const sw::LegacyModifyHint*>(&rHint)) + ++m_nModifyCount; + else if(auto pModifyChangedHint = dynamic_cast<const sw::ModifyChangedHint*>(&rHint)) + { + ++m_nModifyChangedCount; + m_pLastChangedModify = pModifyChangedHint->m_pNew; + } + } + }; + struct OtherTestClient : SwClient + { + int m_nModifyCount; + OtherTestClient() : m_nModifyCount(0) {}; + virtual void SwClientNotify(const SwModify&, const SfxHint& rHint) override + { + if(dynamic_cast<const sw::LegacyModifyHint*>(&rHint)) + ++m_nModifyCount; + } + }; + struct TestListener : SvtListener + { + int m_nNotifyCount; + TestListener() : m_nNotifyCount(0) {}; + virtual void Notify( const SfxHint& ) override + { + ++m_nNotifyCount; + } + }; +} +void SwDocTest::testClientModify() +{ + (void) OtherTestClient(); // avoid loplugin:unreffun + TestModify aMod; + TestClient aClient1, aClient2; + OtherTestClient aOtherClient1; + // test client registration + CPPUNIT_ASSERT(!aMod.HasWriterListeners()); + CPPUNIT_ASSERT(!aMod.HasOnlyOneListener()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aClient1.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aClient2.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aClient2.GetRegisteredIn()); + aMod.Add(&aClient1); + CPPUNIT_ASSERT(aMod.HasWriterListeners()); + CPPUNIT_ASSERT(aMod.HasOnlyOneListener()); + aMod.Add(&aClient2); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(&aMod),aClient1.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(&aMod), aClient2.GetRegisteredIn()); + CPPUNIT_ASSERT(aMod.HasWriterListeners()); + CPPUNIT_ASSERT(!aMod.HasOnlyOneListener()); + // test broadcast + aMod.CallSwClientNotify(sw::LegacyModifyHint(nullptr, nullptr)); + CPPUNIT_ASSERT_EQUAL(1,aClient1.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient2.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(0,aClient1.m_nNotifyCount); + CPPUNIT_ASSERT_EQUAL(0,aClient2.m_nNotifyCount); + aMod.CallSwClientNotify(sw::LegacyModifyHint(nullptr, nullptr)); + CPPUNIT_ASSERT_EQUAL(2,aClient1.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(2,aClient2.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(0,aClient1.m_nNotifyCount); + CPPUNIT_ASSERT_EQUAL(0,aClient2.m_nNotifyCount); + // test notify + { + TestHint aHint; + aMod.CallSwClientNotify(aHint); + CPPUNIT_ASSERT_EQUAL(2,aClient1.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(2,aClient2.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient1.m_nNotifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient2.m_nNotifyCount); + } + // test typed iteration + CPPUNIT_ASSERT(typeid(aClient1) != typeid(OtherTestClient)); + { + SwIterator<OtherTestClient,SwModify> aIter(aMod); + for(OtherTestClient* pClient = aIter.First(); pClient ; pClient = aIter.Next()) + CPPUNIT_ASSERT(false); + } + { + int nCount = 0; + SwIterator<TestClient,SwModify> aIter(aMod); + for(TestClient* pClient = aIter.First(); pClient ; pClient = aIter.Next()) + { + CPPUNIT_ASSERT_EQUAL(2,pClient->m_nModifyCount); + ++nCount; + } + CPPUNIT_ASSERT_EQUAL(2,nCount); + } + aMod.Add(&aOtherClient1); + CPPUNIT_ASSERT_EQUAL(0,aOtherClient1.m_nModifyCount); + { + int nCount = 0; + SwIterator<TestClient,SwModify> aIter(aMod); + for(TestClient* pClient = aIter.First(); pClient ; pClient = aIter.Next()) + { + CPPUNIT_ASSERT_EQUAL(2,pClient->m_nModifyCount); + ++nCount; + } + CPPUNIT_ASSERT_EQUAL(2,nCount); + } + CPPUNIT_ASSERT_EQUAL(0,aOtherClient1.m_nModifyCount); + aMod.Remove(&aOtherClient1); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(&aMod),aClient1.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(&aMod),aClient2.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aOtherClient1.GetRegisteredIn()); + // test client self-deregistration during iteration + { + int nCount = 0; + SwIterator<TestClient,SwModify> aIter(aMod); + for(TestClient* pClient = aIter.First(); pClient ; pClient = aIter.Next()) + { + aMod.Remove(pClient); + ++nCount; + } + CPPUNIT_ASSERT_EQUAL(2,nCount); + } + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aClient1.GetRegisteredIn()); + CPPUNIT_ASSERT_EQUAL(static_cast<SwModify*>(nullptr),aClient2.GetRegisteredIn()); + { + SwIterator<TestClient,SwModify> aIter(aMod); + for(TestClient* pClient = aIter.First(); pClient ; pClient = aIter.Next()) + { + CPPUNIT_ASSERT(false); + } + } + aMod.CallSwClientNotify(sw::LegacyModifyHint(nullptr, nullptr)); + CPPUNIT_ASSERT_EQUAL(2,aClient1.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(2,aClient2.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient1.m_nNotifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient2.m_nNotifyCount); +} +void SwDocTest::testBroadcastingModify() +{ + sw::BroadcastingModify aMod; + TestClient aClient; + TestListener aListener; + + aMod.Add(&aClient); + aListener.StartListening(aMod.GetNotifier()); + + aMod.CallSwClientNotify(sw::LegacyModifyHint(nullptr, nullptr)); + CPPUNIT_ASSERT_EQUAL(1,aClient.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(1,aClient.m_nModifyCount); + CPPUNIT_ASSERT_EQUAL(1,aListener.m_nNotifyCount); +} +void SwDocTest::testWriterMultiListener() +{ + TestModify aMod; + TestClient aClient; + sw::WriterMultiListener aMulti(aClient); + CPPUNIT_ASSERT(!aMulti.IsListeningTo(&aMod)); + aMulti.StartListening(&aMod); + CPPUNIT_ASSERT(aMulti.IsListeningTo(&aMod)); + aMulti.EndListeningAll(); + CPPUNIT_ASSERT(!aMulti.IsListeningTo(&aMod)); + aMulti.StartListening(&aMod); + aMulti.EndListening(&aMod); + CPPUNIT_ASSERT(!aMulti.IsListeningTo(&aMod)); + int nPreDeathChangedCount; + { + TestModify aTempMod; + aMod.Add(&aTempMod); + aMulti.StartListening(&aTempMod); + nPreDeathChangedCount = aClient.m_nModifyChangedCount; + } + CPPUNIT_ASSERT(aMulti.IsListeningTo(&aMod)); + CPPUNIT_ASSERT_EQUAL(nPreDeathChangedCount+1, aClient.m_nModifyChangedCount); + CPPUNIT_ASSERT_EQUAL(static_cast<const SwModify*>(&aMod),aClient.m_pLastChangedModify); +} + +void SwDocTest::test64kPageDescs() +{ + size_t nPageDescCount = 65536; // USHRT_MAX + 1 + + for (size_t i = 0; i < nPageDescCount; ++i) + { + OUString aName = "Page" + OUString::number(i); + m_pDoc->MakePageDesc( aName ); + } + + size_t nCount = m_pDoc->GetPageDescCnt(); + // +1 because Writer always creates a dummy page desc + // in a new SwDoc + CPPUNIT_ASSERT_EQUAL( nPageDescCount + 1, nCount ); + + const SwPageDesc &rDesc = m_pDoc->GetPageDesc( nPageDescCount ); + SwPageDesc &rZeroDesc = m_pDoc->GetPageDesc( 0 ); + CPPUNIT_ASSERT_EQUAL( OUString("Page65535"), rDesc.GetName() ); + + SwPageDesc aDesc( rDesc ); + static const OUStringLiteral aChanged(u"Changed01"); + aDesc.SetName( aChanged ); + m_pDoc->ChgPageDesc( nPageDescCount, aDesc ); + + size_t nPos; + SwPageDesc *pDesc = m_pDoc->FindPageDesc( aChanged, &nPos ); + CPPUNIT_ASSERT( pDesc != nullptr ); + CPPUNIT_ASSERT_EQUAL( nPageDescCount, nPos ); + + // check if we didn't mess up PageDesc at pos 0 + // (happens with 16bit int overflow) + OUString aZeroName = rZeroDesc.GetName(); + rZeroDesc = m_pDoc->GetPageDesc( 0 ); + CPPUNIT_ASSERT_EQUAL( aZeroName, rZeroDesc.GetName() ); + + m_pDoc->DelPageDesc( aChanged, /*bBroadcast*/true ); + pDesc = m_pDoc->FindPageDesc( aChanged, &nPos ); + // not there anymore + CPPUNIT_ASSERT( !pDesc ); + CPPUNIT_ASSERT_EQUAL( std::numeric_limits<size_t>::max(), nPos); + + // check if PageDesc at pos 0 is still there + pDesc = m_pDoc->FindPageDesc( aZeroName, &nPos ); + CPPUNIT_ASSERT( pDesc != nullptr ); + CPPUNIT_ASSERT_EQUAL( static_cast<size_t>(0), nPos ); +} + +void SwDocTest::testTdf92308() +{ + CPPUNIT_ASSERT_EQUAL(false, m_pDoc->HasInvisibleContent()); +} + +void SwDocTest::testTableCellComparison() +{ + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellsByColFirst("A1", "Z1") ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByColFirst("Z1", "A1") ); + CPPUNIT_ASSERT_EQUAL( 0, sw_CompareCellsByColFirst("A1", "A1") ); + + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByColFirst("A2", "A1") ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByColFirst("Z3", "A2") ); + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellsByColFirst("A3", "Z1") ); + + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellsByRowFirst("A1", "Z1") ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByRowFirst("Z1", "A1") ); + CPPUNIT_ASSERT_EQUAL( 0, sw_CompareCellsByRowFirst("A1", "A1") ); + + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByRowFirst("A2", "A1") ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByRowFirst("Z3", "A2") ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellsByRowFirst("A3", "Z1") ); + + CPPUNIT_ASSERT_EQUAL( 0, sw_CompareCellRanges("A1", "A1", "A1", "A1", true) ); + CPPUNIT_ASSERT_EQUAL( 0, sw_CompareCellRanges("A1", "Z1", "A1", "Z1", true) ); + CPPUNIT_ASSERT_EQUAL( 0, sw_CompareCellRanges("A1", "Z1", "A1", "Z1", false) ); + + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellRanges("A1", "Z1", "B1", "Z1", true) ); + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellRanges("A1", "Z1", "A2", "Z2", false) ); + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellRanges("A1", "Z1", "A2", "Z2", true) ); + CPPUNIT_ASSERT_EQUAL( -1, sw_CompareCellRanges("A1", "Z1", "A6", "Z2", true) ); + + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellRanges("B1", "Z1", "A1", "Z1", true) ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellRanges("A2", "Z2", "A1", "Z1", false) ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellRanges("A2", "Z2", "A1", "Z1", true) ); + CPPUNIT_ASSERT_EQUAL( +1, sw_CompareCellRanges("A6", "Z2", "A1", "Z1", true) ); + + OUString rCell1("A1"); + OUString rCell2("C5"); + + sw_NormalizeRange(rCell1, rCell2); + CPPUNIT_ASSERT_EQUAL( OUString("A1"), rCell1 ); + CPPUNIT_ASSERT_EQUAL( OUString("C5"), rCell2 ); + + sw_NormalizeRange(rCell2, rCell1); + CPPUNIT_ASSERT_EQUAL( OUString("C5"), rCell1 ); + CPPUNIT_ASSERT_EQUAL( OUString("A1"), rCell2 ); + + rCell1 = OUString("A5"); + rCell2 = OUString("C1"); + + sw_NormalizeRange(rCell1, rCell2); + CPPUNIT_ASSERT_EQUAL( OUString("A1"), rCell1 ); + CPPUNIT_ASSERT_EQUAL( OUString("C5"), rCell2 ); + + sw_NormalizeRange(rCell2, rCell1); + CPPUNIT_ASSERT_EQUAL( OUString("C5"), rCell1 ); + CPPUNIT_ASSERT_EQUAL( OUString("A1"), rCell2 ); + + CPPUNIT_ASSERT_EQUAL( OUString(), sw_GetCellName(-1, -1) ); +} + +void SwDocTest::setUp() +{ + BootstrapFixture::setUp(); + + SwGlobals::ensure(); + m_pDoc = new SwDoc; + m_xDocShRef = new SwDocShell(*m_pDoc, SfxObjectCreateMode::EMBEDDED); + m_xDocShRef->DoInitNew(); +} + +void SwDocTest::tearDown() +{ + m_pDoc = nullptr; // deleted by DoClose() + m_xDocShRef->DoClose(); + m_xDocShRef.clear(); + + BootstrapFixture::tearDown(); +} + +CPPUNIT_TEST_SUITE_REGISTRATION(SwDocTest); + +CPPUNIT_PLUGIN_IMPLEMENT(); + +/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |