/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace com::sun::star; namespace { class Test : public test::BootstrapFixture { public: Test() {} void setUp() override { test::BootstrapFixture::setUp(); mpItemPool = new EditEngineItemPool(); SfxApplication::GetOrCreate(); } void tearDown() override { mpItemPool.clear(); test::BootstrapFixture::tearDown(); } #if HAVE_MORE_FONTS /// Test text portions position when percentage line spacing is set void testLineSpacing(); #endif void testConstruction(); /// Test UNO service class that implements text field items. void testUnoTextFields(); /// AutoCorrect tests void testAutocorrect(); /// Test Copy/Paste with hyperlinks in text using Legacy Format void testHyperlinkCopyPaste(); /// Test Copy/Paste using Legacy Format void testCopyPaste(); /// Test Paste using HTML void testHTMLPaste(); /// Test Paste using an HTML fragment void testHTMLFragmentPaste(); /// Test Copy/Paste with selective selection over multiple paragraphs void testMultiParaSelCopyPaste(); /// Test Copy/Paste with Tabs void testTabsCopyPaste(); /// Test hyperlinks void testHyperlinkSearch(); /// Test Copy/Paste with Bold/Italic text using Legacy Format void testBoldItalicCopyPaste(); /// Test Copy/Paste with Underline text using Legacy Format void testUnderlineCopyPaste(); /// Test Copy/Paste with multiple paragraphs void testMultiParaCopyPaste(); /// Test Copy/Paste with multiple paragraphs having Bold/Italic text void testParaBoldItalicCopyPaste(); void testParaStartCopyPaste(); void testSectionAttributes(); void testLargeParaCopyPaste(); void testTransliterate(); void testTdf147196(); void testTdf148148(); void testSingleLine(); DECL_STATIC_LINK( Test, CalcFieldValueHdl, EditFieldInfo*, void ); CPPUNIT_TEST_SUITE(Test); #if HAVE_MORE_FONTS CPPUNIT_TEST(testLineSpacing); #endif CPPUNIT_TEST(testConstruction); CPPUNIT_TEST(testUnoTextFields); CPPUNIT_TEST(testAutocorrect); CPPUNIT_TEST(testHyperlinkCopyPaste); CPPUNIT_TEST(testCopyPaste); CPPUNIT_TEST(testHTMLPaste); CPPUNIT_TEST(testHTMLFragmentPaste); CPPUNIT_TEST(testMultiParaSelCopyPaste); CPPUNIT_TEST(testTabsCopyPaste); CPPUNIT_TEST(testHyperlinkSearch); CPPUNIT_TEST(testBoldItalicCopyPaste); CPPUNIT_TEST(testUnderlineCopyPaste); CPPUNIT_TEST(testMultiParaCopyPaste); CPPUNIT_TEST(testParaBoldItalicCopyPaste); CPPUNIT_TEST(testParaStartCopyPaste); CPPUNIT_TEST(testSectionAttributes); CPPUNIT_TEST(testLargeParaCopyPaste); CPPUNIT_TEST(testTransliterate); CPPUNIT_TEST(testTdf147196); CPPUNIT_TEST(testTdf148148); CPPUNIT_TEST(testSingleLine); CPPUNIT_TEST_SUITE_END(); private: rtl::Reference mpItemPool; }; #if HAVE_MORE_FONTS void Test::testLineSpacing() { // Create EditEngine's instance EditEngine aEditEngine(mpItemPool.get()); if(aEditEngine.GetRefDevice()->GetDPIY() != 96 || aEditEngine.GetRefDevice()->GetDPIScaleFactor() != 1.0) return; // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL(sal_Int32(0), rDoc.GetTextLen()); CPPUNIT_ASSERT_EQUAL(OUString(), rDoc.GetParaAsString(sal_Int32(0))); // Set initial text OUString aText = "This is multi-line paragraph"; sal_Int32 aTextLen = aText.getLength(); aEditEngine.SetText(aText); // Assert changes - text insertion CPPUNIT_ASSERT_EQUAL(aTextLen, rDoc.GetTextLen()); CPPUNIT_ASSERT_EQUAL(aText, rDoc.GetParaAsString(sal_Int32(0))); // Select all paragraphs ESelection aSelection(0, 0, 0, aTextLen); auto doTest = [&](sal_uInt16 nSpace, sal_uInt16 nExpMaxAscent, sal_uInt32 nExpLineHeight) { std::unique_ptr pSet(new SfxItemSet(aEditEngine.GetEmptyItemSet())); SvxLineSpacingItem aLineSpacing(LINE_SPACE_DEFAULT_HEIGHT, EE_PARA_SBL); aLineSpacing.SetPropLineSpace(nSpace); pSet->Put(aLineSpacing); // Set font SvxFontItem aFont(EE_CHAR_FONTINFO); aFont.SetFamilyName("Liberation Sans"); pSet->Put(aFont); SvxFontHeightItem aFontSize(240, 100, EE_CHAR_FONTHEIGHT); pSet->Put(aFontSize); CPPUNIT_ASSERT_EQUAL(static_cast(3), pSet->Count()); aEditEngine.QuickSetAttribs(*pSet, aSelection); // Assert changes ParaPortion* pParaPortion = aEditEngine.GetParaPortions()[0]; ContentNode* const pNode = pParaPortion->GetNode(); const SvxLineSpacingItem& rLSItem = pNode->GetContentAttribs().GetItem(EE_PARA_SBL); CPPUNIT_ASSERT_EQUAL(SvxInterLineSpaceRule::Prop, rLSItem.GetInterLineSpaceRule()); CPPUNIT_ASSERT_EQUAL(nSpace, rLSItem.GetPropLineSpace()); // Check the first line ParagraphInfos aInfo = aEditEngine.GetParagraphInfos(0); CPPUNIT_ASSERT_EQUAL(nExpMaxAscent, aInfo.nFirstLineMaxAscent); CPPUNIT_ASSERT_EQUAL(nExpLineHeight, aEditEngine.GetLineHeight(0)); }; // Test first case - 60% doTest(60, 122, 153); // Force multiple lines aEditEngine.SetPaperSize(Size(1000, 6000)); CPPUNIT_ASSERT_EQUAL(sal_Int32(4), aEditEngine.GetLineCount(0)); // Test second case - 150% doTest(150, 337, 382); // Test lower Word limit - 6% (factor 0.06) doTest(6, 12, 15); // Test upper Word limit - 13200% (factor 132) doTest(13200, 33615, 33660); } #endif void Test::testConstruction() { EditEngine aEngine(mpItemPool.get()); aEngine.SetText("I am Edit Engine."); } bool includes(const uno::Sequence& rSeq, std::u16string_view rVal) { for (OUString const & s : rSeq) if (s == rVal) return true; return false; } void Test::testUnoTextFields() { { // DATE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::DATE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.DateTime"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // URL rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::URL)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.URL"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // PAGE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::PAGE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.PageNumber"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // PAGES rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::PAGES)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.PageCount"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // TIME rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::TIME)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.DateTime"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // FILE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::DOCINFO_TITLE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.docinfo.Title"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // TABLE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::TABLE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.SheetName"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // EXTENDED TIME rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::EXTENDED_TIME)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.DateTime"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // EXTENDED FILE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::EXTENDED_FILE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.FileName"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // AUTHOR rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::AUTHOR)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.Author"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // MEASURE rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::MEASURE)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.text.textfield.Measure"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // PRESENTATION HEADER rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::PRESENTATION_HEADER)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.presentation.textfield.Header"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // PRESENTATION FOOTER rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::PRESENTATION_FOOTER)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.presentation.textfield.Footer"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } { // PRESENTATION DATE TIME rtl::Reference xField(new SvxUnoTextField(text::textfield::Type::PRESENTATION_DATE_TIME)); uno::Sequence aSvcs = xField->getSupportedServiceNames(); bool bGood = includes(aSvcs, u"com.sun.star.presentation.textfield.DateTime"); CPPUNIT_ASSERT_MESSAGE("expected service is not present.", bGood); } } class TestAutoCorrDoc : public SvxAutoCorrDoc { public: /// just like the real thing, this dummy modifies the rText parameter :( TestAutoCorrDoc(OUString &rText, LanguageType eLang) : m_rText(rText) , m_eLang(eLang) { } OUString const& getResult() const { return m_rText; } private: OUString & m_rText; LanguageType m_eLang; virtual bool Delete( sal_Int32 nStt, sal_Int32 nEnd ) override { //fprintf(stderr, "TestAutoCorrDoc::Delete\n"); m_rText = m_rText.replaceAt(nStt, nEnd-nStt, u""); return true; } virtual bool Insert( sal_Int32 nPos, const OUString& rTxt ) override { //fprintf(stderr, "TestAutoCorrDoc::Insert\n"); m_rText = m_rText.replaceAt(nPos, 0, rTxt); return true; } virtual bool Replace( sal_Int32 nPos, const OUString& rTxt ) override { //fprintf(stderr, "TestAutoCorrDoc::Replace\n"); return ReplaceRange( nPos, rTxt.getLength(), rTxt ); } virtual bool ReplaceRange( sal_Int32 nPos, sal_Int32 nLen, const OUString& rTxt ) override { //fprintf(stderr, "TestAutoCorrDoc::ReplaceRange %d %d %s\n", nPos, nLen, OUStringToOString(rTxt, RTL_TEXTENCODING_UTF8).getStr()); m_rText = m_rText.replaceAt(nPos, nLen, rTxt); return true; } virtual void SetAttr( sal_Int32, sal_Int32, sal_uInt16, SfxPoolItem& ) override { //fprintf(stderr, "TestAutoCorrDoc::SetAttr\n"); } virtual bool SetINetAttr( sal_Int32, sal_Int32, const OUString& ) override { //fprintf(stderr, "TestAutoCorrDoc::SetINetAttr\n"); return true; } virtual OUString const* GetPrevPara(bool) override { //fprintf(stderr, "TestAutoCorrDoc::GetPrevPara\n"); return nullptr; } virtual bool ChgAutoCorrWord( sal_Int32& rSttPos, sal_Int32 nEndPos, SvxAutoCorrect& rACorrect, OUString* pPara ) override { //fprintf(stderr, "TestAutoCorrDoc::ChgAutoCorrWord\n"); if (m_rText.isEmpty()) return false; LanguageTag aLanguageTag( m_eLang); const SvxAutocorrWord* pFnd = rACorrect.SearchWordsInList( m_rText, rSttPos, nEndPos, *this, aLanguageTag); if (pFnd && pFnd->IsTextOnly()) { m_rText = m_rText.replaceAt(rSttPos, nEndPos, pFnd->GetLong()); if( pPara ) pPara->clear(); // =&pCurNode->GetString(); return true; } return false; } virtual bool TransliterateRTLWord( sal_Int32& /*rSttPos*/, sal_Int32 /*nEndPos*/, bool /*bApply*/ ) override { return false; } }; //https://bugs.libreoffice.org/show_bug.cgi?id=55693 //Two capitalized letters are not corrected if dash or slash are directly //before the two letters void Test::testAutocorrect() { SvxAutoCorrect aAutoCorrect((OUString()), (OUString())); { OUString sInput("TEst-TEst"); sal_Unicode const cNextChar(' '); bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); CPPUNIT_ASSERT_EQUAL_MESSAGE("autocorrect", OUString("Test-Test "), aFoo.getResult()); } { OUString sInput("TEst/TEst"); sal_Unicode const cNextChar(' '); bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); CPPUNIT_ASSERT_EQUAL_MESSAGE("autocorrect", OUString("Test/Test "), aFoo.getResult()); } { // test auto-bolding with '*' OUString sInput("*foo"); sal_Unicode const cNextChar('*'); bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); CPPUNIT_ASSERT_EQUAL(OUString("foo"), aFoo.getResult()); } { OUString sInput("Test. test"); sal_Unicode const cNextChar(' '); bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); CPPUNIT_ASSERT_EQUAL_MESSAGE("autocorrect", OUString("Test. Test "), aFoo.getResult()); } // don't autocapitalize after a field mark { OUString sInput("Test. \x01 test"); sal_Unicode const cNextChar(' '); bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); CPPUNIT_ASSERT_EQUAL_MESSAGE("autocorrect", OUString("Test. \x01 test "), aFoo.getResult()); } // consider field contents as text for auto quotes { OUString sInput("T\x01"); sal_Unicode const cNextChar('"'); static constexpr OUStringLiteral sExpected = u"T\x01\u201d"; bool bNbspRunNext = false; TestAutoCorrDoc aFoo(sInput, LANGUAGE_ENGLISH_US); aAutoCorrect.SetAutoCorrFlag(ACFlags::ChgQuotes, true); aAutoCorrect.DoAutoCorrect(aFoo, sInput, sInput.getLength(), cNextChar, true, bNbspRunNext); fprintf(stderr, "text is %x\n", aFoo.getResult()[aFoo.getResult().getLength() - 1]); CPPUNIT_ASSERT_EQUAL_MESSAGE("autocorrect", OUString(sExpected), aFoo.getResult()); } } IMPL_STATIC_LINK( Test, CalcFieldValueHdl, EditFieldInfo*, pInfo, void ) { if (!pInfo) return; const SvxFieldItem& rField = pInfo->GetField(); const SvxFieldData* pField = rField.GetField(); if (const SvxURLField* pURLField = dynamic_cast(pField)) { // URLField OUString aURL = pURLField->GetURL(); switch ( pURLField->GetFormat() ) { case SvxURLFormat::AppDefault: case SvxURLFormat::Repr: { pInfo->SetRepresentation( pURLField->GetRepresentation() ); } break; case SvxURLFormat::Url: { pInfo->SetRepresentation( aURL ); } break; } } else { OSL_FAIL("Unknown Field"); pInfo->SetRepresentation(OUString('?')); } } void Test::testHyperlinkCopyPaste() { // Create Outliner instance Outliner aOutliner(mpItemPool.get(), OutlinerMode ::TextObject); aOutliner.SetCalcFieldValueHdl( LINK( nullptr, Test, CalcFieldValueHdl ) ); // Create EditEngine's instance EditEngine& aEditEngine = const_cast (aOutliner.GetEditEngine()); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // New instance must be empty - no initial text CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Get corresponding Field Item for inserting URLs in text // URL 1 OUString aURL1 = "mailto:///user@example.com"; OUString aRepres1 = "user@example.com"; SvxURLField aURLField1( aURL1, aRepres1, SvxURLFormat::Repr ); SvxFieldItem aField1( aURLField1, EE_FEATURE_FIELD ); // URL 2 OUString aURL2 = "mailto:///example@domain.com"; OUString aRepres2 = "example@domain.com"; SvxURLField aURLField2( aURL2, aRepres2, SvxURLFormat::Repr ); SvxFieldItem aField2( aURLField2, EE_FEATURE_FIELD ); // Insert initial text OUString aParaText = "sampletextfortestingfeaturefields"; // Positions Ref .............*13....*20.......... sal_Int32 aTextLen = aParaText.getLength(); aEditEngine.SetText( aParaText ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aParaText, rDoc.GetParaAsString(sal_Int32(0)) ); // Insert URL 1 ContentNode *pNode = rDoc.GetObject(0); EditSelection aSel1( EditPaM(pNode, 13), EditPaM(pNode, 13) ); aEditEngine.InsertField( aSel1, aField1 ); // Assert Field Count CPPUNIT_ASSERT_EQUAL( sal_uInt16(1), aEditEngine.GetFieldCount(0) ); // Insert URL 2 EditSelection aSel2( EditPaM(pNode, 20 + 1), EditPaM(pNode, 20 + 1) ); aEditEngine.InsertField( aSel2, aField2 ); // Assert Field Count CPPUNIT_ASSERT_EQUAL( sal_uInt16(2), aEditEngine.GetFieldCount(0) ); // Assert URL Fields and text before copy // Check text CPPUNIT_ASSERT_EQUAL( aTextLen + aRepres1.getLength() + aRepres2.getLength(), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString("sampletextforuser@example.comtestingexample@domain.comfeaturefields"), rDoc.GetParaAsString(sal_Int32(0)) ); // Check Field 1 EFieldInfo aURLFieldInfo1 = aEditEngine.GetFieldInfo( sal_Int32(0), sal_uInt16(0) ); CPPUNIT_ASSERT_EQUAL( sal_Int32(13), aURLFieldInfo1.aPosition.nIndex ); CPPUNIT_ASSERT_EQUAL( sal_uInt16(EE_FEATURE_FIELD), aURLFieldInfo1.pFieldItem->Which() ); SvxURLField* pURLField1 = dynamic_cast ( const_cast (aURLFieldInfo1.pFieldItem->GetField()) ); CPPUNIT_ASSERT(pURLField1); CPPUNIT_ASSERT_EQUAL( aURL1, pURLField1->GetURL() ); CPPUNIT_ASSERT_EQUAL( aRepres1, pURLField1->GetRepresentation() ); // Check Field 2 EFieldInfo aURLFieldInfo2 = aEditEngine.GetFieldInfo( sal_Int32(0), sal_uInt16(1) ); CPPUNIT_ASSERT_EQUAL( sal_Int32(21), aURLFieldInfo2.aPosition.nIndex ); CPPUNIT_ASSERT_EQUAL( sal_uInt16(EE_FEATURE_FIELD), aURLFieldInfo2.pFieldItem->Which() ); SvxURLField* pURLField2 = dynamic_cast ( const_cast (aURLFieldInfo2.pFieldItem->GetField()) ); CPPUNIT_ASSERT(pURLField2); CPPUNIT_ASSERT_EQUAL( aURL2, pURLField2->GetURL() ); CPPUNIT_ASSERT_EQUAL( aRepres2, pURLField2->GetRepresentation() ); // Copy text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,10,0,21) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert Changes ACP, ACP: after Copy/Paste // Check the fields count CPPUNIT_ASSERT_EQUAL( sal_uInt16(3), aEditEngine.GetFieldCount(0) ); // Check the updated text length CPPUNIT_ASSERT_EQUAL( aTextLen + 10 + aRepres1.getLength()*2 + aRepres2.getLength(), rDoc.GetTextLen() ); // Check the updated text contents CPPUNIT_ASSERT_EQUAL( OUString("sampletextforuser@example.comtestingexample@domain.comfeaturefieldsforuser@example.comtesting"), rDoc.GetParaAsString(sal_Int32(0)) ); // Check the Fields and their values // Field 1 EFieldInfo aACPURLFieldInfo1 = aEditEngine.GetFieldInfo( sal_Int32(0), sal_uInt16(0) ); CPPUNIT_ASSERT_EQUAL( sal_Int32(13), aACPURLFieldInfo1.aPosition.nIndex ); CPPUNIT_ASSERT_EQUAL( sal_uInt16(EE_FEATURE_FIELD), aACPURLFieldInfo1.pFieldItem->Which() ); SvxURLField* pACPURLField1 = dynamic_cast ( const_cast (aACPURLFieldInfo1.pFieldItem->GetField()) ); CPPUNIT_ASSERT(pACPURLField1); CPPUNIT_ASSERT_EQUAL( aURL1, pACPURLField1->GetURL() ); CPPUNIT_ASSERT_EQUAL( aRepres1, pACPURLField1->GetRepresentation() ); // Field 2 EFieldInfo aACPURLFieldInfo2 = aEditEngine.GetFieldInfo( sal_Int32(0), sal_uInt16(1) ); CPPUNIT_ASSERT_EQUAL( sal_Int32(21), aACPURLFieldInfo2.aPosition.nIndex ); CPPUNIT_ASSERT_EQUAL( sal_uInt16(EE_FEATURE_FIELD), aACPURLFieldInfo2.pFieldItem->Which() ); SvxURLField* pACPURLField2 = dynamic_cast ( const_cast (aACPURLFieldInfo2.pFieldItem->GetField()) ); CPPUNIT_ASSERT(pACPURLField2); CPPUNIT_ASSERT_EQUAL( aURL2, pACPURLField2->GetURL() ); CPPUNIT_ASSERT_EQUAL( aRepres2, pACPURLField2->GetRepresentation() ) ; // Field 3 EFieldInfo aACPURLFieldInfo3 = aEditEngine.GetFieldInfo( sal_Int32(0), sal_uInt16(2) ); CPPUNIT_ASSERT_EQUAL( sal_Int32(38), aACPURLFieldInfo3.aPosition.nIndex ); CPPUNIT_ASSERT_EQUAL( sal_uInt16(EE_FEATURE_FIELD), aACPURLFieldInfo3.pFieldItem->Which() ); SvxURLField* pACPURLField3 = dynamic_cast ( const_cast (aACPURLFieldInfo3.pFieldItem->GetField()) ); CPPUNIT_ASSERT(pACPURLField3); CPPUNIT_ASSERT_EQUAL( aURL1, pACPURLField3->GetURL() ); CPPUNIT_ASSERT_EQUAL( aRepres1, pACPURLField3->GetRepresentation() ); } void Test::testCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Set initial text OUString aText = "This is custom initial text"; sal_Int32 aTextLen = aText.getLength(); aEditEngine.SetText( aText ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aText, rDoc.GetParaAsString(sal_Int32(0)) ); // Copy initial text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,0,0,aTextLen) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen + aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(aText + aText), rDoc.GetParaAsString(sal_Int32(0)) ); } /// XTransferable implementation that provides simple HTML content. class TestHTMLTransferable : public cppu::WeakImplHelper { OString m_aHTML; public: TestHTMLTransferable(const OString& rHTML); uno::Any SAL_CALL getTransferData(const datatransfer::DataFlavor& rFlavor) override; uno::Sequence SAL_CALL getTransferDataFlavors() override; sal_Bool SAL_CALL isDataFlavorSupported(const datatransfer::DataFlavor& rFlavor) override; }; TestHTMLTransferable::TestHTMLTransferable(const OString& rHTML) : m_aHTML(rHTML) { } uno::Any TestHTMLTransferable::getTransferData(const datatransfer::DataFlavor& rFlavor) { if (rFlavor.MimeType != "text/html") { return {}; } uno::Any aRet; SvMemoryStream aStream; aStream.WriteOString(m_aHTML); aRet <<= uno::Sequence(static_cast(aStream.GetData()), aStream.GetSize()); return aRet; } uno::Sequence TestHTMLTransferable::getTransferDataFlavors() { datatransfer::DataFlavor aFlavor; aFlavor.DataType = cppu::UnoType>::get(); aFlavor.MimeType = "text/html"; aFlavor.HumanPresentableName = aFlavor.MimeType; return { aFlavor }; } sal_Bool TestHTMLTransferable::isDataFlavorSupported(const datatransfer::DataFlavor& rFlavor) { return rFlavor.MimeType == "text/html" && rFlavor.DataType == cppu::UnoType>::get(); } void Test::testHTMLPaste() { // Given an empty editeng document: EditEngine aEditEngine(mpItemPool.get()); EditDoc &rDoc = aEditEngine.GetEditDoc(); OString aHTML("\ntest"_ostr); uno::Reference< datatransfer::XTransferable > xData(new TestHTMLTransferable(aHTML)); // When trying to paste HTML: aEditEngine.InsertText(xData, OUString(), rDoc.GetEndPaM(), true); // Then make sure the text gets pasted: // Without the accompanying fix in place, this test would have failed with: // - Expected: test // - Actual : // i.e. RTF and plain text paste worked, but not HTML. CPPUNIT_ASSERT_EQUAL(OUString("test"), rDoc.GetParaAsString(static_cast(0))); } void Test::testHTMLFragmentPaste() { // Given an empty editeng document: EditEngine aEditEngine(mpItemPool.get()); EditDoc &rDoc = aEditEngine.GetEditDoc(); OString aHTML("abc"_ostr); uno::Reference< datatransfer::XTransferable > xData(new TestHTMLTransferable(aHTML)); // When trying to paste an HTML fragment: aEditEngine.InsertText(xData, OUString(), rDoc.GetEndPaM(), true); // Then make sure the text gets pasted: // Without the accompanying fix in place, this test would have failed with: // - Expected: abc // - Actual : // i.e. a HTML fragment without a proper header was ignored on paste. CPPUNIT_ASSERT_EQUAL(OUString("abc"), rDoc.GetParaAsString(static_cast(0))); } void Test::testMultiParaSelCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Insert initial text OUString aFirstPara = "This is first paragraph"; // Selection Ref ........8.............. OUString aSecondPara = "This is second paragraph"; // Selection Ref .............14......... OUString aThirdPara = "This is third paragraph"; OUString aText = aFirstPara + "\n" + aSecondPara + "\n" + aThirdPara; sal_Int32 aTextLen = aFirstPara.getLength() + aSecondPara.getLength() + aThirdPara.getLength(); aEditEngine.SetText( aText ); OUString aCopyText = "first paragraphThis is second"; sal_Int32 aCopyTextLen = aCopyText.getLength(); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); // Copy initial text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,8,1,14) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes OUString aThirdParaAfterCopyPaste = aThirdPara + "first paragraph"; CPPUNIT_ASSERT_EQUAL( aTextLen + aCopyTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdParaAfterCopyPaste, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( OUString("This is second"), rDoc.GetParaAsString(sal_Int32(3)) ); } void Test::testTabsCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // New instance must be empty - no initial text CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Get corresponding Item for inserting tabs in the text SfxVoidItem aTab( EE_FEATURE_TAB ); // Insert initial text OUString aParaText = "sampletextfortestingtab"; // Positions Ref ......*6...............*23 sal_Int32 aTextLen = aParaText.getLength(); aEditEngine.SetText( aParaText ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aParaText, rDoc.GetParaAsString(sal_Int32(0)) ); // Insert tab 1 at desired position ContentNode *pNode = rDoc.GetObject(0); EditSelection aSel1( EditPaM(pNode, 6), EditPaM(pNode, 6) ); aEditEngine.InsertFeature( aSel1, aTab ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen + 1, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString("sample\ttextfortestingtab"), rDoc.GetParaAsString(sal_Int32(0)) ); // Insert tab 2 at desired position EditSelection aSel2( EditPaM(pNode, 23+1), EditPaM(pNode, 23+1) ); aEditEngine.InsertFeature( aSel2, aTab ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen + 2, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString("sample\ttextfortestingtab\t"), rDoc.GetParaAsString(sal_Int32(0)) ); // Copy text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,6,0,aTextLen+2) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen + aTextLen - 6 + 4, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString("sample\ttextfortestingtab\t\ttextfortestingtab\t"), rDoc.GetParaAsString(sal_Int32(0)) ); } class UrlEditEngine : public EditEngine { public: explicit UrlEditEngine(SfxItemPool *pPool) : EditEngine(pPool) {} virtual OUString CalcFieldValue( const SvxFieldItem&, sal_Int32, sal_Int32, std::optional&, std::optional&, std::optional& ) override { return "jim@bob.com"; // a sophisticated view of value: } }; // Odd accounting for hyperlink position & size etc. // https://bugzilla.novell.com/show_bug.cgi?id=467459 void Test::testHyperlinkSearch() { UrlEditEngine aEngine(mpItemPool.get()); EditDoc &rDoc = aEngine.GetEditDoc(); OUString aSampleText = "Please write email to . if you find a fish(not a dog)."; aEngine.SetText(aSampleText); CPPUNIT_ASSERT_EQUAL_MESSAGE("set text", aSampleText, rDoc.GetParaAsString(sal_Int32(0))); ContentNode *pNode = rDoc.GetObject(0); EditSelection aSel(EditPaM(pNode, 22), EditPaM(pNode, 22)); SvxURLField aURLField("mailto:///jim@bob.com", "jim@bob.com", SvxURLFormat::Repr); SvxFieldItem aField(aURLField, EE_FEATURE_FIELD); aEngine.InsertField(aSel, aField); OUString aContent = pNode->GetExpandedText(); CPPUNIT_ASSERT_EQUAL_MESSAGE("get text", OUString("Please write email to jim@bob.com. if you find a fish(not a dog)."), aContent); CPPUNIT_ASSERT_EQUAL_MESSAGE("wrong length", aContent.getLength(), rDoc.GetTextLen()); // Check expansion and positioning re-work CPPUNIT_ASSERT_EQUAL_MESSAGE("wrong length", aContent.getLength(), pNode->GetExpandedLen()); for (sal_Int32 n = 0; n < aContent.getLength(); n++) { sal_Int32 nStart = n, nEnd = n; pNode->UnExpandPositions(nStart,nEnd); CPPUNIT_ASSERT_MESSAGE("out of bound start", nStart < pNode->Len()); CPPUNIT_ASSERT_MESSAGE("out of bound end", nEnd <= pNode->Len()); } static const struct { sal_Int32 mnStart, mnEnd; sal_Int32 mnNewStart, mnNewEnd; } aTrickyOnes[] = { { 0, 1, /* -> */ 0, 1 }, { 21, 25, /* -> */ 21, 23 }, // the field is really just one char { 25, 27, /* -> */ 22, 23 }, { 50, 56, /* -> */ 40, 46 } }; for (size_t n = 0; n < SAL_N_ELEMENTS(aTrickyOnes); n++) { sal_Int32 nStart = aTrickyOnes[n].mnStart; sal_Int32 nEnd = aTrickyOnes[n].mnEnd; pNode->UnExpandPositions(nStart,nEnd); CPPUNIT_ASSERT_EQUAL_MESSAGE( OString("in row " + OString::number(n)).getStr(), aTrickyOnes[n].mnNewStart, nStart); CPPUNIT_ASSERT_EQUAL_MESSAGE( OString("in row " + OString::number(n)).getStr(), aTrickyOnes[n].mnNewEnd, nEnd); } SvxSearchItem aItem(1); //SID_SEARCH_ITEM); aItem.SetBackward(false); aItem.SetSelection(false); aItem.SetSearchString("fish"); CPPUNIT_ASSERT_MESSAGE("no fish", aEngine.HasText(aItem)); aItem.SetSearchString("dog"); CPPUNIT_ASSERT_MESSAGE("no dog", aEngine.HasText(aItem)); } bool hasBold(const editeng::Section& rSecAttr) { return std::any_of(rSecAttr.maAttributes.begin(), rSecAttr.maAttributes.end(), [](const SfxPoolItem* p) { return p->Which() == EE_CHAR_WEIGHT && static_cast(p)->GetWeight() == WEIGHT_BOLD; }); } bool hasItalic(const editeng::Section& rSecAttr) { return std::any_of(rSecAttr.maAttributes.begin(), rSecAttr.maAttributes.end(), [](const SfxPoolItem* p) { return p->Which() == EE_CHAR_ITALIC && static_cast(p)->GetPosture() == ITALIC_NORMAL; }); } void Test::testBoldItalicCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // New instance must be empty - no initial text CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Get corresponding ItemSet for inserting Bold/Italic text std::unique_ptr pSet( new SfxItemSet(aEditEngine.GetEmptyItemSet()) ); SvxWeightItem aBold( WEIGHT_BOLD, EE_CHAR_WEIGHT ); SvxPostureItem aItalic( ITALIC_NORMAL, EE_CHAR_ITALIC ); // Insert initial text OUString aParaText = "boldeditengineitalic"; // Positions Ref ..*2....*8...*13.*17 // Bold Ref ..[ BOLD ]...... // Italic Ref ........[ ITALIC ].. sal_Int32 aTextLen = aParaText.getLength(); aEditEngine.SetText( aParaText ); // Assert changes - text insertion CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aParaText, rDoc.GetParaAsString(sal_Int32(0)) ); // Apply Bold to appropriate selection pSet->Put(aBold); CPPUNIT_ASSERT_EQUAL( static_cast(1), pSet->Count() ); aEditEngine.QuickSetAttribs( *pSet, ESelection(0,2,0,14) ); // Assert changes std::unique_ptr pEditText1( aEditEngine.CreateTextObject() ); std::vector aAttrs1; pEditText1->GetAllSections( aAttrs1 ); // There should be 3 sections - woBold - wBold - woBold (w - with, wo - without) CPPUNIT_ASSERT_EQUAL( size_t(3), aAttrs1.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs1[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs1[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs1[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs1[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs1[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 20, static_cast(aAttrs1[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].maAttributes.size()) ); // Apply Italic to appropriate selection pSet.reset( new SfxItemSet(aEditEngine.GetEmptyItemSet()) ); pSet->Put(aItalic); CPPUNIT_ASSERT_EQUAL( static_cast(1), pSet->Count() ); aEditEngine.QuickSetAttribs( *pSet, ESelection(0,8,0,18) ); // Assert changes std::unique_ptr pEditText2( aEditEngine.CreateTextObject() ); std::vector aAttrs2; pEditText2->GetAllSections( aAttrs2 ); // There should be 5 sections - woBold&woItalic - wBold&woItalic - wBold&wItalic - woBold&wItalic - woBold&woItalic (w - with, wo - without) CPPUNIT_ASSERT_EQUAL( size_t(5), aAttrs2.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 8, static_cast(aAttrs2[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs2[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 8, static_cast(aAttrs2[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs2[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[2].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs2[2]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs2[2]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs2[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[3].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs2[3]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[4].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[4].mnStart) ); CPPUNIT_ASSERT_EQUAL( 20, static_cast(aAttrs2[4].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[4].maAttributes.size()) ); // Copy text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,1,0,aTextLen-1) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen + aTextLen - 2, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(aParaText + "oldeditengineitali" ), rDoc.GetParaAsString(sal_Int32(0)) ); // Check updated text for appropriate Bold/Italics std::unique_ptr pEditText3( aEditEngine.CreateTextObject() ); std::vector aAttrs3; pEditText3->GetAllSections( aAttrs3 ); // There should be 9 sections - woBold&woItalic - wBold&woItalic - wBold&wItalic - woBold&wItalic - woBold&woItalic - wBold&woItalic // - wBold&wItalic - woBold&wItalic - woBold&woItalic(w - with, wo - without) CPPUNIT_ASSERT_EQUAL( size_t(9), aAttrs3.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 8, static_cast(aAttrs3[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 8, static_cast(aAttrs3[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs3[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[2].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[2]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[2]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs3[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs3[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[3].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[3]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[4].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs3[4].mnStart) ); CPPUNIT_ASSERT_EQUAL( 21, static_cast(aAttrs3[4].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[4].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[5].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 21, static_cast(aAttrs3[5].mnStart) ); CPPUNIT_ASSERT_EQUAL( 27, static_cast(aAttrs3[5].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[5].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[5]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[6].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 27, static_cast(aAttrs3[6].mnStart) ); CPPUNIT_ASSERT_EQUAL( 33, static_cast(aAttrs3[6].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[6].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[6]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[6]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[7].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 33, static_cast(aAttrs3[7].mnStart) ); CPPUNIT_ASSERT_EQUAL( 37, static_cast(aAttrs3[7].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[7].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[7]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[8].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 37, static_cast(aAttrs3[8].mnStart) ); CPPUNIT_ASSERT_EQUAL( 38, static_cast(aAttrs3[8].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[8].maAttributes.size()) ); } // Auxiliary function to test Underline text Copy/Paste using Legacy Format bool hasUnderline(const editeng::Section& rSecAttr) { return std::any_of(rSecAttr.maAttributes.begin(), rSecAttr.maAttributes.end(), [](const SfxPoolItem* p) { return p->Which() == EE_CHAR_UNDERLINE && static_cast(p)->GetLineStyle() == LINESTYLE_SINGLE; }); } void Test::testUnderlineCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // New instance must be empty - no initial text CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Get corresponding ItemSet for inserting Underline text std::unique_ptr pSet( new SfxItemSet(aEditEngine.GetEmptyItemSet()) ); SvxUnderlineItem aULine( LINESTYLE_SINGLE, EE_CHAR_UNDERLINE ); // Insert initial text OUString aParaText = "sampletextforunderline"; // Positions Ref ......*6.........*17.. // Underline Ref ......[UNDERLINE ].... sal_Int32 aTextLen = aParaText.getLength(); aEditEngine.SetText( aParaText ); // Apply Underline style pSet->Put( aULine ); CPPUNIT_ASSERT_EQUAL( static_cast(1), pSet->Count() ); aEditEngine.QuickSetAttribs( *pSet, ESelection(0,6,0,18) ); // Assert changes std::unique_ptr pEditText1( aEditEngine.CreateTextObject() ); std::vector aAttrs1; pEditText1->GetAllSections( aAttrs1 ); // There should be 3 sections - woUnderline - wUnderline - woUnderline (w - with, wo - without) CPPUNIT_ASSERT_EQUAL( size_t(3), aAttrs1.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 6, static_cast(aAttrs1[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 6, static_cast(aAttrs1[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs1[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be underlined.", hasUnderline(aAttrs1[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs1[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 22, static_cast(aAttrs1[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].maAttributes.size()) ); // Copy text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,6,0,aTextLen-4) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes CPPUNIT_ASSERT_EQUAL( static_cast(aTextLen + strlen("textforunder")), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(aParaText + "textforunder" ), rDoc.GetParaAsString(sal_Int32(0)) ); // Check updated text for appropriate Underline std::unique_ptr pEditText2( aEditEngine.CreateTextObject() ); std::vector aAttrs2; pEditText2->GetAllSections( aAttrs2 ); // There should be 4 sections - woUnderline - wUnderline - woUnderline - wUnderline (w - with, wo - without) CPPUNIT_ASSERT_EQUAL( size_t(4), aAttrs2.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 6, static_cast(aAttrs2[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 6, static_cast(aAttrs2[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be underlined.", hasUnderline(aAttrs2[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 22, static_cast(aAttrs2[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[2].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 22, static_cast(aAttrs2[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 34, static_cast(aAttrs2[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[3].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be underlined.", hasUnderline(aAttrs2[3]) ); } void Test::testMultiParaCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Insert initial text OUString aFirstPara = "This is first paragraph"; OUString aSecondPara = "This is second paragraph"; OUString aThirdPara = "This is third paragraph"; OUString aText = aFirstPara + "\n" + aSecondPara + "\n" + aThirdPara; sal_Int32 aTextLen = aFirstPara.getLength() + aSecondPara.getLength() + aThirdPara.getLength(); aEditEngine.SetText( aText ); sal_Int32 aCopyTextLen = aFirstPara.getLength() + aSecondPara.getLength(); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); // Copy initial text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,0,1,aSecondPara.getLength()) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes OUString aThirdParaAfterCopyPaste = aThirdPara + aFirstPara; CPPUNIT_ASSERT_EQUAL( aTextLen + aCopyTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdParaAfterCopyPaste, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(3)) ); } void Test::testParaBoldItalicCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Get corresponding ItemSet for inserting Bold/Italic text std::unique_ptr pSet( new SfxItemSet(aEditEngine.GetEmptyItemSet()) ); SvxWeightItem aBold( WEIGHT_BOLD, EE_CHAR_WEIGHT ); SvxPostureItem aItalic( ITALIC_NORMAL, EE_CHAR_ITALIC ); // Insert initial text OUString aFirstPara = "This is first paragraph"; // Positions Ref .....*5.*8....*14*17... // Bold Ref .....[ BOLD ]..... // Italic Ref ..............[ ITA // Copy Ref ........[ Copy Sel OUString aSecondPara = "This is second paragraph"; // Positions Ref .....*5.*8...*13..*18... // Bold Ref .....[ BOLD ]..... // Italic Ref LIC ]............... // Copy Ref ection ].......... OUString aThirdPara = "This is third paragraph"; OUString aText = aFirstPara + "\n" + aSecondPara + "\n" + aThirdPara; sal_Int32 aTextLen = aFirstPara.getLength() + aSecondPara.getLength() + aThirdPara.getLength(); aEditEngine.SetText( aText ); OUString aCopyText = "first paragraphThis is second"; sal_Int32 aCopyTextLen = aCopyText.getLength(); // Assert changes - text insertion CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); // Apply Bold to appropriate selections pSet->Put(aBold); CPPUNIT_ASSERT_EQUAL( static_cast(1), pSet->Count() ); aEditEngine.QuickSetAttribs( *pSet, ESelection(0,5,0,18) ); aEditEngine.QuickSetAttribs( *pSet, ESelection(1,5,1,19) ); // Assert changes std::unique_ptr pEditText1( aEditEngine.CreateTextObject() ); std::vector aAttrs1; pEditText1->GetAllSections( aAttrs1 ); // There should be 7 sections - woB - wB - woB -woB -wB -woB -woB (w - with, wo - without, B - Bold, I - Italic) CPPUNIT_ASSERT_EQUAL( size_t(7), aAttrs1.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs1[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs1[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs1[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs1[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs1[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs1[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[2].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs1[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[3].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[4].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs1[4].mnStart) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs1[4].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[4].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs1[4]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs1[5].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs1[5].mnStart) ); CPPUNIT_ASSERT_EQUAL( 24, static_cast(aAttrs1[5].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[5].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs1[6].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[6].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs1[6].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs1[6].maAttributes.size()) ); // Apply Italic to appropriate selection pSet.reset( new SfxItemSet(aEditEngine.GetEmptyItemSet()) ); pSet->Put(aItalic); CPPUNIT_ASSERT_EQUAL( static_cast(1), pSet->Count() ); aEditEngine.QuickSetAttribs( *pSet, ESelection(0,14,1,9) ); // Assert changes std::unique_ptr pEditText2( aEditEngine.CreateTextObject() ); std::vector aAttrs2; pEditText2->GetAllSections( aAttrs2 ); // There should be 9 sections - woB&woI - wB&woI - wB&wI -woB&wI - woB&wI - wB&wI - wB&woI - woB&woI - woB&woI (w - with, wo - without, B - Bold, I - Italic) CPPUNIT_ASSERT_EQUAL( size_t(9), aAttrs2.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs2[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs2[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs2[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs2[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs2[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[2].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs2[2]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs2[2]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs2[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs2[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[3].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs2[3]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[4].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[4].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs2[4].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[4].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs2[4]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[5].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs2[5].mnStart) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs2[5].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[5].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs2[5]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs2[5]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[6].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs2[6].mnStart) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs2[6].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[6].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs2[6]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs2[7].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs2[7].mnStart) ); CPPUNIT_ASSERT_EQUAL( 24, static_cast(aAttrs2[7].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[7].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs2[8].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[8].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs2[8].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs2[8].maAttributes.size()) ); // Copy text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,8,1,14) ); // Paste text at the end aEditEngine.InsertText( xData, OUString(), rDoc.GetEndPaM(), true ); // Assert changes OUString aThirdParaAfterCopyPaste = aThirdPara + "first paragraph"; CPPUNIT_ASSERT_EQUAL( aTextLen + aCopyTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdParaAfterCopyPaste, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( OUString("This is second"), rDoc.GetParaAsString(sal_Int32(3)) ); // Check updated text for appropriate Bold/Italics std::unique_ptr pEditText3( aEditEngine.CreateTextObject() ); std::vector aAttrs3; pEditText3->GetAllSections( aAttrs3 ); // There should be 15 sections - woB&woI - wB&woI - wB&wI -woB&wI - woB&wI - wB&wI - wB&woI - woB&woI - woB&woI // - wB&woI - wB&wI - woB&wI - -woB&wI - wB&wI - wB&woI (w - with, wo - without, B - Bold, I - Italic) CPPUNIT_ASSERT_EQUAL( size_t(15), aAttrs3.size() ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[0].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[0].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[1].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[1].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs3[1].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[1].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[1]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[2].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs3[2].mnStart) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs3[2].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[2].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[2]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[2]) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[3].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 18, static_cast(aAttrs3[3].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs3[3].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[3].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[3]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[4].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[4].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[4].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[4].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[4]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[5].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[5].mnStart) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs3[5].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[5].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[5]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[5]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[6].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs3[6].mnStart) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs3[6].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[6].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[6]) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[7].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 19, static_cast(aAttrs3[7].mnStart) ); CPPUNIT_ASSERT_EQUAL( 24, static_cast(aAttrs3[7].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[7].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[8].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[8].mnStart) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs3[8].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[8].maAttributes.size()) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[9].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 23, static_cast(aAttrs3[9].mnStart) ); CPPUNIT_ASSERT_EQUAL( 29, static_cast(aAttrs3[9].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[9].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[9]) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[10].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 29, static_cast(aAttrs3[10].mnStart) ); CPPUNIT_ASSERT_EQUAL( 33, static_cast(aAttrs3[10].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[10].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[10]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[10]) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[11].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 33, static_cast(aAttrs3[11].mnStart) ); CPPUNIT_ASSERT_EQUAL( 38, static_cast(aAttrs3[11].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[11].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[11]) ); CPPUNIT_ASSERT_EQUAL( 3, static_cast(aAttrs3[12].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 0, static_cast(aAttrs3[12].mnStart) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[12].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[12].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be italic.", hasItalic(aAttrs3[12]) ); CPPUNIT_ASSERT_EQUAL( 3, static_cast(aAttrs3[13].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 5, static_cast(aAttrs3[13].mnStart) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs3[13].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 2, static_cast(aAttrs3[13].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasBold(aAttrs3[13]) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold and italic.", hasItalic(aAttrs3[13]) ); CPPUNIT_ASSERT_EQUAL( 3, static_cast(aAttrs3[14].mnParagraph) ); CPPUNIT_ASSERT_EQUAL( 9, static_cast(aAttrs3[14].mnStart) ); CPPUNIT_ASSERT_EQUAL( 14, static_cast(aAttrs3[14].mnEnd) ); CPPUNIT_ASSERT_EQUAL( 1, static_cast(aAttrs3[14].maAttributes.size()) ); CPPUNIT_ASSERT_MESSAGE( "This section must be bold.", hasBold(aAttrs3[14]) ); } void Test::testParaStartCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Insert initial text OUString aFirstPara = "This is first paragraph"; // Selection Ref ........8.............. OUString aSecondPara = "This is second paragraph"; // Selection Ref .............14......... OUString aThirdPara = "This is third paragraph"; OUString aText = aFirstPara + "\n" + aSecondPara + "\n" + aThirdPara; sal_Int32 aTextLen = aFirstPara.getLength() + aSecondPara.getLength() + aThirdPara.getLength(); aEditEngine.SetText( aText ); OUString aCopyText = "first paragraphThis is second"; sal_Int32 aCopyTextLen = aCopyText.getLength(); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); // Copy initial text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(0,8,1,14) ); // Paste text at the start aEditEngine.InsertText( xData, OUString(), rDoc.GetStartPaM(), true ); // Assert changes OUString aSecondParaAfterCopyPaste = "This is second" + aFirstPara; CPPUNIT_ASSERT_EQUAL( aTextLen + aCopyTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString("first paragraph"), rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondParaAfterCopyPaste, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(3)) ); } void Test::testSectionAttributes() { EditEngine aEngine(mpItemPool.get()); std::unique_ptr pSet(new SfxItemSet(aEngine.GetEmptyItemSet())); SvxWeightItem aBold(WEIGHT_BOLD, EE_CHAR_WEIGHT); SvxPostureItem aItalic(ITALIC_NORMAL, EE_CHAR_ITALIC); { aEngine.SetText("aaabbbccc"); pSet->Put(aBold); CPPUNIT_ASSERT_EQUAL_MESSAGE("There should be exactly one item.", static_cast(1), pSet->Count()); aEngine.QuickSetAttribs(*pSet, ESelection(0,0,0,6)); // 'aaabbb' - end point is not inclusive. pSet.reset(new SfxItemSet(aEngine.GetEmptyItemSet())); pSet->Put(aItalic); CPPUNIT_ASSERT_EQUAL_MESSAGE("There should be exactly one item.", static_cast(1), pSet->Count()); aEngine.QuickSetAttribs(*pSet, ESelection(0,3,0,9)); // 'bbbccc' std::unique_ptr pEditText(aEngine.CreateTextObject()); CPPUNIT_ASSERT_MESSAGE("Failed to create text object.", pEditText); std::vector aAttrs; pEditText->GetAllSections(aAttrs); // Now, we should have a total of 3 sections. CPPUNIT_ASSERT_EQUAL_MESSAGE("There should be 3 sections.", static_cast(3), aAttrs.size()); // First section should be 0-3 of paragraph 0, and it should only have boldness applied. CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnStart)); CPPUNIT_ASSERT_EQUAL(3, static_cast(aAttrs[0].mnEnd)); CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[0].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be bold.", hasBold(aAttrs[0])); // Second section should be 3-6, and it should be both bold and italic. CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[1].mnParagraph)); CPPUNIT_ASSERT_EQUAL(3, static_cast(aAttrs[1].mnStart)); CPPUNIT_ASSERT_EQUAL(6, static_cast(aAttrs[1].mnEnd)); CPPUNIT_ASSERT_EQUAL(2, static_cast(aAttrs[1].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be bold and italic.", hasBold(aAttrs[1])); CPPUNIT_ASSERT_MESSAGE("This section must be bold and italic.", hasItalic(aAttrs[1])); // Third section should be 6-9, and it should be only italic. CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[2].mnParagraph)); CPPUNIT_ASSERT_EQUAL(6, static_cast(aAttrs[2].mnStart)); CPPUNIT_ASSERT_EQUAL(9, static_cast(aAttrs[2].mnEnd)); CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[2].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be italic.", hasItalic(aAttrs[2])); } { // Set text consisting of 5 paragraphs with the 2nd and 4th paragraphs // being empty. aEngine.Clear(); aEngine.SetText("one\n\ntwo\n\nthree"); sal_Int32 nParaCount = aEngine.GetParagraphCount(); CPPUNIT_ASSERT_EQUAL(sal_Int32(5), nParaCount); // Apply boldness to paragraphs 1, 3, 5 only. Leave 2 and 4 unformatted. pSet.reset(new SfxItemSet(aEngine.GetEmptyItemSet())); pSet->Put(aBold); CPPUNIT_ASSERT_EQUAL_MESSAGE("There should be exactly one item.", static_cast(1), pSet->Count()); aEngine.QuickSetAttribs(*pSet, ESelection(0,0,0,3)); aEngine.QuickSetAttribs(*pSet, ESelection(2,0,2,3)); aEngine.QuickSetAttribs(*pSet, ESelection(4,0,4,5)); std::unique_ptr pEditText(aEngine.CreateTextObject()); CPPUNIT_ASSERT_MESSAGE("Failed to create text object.", pEditText); std::vector aAttrs; pEditText->GetAllSections(aAttrs); CPPUNIT_ASSERT_EQUAL(size_t(5), aAttrs.size()); // 1st, 3rd and 5th sections should correspond with 1st, 3rd and 5th paragraphs. CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnStart)); CPPUNIT_ASSERT_EQUAL(3, static_cast(aAttrs[0].mnEnd)); CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[0].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be bold.", hasBold(aAttrs[0])); CPPUNIT_ASSERT_EQUAL(2, static_cast(aAttrs[2].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[2].mnStart)); CPPUNIT_ASSERT_EQUAL(3, static_cast(aAttrs[2].mnEnd)); CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[2].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be bold.", hasBold(aAttrs[2])); CPPUNIT_ASSERT_EQUAL(4, static_cast(aAttrs[4].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[4].mnStart)); CPPUNIT_ASSERT_EQUAL(5, static_cast(aAttrs[4].mnEnd)); CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[4].maAttributes.size())); CPPUNIT_ASSERT_MESSAGE("This section must be bold.", hasBold(aAttrs[4])); // The 2nd and 4th paragraphs should be empty. CPPUNIT_ASSERT_EQUAL(1, static_cast(aAttrs[1].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[1].mnStart)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[1].mnEnd)); CPPUNIT_ASSERT_MESSAGE("Attribute array should be empty.", aAttrs[1].maAttributes.empty()); CPPUNIT_ASSERT_EQUAL(3, static_cast(aAttrs[3].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[3].mnStart)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[3].mnEnd)); CPPUNIT_ASSERT_MESSAGE("Attribute array should be empty.", aAttrs[3].maAttributes.empty()); } { aEngine.Clear(); aEngine.SetText("one\ntwo"); CPPUNIT_ASSERT_EQUAL(static_cast(2), aEngine.GetParagraphCount()); // embolden 2nd paragraph pSet.reset(new SfxItemSet(aEngine.GetEmptyItemSet())); pSet->Put(aBold); aEngine.QuickSetAttribs(*pSet, ESelection(1,0,1,3)); // disboldify 1st paragraph SvxWeightItem aNotSoBold(WEIGHT_NORMAL, EE_CHAR_WEIGHT); pSet->Put(aNotSoBold); aEngine.QuickSetAttribs(*pSet, ESelection(0,0,0,3)); // now delete & join the paragraphs - this is fdo#85496 scenario aEngine.QuickDelete(ESelection(0,0,1,3)); CPPUNIT_ASSERT_EQUAL(static_cast(1), aEngine.GetParagraphCount()); std::unique_ptr pEditText(aEngine.CreateTextObject()); CPPUNIT_ASSERT_MESSAGE("Failed to create text object.", pEditText); std::vector aAttrs; pEditText->GetAllSections(aAttrs); CPPUNIT_ASSERT_EQUAL(static_cast(1), aAttrs.size()); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnParagraph)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnStart)); CPPUNIT_ASSERT_EQUAL(0, static_cast(aAttrs[0].mnEnd)); std::set whiches; for (size_t i = 0; i < aAttrs[0].maAttributes.size(); ++i) { sal_uInt16 const nWhich(aAttrs[0].maAttributes[i]->Which()); CPPUNIT_ASSERT_MESSAGE("duplicate item in text portion attributes", whiches.insert(nWhich).second); } } } void Test::testLargeParaCopyPaste() { // Create EditEngine's instance EditEngine aEditEngine( mpItemPool.get() ); // Get EditDoc for current EditEngine's instance EditDoc &rDoc = aEditEngine.GetEditDoc(); // Initially no text should be there CPPUNIT_ASSERT_EQUAL( sal_Int32(0), rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( OUString(), rDoc.GetParaAsString(sal_Int32(0)) ); // Insert initial text OUString aFirstPara = "This is first paragraph"; OUString aSecondPara = "This is second paragraph"; OUString aThirdPara = "This is third paragraph"; OUString aFourthPara = "This is fourth paragraph"; OUString aFifthPara = "This is fifth paragraph"; OUString aSixthPara = "This is sixth paragraph"; //Positions Ref: ........*8............. OUString aSeventhPara = "This is seventh paragraph"; OUString aEighthPara = "This is eighth paragraph"; //Positions Ref: .............*13 OUString aNinthPara = "This is ninth paragraph"; OUString aTenthPara = "This is tenth paragraph"; OUString aText = aFirstPara + "\n" + aSecondPara + "\n" + aThirdPara + "\n" + aFourthPara + "\n" + aFifthPara + "\n" + aSixthPara + "\n" + aSeventhPara + "\n" + aEighthPara + "\n" + aNinthPara + "\n" + aTenthPara; sal_Int32 aTextLen = aFirstPara.getLength() + aSecondPara.getLength() + aThirdPara.getLength() + aFourthPara.getLength() + aFifthPara.getLength() + aSixthPara.getLength() + aSeventhPara.getLength() + aEighthPara.getLength() + aNinthPara.getLength() + aTenthPara.getLength(); aEditEngine.SetText( aText ); OUString aCopyText = "sixth paragraphThis is seventh paragraphThis is eighth"; sal_Int32 aCopyTextLen = aCopyText.getLength(); // Assert changes CPPUNIT_ASSERT_EQUAL( aTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( aFourthPara, rDoc.GetParaAsString(sal_Int32(3)) ); CPPUNIT_ASSERT_EQUAL( aFifthPara, rDoc.GetParaAsString(sal_Int32(4)) ); CPPUNIT_ASSERT_EQUAL( aSixthPara, rDoc.GetParaAsString(sal_Int32(5)) ); CPPUNIT_ASSERT_EQUAL( aSeventhPara, rDoc.GetParaAsString(sal_Int32(6)) ); CPPUNIT_ASSERT_EQUAL( aEighthPara, rDoc.GetParaAsString(sal_Int32(7)) ); CPPUNIT_ASSERT_EQUAL( aNinthPara, rDoc.GetParaAsString(sal_Int32(8)) ); CPPUNIT_ASSERT_EQUAL( aTenthPara, rDoc.GetParaAsString(sal_Int32(9)) ); // Copy initial text using legacy format uno::Reference< datatransfer::XTransferable > xData = aEditEngine.CreateTransferable( ESelection(5,8,7,14) ); // Paste text at the end of 4th Para ContentNode* pLastNode = rDoc.GetObject(3); aEditEngine.InsertText( xData, OUString(), EditPaM( pLastNode, pLastNode->Len() ), true ); // Assert changes OUString aFourthParaAfterCopyPaste = aFourthPara + "sixth paragraph"; CPPUNIT_ASSERT_EQUAL( aTextLen + aCopyTextLen, rDoc.GetTextLen() ); CPPUNIT_ASSERT_EQUAL( aFirstPara, rDoc.GetParaAsString(sal_Int32(0)) ); CPPUNIT_ASSERT_EQUAL( aSecondPara, rDoc.GetParaAsString(sal_Int32(1)) ); CPPUNIT_ASSERT_EQUAL( aThirdPara, rDoc.GetParaAsString(sal_Int32(2)) ); CPPUNIT_ASSERT_EQUAL( aFourthParaAfterCopyPaste, rDoc.GetParaAsString(sal_Int32(3)) ); CPPUNIT_ASSERT_EQUAL( aSeventhPara, rDoc.GetParaAsString(sal_Int32(4)) ); CPPUNIT_ASSERT_EQUAL( OUString("This is eighth"), rDoc.GetParaAsString(sal_Int32(5)) ); CPPUNIT_ASSERT_EQUAL( aFifthPara, rDoc.GetParaAsString(sal_Int32(6)) ); CPPUNIT_ASSERT_EQUAL( aSixthPara, rDoc.GetParaAsString(sal_Int32(7)) ); CPPUNIT_ASSERT_EQUAL( aSeventhPara, rDoc.GetParaAsString(sal_Int32(8)) ); CPPUNIT_ASSERT_EQUAL( aEighthPara, rDoc.GetParaAsString(sal_Int32(9)) ); CPPUNIT_ASSERT_EQUAL( aNinthPara, rDoc.GetParaAsString(sal_Int32(10)) ); CPPUNIT_ASSERT_EQUAL( aTenthPara, rDoc.GetParaAsString(sal_Int32(11)) ); } OUString lcl_translitTest(EditEngine& aEditEngine, const OUString& text, const ESelection& esel, const TransliterationFlags nType) { aEditEngine.SetText(text); aEditEngine.TransliterateText(esel, nType); return aEditEngine.GetText(); } void Test::testTransliterate() { // Create EditEngine's instance EditEngine editEng( mpItemPool.get() ); OUString sText("one (two) three"); editEng.SetText(sText); editEng.TransliterateText(ESelection(0, 0, 0, sText.getLength()), TransliterationFlags::TITLE_CASE); CPPUNIT_ASSERT_EQUAL(OUString("One (Two) Three"), editEng.GetText()); using TF = TransliterationFlags; constexpr OUString sText2 = u"Mary Jones met joe Smith. Time Passed."_ustr; int selStart = 12; int selEnd = 12; ESelection esel(0, selStart, 0, selEnd); /* DocumentContentOperationsManager checks if the cursor is inside of a word before transliterating, * but Edit Engine has no such check. Therefore, behavior is different between these two when the * cursor is on a word boundary. */ /* No selection tests. Cursor between the ' ' and 'm' before 'met'. */ CPPUNIT_ASSERT_EQUAL(OUString(""), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary jones met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones MET joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* No selection tests. Cursor between the 't' and the ' ' after 'met'. */ selStart = 14; selEnd = 14; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString(""), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary jones met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* No selection tests. Cursor between the 'h' and the '.' after 'Smith'. */ selStart = 24; selEnd = 24; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString(""), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary jones met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* No selection tests. Cursor between the 'm' and 'e' in 'met'. */ selStart = 12; selEnd = 12; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString(""), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary jones met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones MET joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when there is a selection that does not cross a word boundary: "met" */ selStart = 11; selEnd = 14; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString("met"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones MET joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when there is a selection that does not begin at a word boundary: "et" */ selStart = 12; selEnd = 14; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString("et"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones mEt joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones mEt joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones mET joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when there is a selection that ends in the middle of a word */ selStart = 11; selEnd = 13; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString("me"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones Met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones MEt joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when there is a selection that crosses a word boundary: "nes met joe Sm" */ selStart = 7; selEnd = 21; esel = ESelection(0, selStart, 0, selEnd); CPPUNIT_ASSERT_EQUAL(OUString("nes met joe Sm"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary JoNes met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary JoNes Met Joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary JoNES MET JOE SMith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when there is a selection that crosses a sentence boundary: "joe Smith. Time Passed." */ selStart = 15; selEnd = 38; esel = ESelection(0, selStart, 0, selEnd); editEng.SetText(sText2); CPPUNIT_ASSERT_EQUAL(OUString("joe Smith. Time Passed."), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met Joe smith. Time passed."), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met Joe Smith. Time Passed."), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met JOE SMITH. TIME PASSED."), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("Mary Jones met joe smith. time passed."), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test behavior when sentence ends with a capital that is not selected: "CURRENT IS EQUAL TO 10 A" */ selStart = 0; selEnd = 19; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText3(u"CURRENT IS EQUAL TO 10 A"_ustr); editEng.SetText(sText3); CPPUNIT_ASSERT_EQUAL(OUString("CURRENT IS EQUAL TO"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("Current is equal to 10 A"), lcl_translitTest(editEng, sText3, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("Current Is Equal To 10 A"), lcl_translitTest(editEng, sText3, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("CURRENT IS EQUAL TO 10 A"), lcl_translitTest(editEng, sText3, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("current is equal to 10 A"), lcl_translitTest(editEng, sText3, esel, TF::UPPERCASE_LOWERCASE)); } void Test::testTdf147196() { EditEngine editEng( mpItemPool.get() ); editEng.SetText("2.2 Publication of information - CAA\nSection 4.2 of a CA\'s Certificate Policy and/or Certification Practice Statement SHALL state the CA\'s policy or practice on processing CAA Records for Fully Qualified Domain Names; that policy shall be consistent with these Requirements. \n\nIt shall clearly specify the set of Issuer Domain Names that the CA recognises in CAA \"issue\" or \"issuewild\" records as permitting it to issue. The CA SHALL log all actions taken, if any, consistent with its processing practice."); editEng.TransliterateText(ESelection(0, 0, 3, 232), TransliterationFlags::TITLE_CASE); CPPUNIT_ASSERT_EQUAL(OUString("2.2 Publication Of Information - Caa\nSection 4.2 Of A Ca\'s Certificate Policy And/Or Certification Practice Statement Shall State The Ca\'s Policy Or Practice On Processing Caa Records For Fully Qualified Domain Names; That Policy Shall Be Consistent With These Requirements. \n\nIt Shall Clearly Specify The Set Of Issuer Domain Names That The Ca Recognises In Caa \"Issue\" Or \"Issuewild\" Records As Permitting It To Issue. The Ca Shall Log All Actions Taken, If Any, Consistent With Its Processing Practice."), editEng.GetText()); } void Test::testTdf148148() { using TF = TransliterationFlags; EditEngine editEng( mpItemPool.get() ); /* Test what happens when node contains text but selection does not contain any text */ int selStart = 0; int selEnd = 3; ESelection esel(0, selStart, 0, selEnd); constexpr OUString sText1(u" text"_ustr); editEng.SetText(sText1); CPPUNIT_ASSERT_EQUAL(OUString(" "), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString(" text"), lcl_translitTest(editEng, sText1, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" text"), lcl_translitTest(editEng, sText1, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" text"), lcl_translitTest(editEng, sText1, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString(" text"), lcl_translitTest(editEng, sText1, esel, TF::UPPERCASE_LOWERCASE)); selStart = 4; selEnd = 8; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText2(u"text "_ustr); editEng.SetText(sText2); CPPUNIT_ASSERT_EQUAL(OUString(" "), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("text "), lcl_translitTest(editEng, sText2, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("text "), lcl_translitTest(editEng, sText2, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("text "), lcl_translitTest(editEng, sText2, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("text "), lcl_translitTest(editEng, sText2, esel, TF::UPPERCASE_LOWERCASE)); /* Test what happens when node contains only non-word text but selection does not contain any text */ selStart = 0; selEnd = 3; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText3(u" -1"_ustr); editEng.SetText(sText3); CPPUNIT_ASSERT_EQUAL(OUString(" "), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText3, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText3, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText3, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText3, esel, TF::UPPERCASE_LOWERCASE)); selStart = 2; selEnd = 6; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText4(u"-1 "_ustr); editEng.SetText(sText4); CPPUNIT_ASSERT_EQUAL(OUString(" "), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText4, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText4, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText4, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText4, esel, TF::UPPERCASE_LOWERCASE)); /* Test what happens when node and selection contains only non-word text */ selStart = 0; selEnd = 5; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText5(u" -1"_ustr); editEng.SetText(sText3); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText5, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText5, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText5, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString(" -1"), lcl_translitTest(editEng, sText5, esel, TF::UPPERCASE_LOWERCASE)); selStart = 0; selEnd = 5; esel = ESelection(0, selStart, 0, selEnd); constexpr OUString sText6(u"-1 "_ustr); editEng.SetText(sText4); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), editEng.GetText(esel)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText6, esel, TF::SENTENCE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText6, esel, TF::TITLE_CASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText6, esel, TF::LOWERCASE_UPPERCASE)); CPPUNIT_ASSERT_EQUAL(OUString("-1 "), lcl_translitTest(editEng, sText6, esel, TF::UPPERCASE_LOWERCASE)); } void Test::testSingleLine() { EditEngine aEditEngine( mpItemPool.get() ); aEditEngine.SetSingleLine(true); aEditEngine.SetText("Bolivian\nSanta Cruz de la Sierra"); aEditEngine.QuickFormatDoc(true); CPPUNIT_ASSERT_EQUAL(true, aEditEngine.IsFormatted()); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), aEditEngine.GetParagraphCount()); CPPUNIT_ASSERT_EQUAL(sal_Int32(1), aEditEngine.GetLineCount(0)); } CPPUNIT_TEST_SUITE_REGISTRATION(Test); } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */