2020-08-03 21:03:43 +02:00
|
|
|
/* -*- 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/.
|
|
|
|
*/
|
|
|
|
|
2022-11-03 11:42:17 +01:00
|
|
|
#include <test/unoapixml_test.hxx>
|
2020-08-03 21:03:43 +02:00
|
|
|
|
2021-01-15 11:57:50 +01:00
|
|
|
#include <com/sun/star/beans/XPropertySet.hpp>
|
2022-05-09 10:14:29 +02:00
|
|
|
#include <com/sun/star/beans/PropertyValues.hpp>
|
2021-01-15 11:57:50 +01:00
|
|
|
#include <com/sun/star/frame/XStorable.hpp>
|
|
|
|
#include <com/sun/star/text/XTextDocument.hpp>
|
2021-07-23 13:41:27 +02:00
|
|
|
#include <com/sun/star/text/BibliographyDataType.hpp>
|
2022-05-19 09:12:21 +02:00
|
|
|
#include <com/sun/star/text/TextContentAnchorType.hpp>
|
2021-09-02 16:39:26 +02:00
|
|
|
#include <com/sun/star/style/XStyleFamiliesSupplier.hpp>
|
2023-01-28 18:43:51 +09:00
|
|
|
#include <com/sun/star/drawing/XDrawPageSupplier.hpp>
|
2023-03-03 18:04:46 +03:00
|
|
|
#include <com/sun/star/drawing/XDrawPagesSupplier.hpp>
|
2023-02-23 08:21:46 +01:00
|
|
|
#include <com/sun/star/text/XTextFramesSupplier.hpp>
|
2021-01-15 11:57:50 +01:00
|
|
|
|
|
|
|
#include <comphelper/propertysequence.hxx>
|
2021-07-23 13:41:27 +02:00
|
|
|
#include <comphelper/propertyvalue.hxx>
|
|
|
|
#include <comphelper/sequenceashashmap.hxx>
|
2021-01-15 11:57:50 +01:00
|
|
|
#include <unotools/tempfile.hxx>
|
2023-01-28 18:43:51 +09:00
|
|
|
#include <docmodel/uno/UnoTheme.hxx>
|
2023-02-25 19:12:23 +09:00
|
|
|
#include <docmodel/theme/Theme.hxx>
|
2020-08-03 21:03:43 +02:00
|
|
|
|
|
|
|
using namespace ::com::sun::star;
|
|
|
|
|
|
|
|
/// Covers xmloff/source/text/ fixes.
|
2022-11-03 11:42:17 +01:00
|
|
|
class XmloffStyleTest : public UnoApiXmlTest
|
2020-08-03 21:03:43 +02:00
|
|
|
{
|
|
|
|
public:
|
2022-10-24 13:43:14 +02:00
|
|
|
XmloffStyleTest();
|
2020-08-03 21:03:43 +02:00
|
|
|
};
|
|
|
|
|
2022-10-24 13:43:14 +02:00
|
|
|
XmloffStyleTest::XmloffStyleTest()
|
2022-11-03 11:42:17 +01:00
|
|
|
: UnoApiXmlTest("/xmloff/qa/unit/data/")
|
2021-09-03 16:42:46 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-08-03 21:03:43 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testMailMergeInEditeng)
|
|
|
|
{
|
|
|
|
// Without the accompanying fix in place, this test would have failed, as unexpected
|
2020-08-04 12:34:03 +02:00
|
|
|
// <text:database-display> in editeng text aborted the whole import process.
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"mail-merge-editeng.odt");
|
2020-08-03 21:03:43 +02:00
|
|
|
}
|
|
|
|
|
2021-01-15 11:57:50 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCommentResolved)
|
|
|
|
{
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
2021-01-15 11:57:50 +01:00
|
|
|
uno::Sequence<beans::PropertyValue> aCommentProps = comphelper::InitPropertySequence({
|
2022-05-02 14:25:43 +02:00
|
|
|
{ "Text", uno::Any(OUString("comment")) },
|
2021-01-15 11:57:50 +01:00
|
|
|
});
|
2022-10-24 13:43:14 +02:00
|
|
|
dispatchCommand(mxComponent, ".uno:InsertAnnotation", aCommentProps);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2021-01-15 11:57:50 +01:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParaEnumAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParaEnum = xParaEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xPara(xParaEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortionEnum = xPara->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xPortion(xPortionEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xField(xPortion->getPropertyValue("TextField"),
|
|
|
|
uno::UNO_QUERY);
|
2022-05-02 14:25:43 +02:00
|
|
|
xField->setPropertyValue("Resolved", uno::Any(true));
|
2021-01-15 11:57:50 +01:00
|
|
|
|
2022-10-24 13:43:14 +02:00
|
|
|
saveAndReload("writer8");
|
|
|
|
xTextDocument.set(mxComponent, uno::UNO_QUERY);
|
2021-01-15 11:57:50 +01:00
|
|
|
xParaEnumAccess.set(xTextDocument->getText(), uno::UNO_QUERY);
|
|
|
|
xParaEnum = xParaEnumAccess->createEnumeration();
|
|
|
|
xPara.set(xParaEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
xPortionEnum = xPara->createEnumeration();
|
|
|
|
xPortion.set(xPortionEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
xField.set(xPortion->getPropertyValue("TextField"), uno::UNO_QUERY);
|
|
|
|
bool bResolved = false;
|
|
|
|
xField->getPropertyValue("Resolved") >>= bResolved;
|
|
|
|
// Without the accompanying fix in place, this test would have failed, as the resolved state was
|
|
|
|
// not saved for non-range comments.
|
|
|
|
CPPUNIT_ASSERT(bResolved);
|
|
|
|
}
|
|
|
|
|
2021-07-23 13:41:27 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testBibliographyLocalUrl)
|
|
|
|
{
|
|
|
|
// Given a document with a biblio field, with non-empty LocalURL:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xFactory(mxComponent, uno::UNO_QUERY);
|
2021-07-23 13:41:27 +02:00
|
|
|
uno::Reference<beans::XPropertySet> xField(
|
|
|
|
xFactory->createInstance("com.sun.star.text.TextField.Bibliography"), uno::UNO_QUERY);
|
|
|
|
uno::Sequence<beans::PropertyValue> aFields = {
|
|
|
|
comphelper::makePropertyValue("BibiliographicType", text::BibliographyDataType::WWW),
|
|
|
|
comphelper::makePropertyValue("Identifier", OUString("AT")),
|
|
|
|
comphelper::makePropertyValue("Author", OUString("Author")),
|
|
|
|
comphelper::makePropertyValue("Title", OUString("Title")),
|
|
|
|
comphelper::makePropertyValue("URL", OUString("http://www.example.com/test.pdf#page=1")),
|
|
|
|
comphelper::makePropertyValue("LocalURL", OUString("file:///home/me/test.pdf")),
|
|
|
|
};
|
2022-05-02 14:25:43 +02:00
|
|
|
xField->setPropertyValue("Fields", uno::Any(aFields));
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2021-07-23 13:41:27 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
uno::Reference<text::XTextContent> xContent(xField, uno::UNO_QUERY);
|
|
|
|
xText->insertTextContent(xCursor, xContent, /*bAbsorb=*/false);
|
|
|
|
|
|
|
|
// When invoking ODT export + import on it:
|
2022-10-25 10:18:11 +02:00
|
|
|
saveAndReload("writer8");
|
2021-07-23 13:41:27 +02:00
|
|
|
// Without the accompanying fix in place, this test would have resulted in an assertion failure,
|
|
|
|
// as LocalURL was mapped to XML_TOKEN_INVALID.
|
|
|
|
|
|
|
|
// Then make sure that LocalURL is preserved:
|
2022-10-24 13:43:14 +02:00
|
|
|
xTextDocument.set(mxComponent, uno::UNO_QUERY);
|
2021-07-23 13:41:27 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParaEnumAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParaEnum = xParaEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xPara(xParaEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortionEnum = xPara->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xPortion(xPortionEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
xField.set(xPortion->getPropertyValue("TextField"), uno::UNO_QUERY);
|
|
|
|
comphelper::SequenceAsHashMap aMap(xField->getPropertyValue("Fields"));
|
|
|
|
CPPUNIT_ASSERT(aMap.find("LocalURL") != aMap.end());
|
|
|
|
auto aActual = aMap["LocalURL"].get<OUString>();
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("file:///home/me/test.pdf"), aActual);
|
|
|
|
}
|
|
|
|
|
2023-03-07 11:35:04 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testBibliographyTargetURL1)
|
|
|
|
{
|
|
|
|
// Given a document with a biblio field, with non-empty LocalURL:
|
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xFactory(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xField(
|
|
|
|
xFactory->createInstance("com.sun.star.text.TextField.Bibliography"), uno::UNO_QUERY);
|
|
|
|
uno::Sequence<beans::PropertyValue> aFields = {
|
|
|
|
comphelper::makePropertyValue("Identifier", OUString("AT")),
|
|
|
|
comphelper::makePropertyValue("URL", OUString("https://display.url/test1.pdf#page=1")),
|
2023-04-06 00:25:34 +02:00
|
|
|
comphelper::makePropertyValue("TargetType", OUString("1")),
|
2023-03-07 11:35:04 +01:00
|
|
|
comphelper::makePropertyValue("TargetURL", OUString("https://target.url/test2.pdf#page=2")),
|
|
|
|
};
|
|
|
|
xField->setPropertyValue("Fields", uno::Any(aFields));
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
uno::Reference<text::XTextContent> xContent(xField, uno::UNO_QUERY);
|
|
|
|
xText->insertTextContent(xCursor, xContent, /*bAbsorb=*/false);
|
|
|
|
|
|
|
|
// When invoking ODT export + import on it:
|
|
|
|
saveAndReload("writer8");
|
|
|
|
|
|
|
|
// Then make sure that URL, TargetURL and UseTargetURL are preserved and independent:
|
|
|
|
xTextDocument.set(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParaEnumAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParaEnum = xParaEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xPara(xParaEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortionEnum = xPara->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xPortion(xPortionEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
xField.set(xPortion->getPropertyValue("TextField"), uno::UNO_QUERY);
|
|
|
|
comphelper::SequenceAsHashMap aMap(xField->getPropertyValue("Fields"));
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT(aMap.find("URL") != aMap.end());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("https://display.url/test1.pdf#page=1"),
|
|
|
|
aMap["URL"].get<OUString>());
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT(aMap.find("TargetURL") != aMap.end());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("https://target.url/test2.pdf#page=2"),
|
|
|
|
aMap["TargetURL"].get<OUString>());
|
|
|
|
|
2023-04-06 00:25:34 +02:00
|
|
|
CPPUNIT_ASSERT(aMap.find("TargetType") != aMap.end());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("1"), aMap["TargetType"].get<OUString>());
|
2023-03-07 11:35:04 +01:00
|
|
|
}
|
|
|
|
|
2021-08-30 20:12:46 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCommentTableBorder)
|
|
|
|
{
|
|
|
|
// Without the accompanying fix in place, this failed to load, as a comment that started in a
|
|
|
|
// table and ended outside a table aborted the whole importer.
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"comment-table-border.fodt");
|
2021-08-30 20:12:46 +02:00
|
|
|
}
|
|
|
|
|
2021-09-02 16:39:26 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testParaStyleListLevel)
|
|
|
|
{
|
|
|
|
// Given a document with style:list-level="...":
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"para-style-list-level.fodt");
|
2021-09-02 16:39:26 +02:00
|
|
|
|
|
|
|
// Then make sure we map that to the paragraph style's numbering level:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<style::XStyleFamiliesSupplier> xStyleFamiliesSupplier(mxComponent,
|
2021-09-02 16:39:26 +02:00
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XNameAccess> xStyleFamilies
|
|
|
|
= xStyleFamiliesSupplier->getStyleFamilies();
|
|
|
|
uno::Reference<container::XNameAccess> xStyleFamily(
|
|
|
|
xStyleFamilies->getByName("ParagraphStyles"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xStyle(xStyleFamily->getByName("mystyle"), uno::UNO_QUERY);
|
|
|
|
sal_Int16 nNumberingLevel{};
|
|
|
|
CPPUNIT_ASSERT(xStyle->getPropertyValue("NumberingLevel") >>= nNumberingLevel);
|
2021-12-13 17:42:00 +02:00
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int16>(1), nNumberingLevel);
|
2021-09-03 16:42:46 +02:00
|
|
|
|
|
|
|
// Test the export as well:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2021-09-03 16:42:46 +02:00
|
|
|
|
|
|
|
// Then make sure we save the style's numbering level:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("styles.xml");
|
2021-09-03 16:42:46 +02:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - XPath '/office:document-styles/office:styles/style:style[@style:name='mystyle']' no attribute 'list-level' exist
|
|
|
|
// i.e. a custom NumberingLevel was lost on save.
|
|
|
|
assertXPath(pXmlDoc, "/office:document-styles/office:styles/style:style[@style:name='mystyle']",
|
|
|
|
"list-level", "2");
|
2021-09-02 16:39:26 +02:00
|
|
|
}
|
|
|
|
|
2022-02-24 16:43:19 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContinueNumberingWord)
|
|
|
|
{
|
|
|
|
// Given a document, which is produced by Word and contains text:continue-numbering="true":
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"continue-numbering-word.odt");
|
2022-02-24 16:43:19 +01:00
|
|
|
|
|
|
|
// Then make sure that the numbering from the 1st para is continued on the 3rd para:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-02-24 16:43:19 +01:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParaEnumAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParaEnum = xParaEnumAccess->createEnumeration();
|
|
|
|
xParaEnum->nextElement();
|
|
|
|
xParaEnum->nextElement();
|
|
|
|
uno::Reference<beans::XPropertySet> xPara(xParaEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
auto aActual = xPara->getPropertyValue("ListLabelString").get<OUString>();
|
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: 2.
|
|
|
|
// - Actual : 1.
|
|
|
|
// i.e. the numbering was not continued, like in Word.
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("2."), aActual);
|
|
|
|
}
|
|
|
|
|
2022-03-10 16:29:48 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testListId)
|
|
|
|
{
|
|
|
|
// Given a document with a simple list (no continue-list="..." attribute):
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"list-id.fodt");
|
2022-03-10 16:29:48 +01:00
|
|
|
|
|
|
|
// When storing that document as ODF:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-03-10 16:29:48 +01:00
|
|
|
|
|
|
|
// Then make sure that unreferenced xml:id="..." attributes are not written:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-03-10 16:29:48 +01:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - XPath '//text:list' unexpected 'id' attribute
|
|
|
|
// i.e. xml:id="..." was written unconditionally, even when no other list needed it.
|
|
|
|
assertXPathNoAttribute(pXmlDoc, "//text:list", "id");
|
|
|
|
}
|
|
|
|
|
2022-03-11 12:29:30 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testClearingBreakExport)
|
|
|
|
{
|
|
|
|
// Given a document with a clearing break:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-03-11 12:29:30 +01:00
|
|
|
uno::Reference<text::XTextContent> xLineBreak(
|
|
|
|
xMSF->createInstance("com.sun.star.text.LineBreak"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xLineBreakProps(xLineBreak, uno::UNO_QUERY);
|
|
|
|
// SwLineBreakClear::ALL;
|
|
|
|
sal_Int16 eClear = 3;
|
2022-05-02 14:25:43 +02:00
|
|
|
xLineBreakProps->setPropertyValue("Clear", uno::Any(eClear));
|
2022-03-11 12:29:30 +01:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertTextContent(xCursor, xLineBreak, /*bAbsorb=*/false);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-03-11 12:29:30 +01:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-03-11 12:29:30 +01:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - XPath '//text:line-break' number of nodes is incorrect
|
|
|
|
// i.e. the clearing break was lost on export.
|
|
|
|
assertXPath(pXmlDoc, "//text:line-break", "clear", "all");
|
|
|
|
}
|
|
|
|
|
2022-03-16 09:40:48 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testClearingBreakImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a clearing break:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"clearing-break.fodt");
|
2022-03-16 09:40:48 +01:00
|
|
|
|
|
|
|
// Then make sure that the "clear" attribute is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-03-16 09:40:48 +01:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
// First portion is the image.
|
|
|
|
xPortions->nextElement();
|
|
|
|
// Second portion is "foo".
|
|
|
|
xPortions->nextElement();
|
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// An uncaught exception of type com.sun.star.container.NoSuchElementException
|
|
|
|
// i.e. the line break was a non-clearing one, so we only had 2 portions, not 4 (image, text,
|
|
|
|
// linebreak, text).
|
|
|
|
uno::Reference<beans::XPropertySet> xPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aTextPortionType;
|
|
|
|
xPortion->getPropertyValue("TextPortionType") >>= aTextPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("LineBreak"), aTextPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xLineBreak;
|
|
|
|
xPortion->getPropertyValue("LineBreak") >>= xLineBreak;
|
|
|
|
uno::Reference<beans::XPropertySet> xLineBreakProps(xLineBreak, uno::UNO_QUERY);
|
|
|
|
sal_Int16 eClear{};
|
|
|
|
xLineBreakProps->getPropertyValue("Clear") >>= eClear;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int16>(3), eClear);
|
|
|
|
}
|
|
|
|
|
2022-03-16 15:35:50 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testRelativeWidth)
|
|
|
|
{
|
|
|
|
// Given a document with an 50% wide text frame:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<style::XStyleFamiliesSupplier> xStyleFamiliesSupplier(mxComponent,
|
2022-03-16 15:35:50 +01:00
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XNameAccess> xStyleFamilies
|
|
|
|
= xStyleFamiliesSupplier->getStyleFamilies();
|
|
|
|
uno::Reference<container::XNameAccess> xStyleFamily(xStyleFamilies->getByName("PageStyles"),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xStyle(xStyleFamily->getByName("Standard"), uno::UNO_QUERY);
|
|
|
|
// Body frame width is 6cm (2+2cm margin).
|
2022-05-02 14:25:43 +02:00
|
|
|
xStyle->setPropertyValue("Width", uno::Any(static_cast<sal_Int32>(10000)));
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-03-16 15:35:50 +01:00
|
|
|
uno::Reference<text::XTextContent> xTextFrame(
|
|
|
|
xMSF->createInstance("com.sun.star.text.TextFrame"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xTextFrameProps(xTextFrame, uno::UNO_QUERY);
|
2022-05-02 14:25:43 +02:00
|
|
|
xTextFrameProps->setPropertyValue("RelativeWidth", uno::Any(static_cast<sal_Int16>(50)));
|
2022-03-16 15:35:50 +01:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertTextContent(xCursor, xTextFrame, /*bAbsorb=*/false);
|
|
|
|
// Body frame width is 16cm.
|
2022-05-02 14:25:43 +02:00
|
|
|
xStyle->setPropertyValue("Width", uno::Any(static_cast<sal_Int32>(20000)));
|
2022-03-16 15:35:50 +01:00
|
|
|
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-03-16 15:35:50 +01:00
|
|
|
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-03-16 15:35:50 +01:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: 3.1492in (8cm)
|
|
|
|
// - Actual : 0.0161in (0.04 cm)
|
|
|
|
// i.e. the fallback width value wasn't the expected half of the body frame width, but a smaller
|
|
|
|
// value.
|
|
|
|
assertXPath(pXmlDoc, "//draw:frame", "width", "3.1492in");
|
|
|
|
}
|
|
|
|
|
2022-09-20 09:10:25 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testScaleWidthAndHeight)
|
|
|
|
{
|
|
|
|
// Given a broken document where both IsSyncHeightToWidth and IsSyncWidthToHeight are set to
|
|
|
|
// true:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-09-20 09:10:25 +02:00
|
|
|
uno::Reference<text::XTextContent> xTextFrame(
|
|
|
|
xMSF->createInstance("com.sun.star.text.TextFrame"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xTextFrameProps(xTextFrame, uno::UNO_QUERY);
|
|
|
|
xTextFrameProps->setPropertyValue("Width", uno::Any(static_cast<sal_Int16>(2000)));
|
|
|
|
xTextFrameProps->setPropertyValue("Height", uno::Any(static_cast<sal_Int16>(1000)));
|
|
|
|
xTextFrameProps->setPropertyValue("IsSyncHeightToWidth", uno::Any(true));
|
|
|
|
xTextFrameProps->setPropertyValue("IsSyncWidthToHeight", uno::Any(true));
|
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertTextContent(xCursor, xTextFrame, /*bAbsorb=*/false);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-09-20 09:10:25 +02:00
|
|
|
|
|
|
|
// Then make sure that we still export a non-zero size:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-09-20 09:10:25 +02:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: 0.7874in
|
|
|
|
// - Actual : 0in
|
|
|
|
// i.e. the exported size was 0, not 2000 mm100 in inches.
|
|
|
|
assertXPath(pXmlDoc, "//draw:frame", "width", "0.7874in");
|
|
|
|
}
|
|
|
|
|
2022-04-08 11:32:22 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a content control around one or more text portions:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-04-08 11:32:22 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "test", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
2022-05-02 14:25:43 +02:00
|
|
|
xContentControlProps->setPropertyValue("ShowingPlaceHolder", uno::Any(true));
|
2022-04-08 11:32:22 +02:00
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-04-08 11:32:22 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-04-08 11:32:22 +02:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - XPath '//loext:content-control' number of nodes is incorrect
|
|
|
|
// i.e. the content control was lost on export.
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "showing-place-holder", "true");
|
|
|
|
}
|
|
|
|
|
2022-04-11 11:27:13 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control.fodt");
|
2022-04-11 11:27:13 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-04-11 11:27:13 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: ContentControl
|
|
|
|
// - Actual : Text
|
|
|
|
// i.e. the content control was lost on import.
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<text::XTextRange> xContentControlRange(xContentControl, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XText> xText = xContentControlRange->getText();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xContentEnumAccess(xText, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xContentEnum = xContentEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<text::XTextRange> xContent(xContentEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("test"), xContent->getString());
|
|
|
|
}
|
|
|
|
|
2022-04-27 08:53:13 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCheckboxContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a checkbox content control around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-04-27 08:53:13 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, OUString(u"☐"), /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
2022-05-02 14:25:43 +02:00
|
|
|
xContentControlProps->setPropertyValue("Checkbox", uno::Any(true));
|
|
|
|
xContentControlProps->setPropertyValue("Checked", uno::Any(true));
|
|
|
|
xContentControlProps->setPropertyValue("CheckedState", uno::Any(OUString(u"☒")));
|
|
|
|
xContentControlProps->setPropertyValue("UncheckedState", uno::Any(OUString(u"☐")));
|
2022-04-27 08:53:13 +02:00
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-04-27 08:53:13 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-04-27 08:53:13 +02:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "checkbox", "true");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "checked", "true");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "checked-state", u"☒");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "unchecked-state", u"☐");
|
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCheckboxContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a checkbox content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-checkbox.fodt");
|
2022-04-27 08:53:13 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-04-27 08:53:13 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
bool bCheckbox{};
|
|
|
|
xContentControlProps->getPropertyValue("Checkbox") >>= bCheckbox;
|
|
|
|
// Without the accompanying fix in place, this failed, as the checkbox-related attributes were
|
|
|
|
// ignored on import.
|
|
|
|
CPPUNIT_ASSERT(bCheckbox);
|
|
|
|
bool bChecked{};
|
|
|
|
xContentControlProps->getPropertyValue("Checked") >>= bChecked;
|
|
|
|
CPPUNIT_ASSERT(bChecked);
|
|
|
|
OUString aCheckedState;
|
|
|
|
xContentControlProps->getPropertyValue("CheckedState") >>= aCheckedState;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString(u"☒"), aCheckedState);
|
|
|
|
OUString aUncheckedState;
|
|
|
|
xContentControlProps->getPropertyValue("UncheckedState") >>= aUncheckedState;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString(u"☐"), aUncheckedState);
|
|
|
|
uno::Reference<text::XTextRange> xContentControlRange(xContentControl, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XText> xText = xContentControlRange->getText();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xContentEnumAccess(xText, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xContentEnum = xContentEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<text::XTextRange> xContent(xContentEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString(u"☒"), xContent->getString());
|
|
|
|
}
|
|
|
|
|
2022-05-09 10:14:29 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDropdownContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a dropdown content control around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-09 10:14:29 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "choose an item", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
{
|
2022-11-09 15:50:01 +01:00
|
|
|
xContentControlProps->setPropertyValue("DropDown", uno::Any(true));
|
2022-05-09 10:14:29 +02:00
|
|
|
uno::Sequence<beans::PropertyValues> aListItems = {
|
|
|
|
{
|
|
|
|
comphelper::makePropertyValue("DisplayText", uno::Any(OUString("red"))),
|
|
|
|
comphelper::makePropertyValue("Value", uno::Any(OUString("R"))),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
comphelper::makePropertyValue("DisplayText", uno::Any(OUString("green"))),
|
|
|
|
comphelper::makePropertyValue("Value", uno::Any(OUString("G"))),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
comphelper::makePropertyValue("DisplayText", uno::Any(OUString("blue"))),
|
|
|
|
comphelper::makePropertyValue("Value", uno::Any(OUString("B"))),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
xContentControlProps->setPropertyValue("ListItems", uno::Any(aListItems));
|
|
|
|
}
|
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-05-09 10:14:29 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-11-09 15:50:01 +01:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "dropdown", "true");
|
2022-05-09 10:14:29 +02:00
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: 1
|
|
|
|
// - Actual : 0
|
|
|
|
// - XPath '//loext:content-control/loext:list-item[1]' number of nodes is incorrect
|
|
|
|
// i.e. the list items were lost on export.
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[1]", "display-text", "red");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[1]", "value", "R");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[2]", "display-text", "green");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[2]", "value", "G");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[3]", "display-text", "blue");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control/loext:list-item[3]", "value", "B");
|
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDropdownContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a dropdown content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-dropdown.fodt");
|
2022-05-09 10:14:29 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-09 10:14:29 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
uno::Sequence<beans::PropertyValues> aListItems;
|
|
|
|
xContentControlProps->getPropertyValue("ListItems") >>= aListItems;
|
|
|
|
// Without the accompanying fix in place, this failed with:
|
|
|
|
// - Expected: 3
|
|
|
|
// - Actual : 0
|
|
|
|
// i.e. the list items were lost on import.
|
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(3), aListItems.getLength());
|
|
|
|
comphelper::SequenceAsHashMap aMap0(aListItems[0]);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("red"), aMap0["DisplayText"].get<OUString>());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("R"), aMap0["Value"].get<OUString>());
|
|
|
|
comphelper::SequenceAsHashMap aMap1(aListItems[1]);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("green"), aMap1["DisplayText"].get<OUString>());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("G"), aMap1["Value"].get<OUString>());
|
|
|
|
comphelper::SequenceAsHashMap aMap2(aListItems[2]);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("blue"), aMap2["DisplayText"].get<OUString>());
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("B"), aMap2["Value"].get<OUString>());
|
|
|
|
uno::Reference<text::XTextRange> xContentControlRange(xContentControl, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XText> xText = xContentControlRange->getText();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xContentEnumAccess(xText, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xContentEnum = xContentEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<text::XTextRange> xContent(xContentEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("choose a color"), xContent->getString());
|
|
|
|
}
|
|
|
|
|
2022-05-19 09:12:21 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPictureContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a picture content control around an as-char image:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-19 09:12:21 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextGraphic(
|
|
|
|
xMSF->createInstance("com.sun.star.text.TextGraphicObject"), uno::UNO_QUERY);
|
|
|
|
xTextGraphic->setPropertyValue("AnchorType",
|
|
|
|
uno::Any(text::TextContentAnchorType_AS_CHARACTER));
|
|
|
|
uno::Reference<text::XTextContent> xTextContent(xTextGraphic, uno::UNO_QUERY);
|
|
|
|
xText->insertTextContent(xCursor, xTextContent, false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
xContentControlProps->setPropertyValue("Picture", uno::Any(true));
|
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-05-19 09:12:21 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-05-19 09:12:21 +02:00
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - XPath '//loext:content-control' no attribute 'picture' exist
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "picture", "true");
|
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPictureContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a picture content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-picture.fodt");
|
2022-05-19 09:12:21 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-19 09:12:21 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
bool bPicture{};
|
|
|
|
xContentControlProps->getPropertyValue("Picture") >>= bPicture;
|
|
|
|
// Without the accompanying fix in place, this failed, as the picture attribute was ignored on
|
|
|
|
// import.
|
|
|
|
CPPUNIT_ASSERT(bPicture);
|
|
|
|
}
|
|
|
|
|
2022-05-26 08:25:50 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDateContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a date content control around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-26 08:25:50 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "choose a date", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
xContentControlProps->setPropertyValue("Date", uno::Any(true));
|
|
|
|
xContentControlProps->setPropertyValue("DateFormat", uno::Any(OUString("YYYY-MM-DD")));
|
|
|
|
xContentControlProps->setPropertyValue("DateLanguage", uno::Any(OUString("en-US")));
|
2022-05-27 11:38:42 +02:00
|
|
|
xContentControlProps->setPropertyValue("CurrentDate",
|
|
|
|
uno::Any(OUString("2022-05-25T00:00:00Z")));
|
2022-05-26 08:25:50 +02:00
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-05-26 08:25:50 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-05-26 08:25:50 +02:00
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - XPath '//loext:content-control' no attribute 'date' exist
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "date", "true");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "date-format", "YYYY-MM-DD");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "date-rfc-language-tag", "en-US");
|
2022-05-27 11:38:42 +02:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "current-date", "2022-05-25T00:00:00Z");
|
2022-05-26 08:25:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDateContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a date content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-date.fodt");
|
2022-05-26 08:25:50 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-05-26 08:25:50 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
bool bDate{};
|
|
|
|
xContentControlProps->getPropertyValue("Date") >>= bDate;
|
|
|
|
// Without the accompanying fix in place, this test would have failed, the content control was
|
|
|
|
// imported as a default rich text one.
|
|
|
|
CPPUNIT_ASSERT(bDate);
|
|
|
|
OUString aDateFormat;
|
|
|
|
xContentControlProps->getPropertyValue("DateFormat") >>= aDateFormat;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("YYYY-MM-DD"), aDateFormat);
|
|
|
|
OUString aDateLanguage;
|
|
|
|
xContentControlProps->getPropertyValue("DateLanguage") >>= aDateLanguage;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("en-US"), aDateLanguage);
|
2022-05-27 11:38:42 +02:00
|
|
|
OUString aCurrentDate;
|
|
|
|
xContentControlProps->getPropertyValue("CurrentDate") >>= aCurrentDate;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("2022-05-25T00:00:00Z"), aCurrentDate);
|
2022-05-26 08:25:50 +02:00
|
|
|
}
|
|
|
|
|
2022-07-22 08:35:10 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPlainTextContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a plain text content control around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-07-22 08:35:10 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "test", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
xContentControlProps->setPropertyValue("PlainText", uno::Any(true));
|
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-07-22 08:35:10 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-07-22 08:35:10 +02:00
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - XPath '//loext:content-control' no attribute 'plain-text' exist
|
|
|
|
// i.e. the plain text content control was turned into a rich text one on export.
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "plain-text", "true");
|
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPlainTextContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a plain-text content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-plain-text.fodt");
|
2022-07-22 08:35:10 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-07-22 08:35:10 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
bool bPlainText{};
|
|
|
|
xContentControlProps->getPropertyValue("PlainText") >>= bPlainText;
|
|
|
|
// Without the accompanying fix in place, this test would have failed, the import result was a
|
|
|
|
// rich text content control (not a plain text one).
|
|
|
|
CPPUNIT_ASSERT(bPlainText);
|
|
|
|
}
|
|
|
|
|
2022-09-22 08:36:51 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testComboBoxContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a combo box content control around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-09-22 08:36:51 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "test", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
xContentControlProps->setPropertyValue("ComboBox", uno::Any(true));
|
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-09-22 08:36:51 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-09-22 08:36:51 +02:00
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - XPath '//loext:content-control' no attribute 'combobox' exist
|
|
|
|
// i.e. the combo box content control was turned into a drop-down one on export.
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "combobox", "true");
|
|
|
|
}
|
|
|
|
|
2022-10-18 11:33:20 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testAliasContentControlExport)
|
|
|
|
{
|
|
|
|
// Given a document with a content control and its alias around a text portion:
|
2022-10-24 13:43:14 +02:00
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-10-18 11:33:20 +02:00
|
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
|
|
xText->insertString(xCursor, "test", /*bAbsorb=*/false);
|
|
|
|
xCursor->gotoStart(/*bExpand=*/false);
|
|
|
|
xCursor->gotoEnd(/*bExpand=*/true);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl(
|
|
|
|
xMSF->createInstance("com.sun.star.text.ContentControl"), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
xContentControlProps->setPropertyValue("Alias", uno::Any(OUString("my alias")));
|
|
|
|
xContentControlProps->setPropertyValue("Tag", uno::Any(OUString("my tag")));
|
2022-11-16 21:56:51 -05:00
|
|
|
xContentControlProps->setPropertyValue("Id", uno::Any(static_cast<sal_Int32>(-2147483648)));
|
2022-12-02 10:48:46 -05:00
|
|
|
xContentControlProps->setPropertyValue("TabIndex", uno::Any(sal_uInt32(3)));
|
2022-11-30 13:09:07 -05:00
|
|
|
xContentControlProps->setPropertyValue("Lock", uno::Any(OUString("unlocked")));
|
2022-10-18 11:33:20 +02:00
|
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
|
|
|
|
// When exporting to ODT:
|
2022-11-04 17:31:55 +01:00
|
|
|
save("writer8");
|
2022-10-18 11:33:20 +02:00
|
|
|
|
|
|
|
// Then make sure the expected markup is used:
|
2022-11-04 17:31:55 +01:00
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
2022-10-18 11:33:20 +02:00
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - Expression: prop
|
|
|
|
// - XPath '//loext:content-control' no attribute 'alias' exist
|
|
|
|
// i.e. alias was lost on export.
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "alias", "my alias");
|
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "tag", "my tag");
|
2022-11-16 21:56:51 -05:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "id", "-2147483648");
|
2022-12-02 10:48:46 -05:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "tab-index", "3");
|
2022-11-30 13:09:07 -05:00
|
|
|
assertXPath(pXmlDoc, "//loext:content-control", "lock", "unlocked");
|
2022-10-18 11:33:20 +02:00
|
|
|
}
|
|
|
|
|
2022-09-22 08:36:51 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testComboBoxContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a plain-text content control:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-combo-box.fodt");
|
2022-09-22 08:36:51 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-09-22 08:36:51 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
bool bComboBox{};
|
|
|
|
xContentControlProps->getPropertyValue("ComboBox") >>= bComboBox;
|
|
|
|
// Without the accompanying fix in place, this test would have failed, the import result was a
|
|
|
|
// drop-down content control (not a combo box one).
|
|
|
|
CPPUNIT_ASSERT(bComboBox);
|
|
|
|
}
|
|
|
|
|
2022-10-18 11:33:20 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testAliasContentControlImport)
|
|
|
|
{
|
|
|
|
// Given an ODF document with a content control and its alias/tag:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-alias.fodt");
|
2022-10-18 11:33:20 +02:00
|
|
|
|
|
|
|
// Then make sure that the content control is not lost on import:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
2022-10-18 11:33:20 +02:00
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraphsAccess(xTextDocument->getText(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xParagraphs = xParagraphsAccess->createEnumeration();
|
|
|
|
uno::Reference<container::XEnumerationAccess> xParagraph(xParagraphs->nextElement(),
|
|
|
|
uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> xPortions = xParagraph->createEnumeration();
|
|
|
|
uno::Reference<beans::XPropertySet> xTextPortion(xPortions->nextElement(), uno::UNO_QUERY);
|
|
|
|
OUString aPortionType;
|
|
|
|
xTextPortion->getPropertyValue("TextPortionType") >>= aPortionType;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("ContentControl"), aPortionType);
|
|
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
|
|
xTextPortion->getPropertyValue("ContentControl") >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(xContentControl, uno::UNO_QUERY);
|
|
|
|
OUString aAlias;
|
|
|
|
xContentControlProps->getPropertyValue("Alias") >>= aAlias;
|
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - Expected: my alias
|
|
|
|
// - Actual :
|
|
|
|
// i.e. the alias was lost on import.
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("my alias"), aAlias);
|
|
|
|
OUString aTag;
|
|
|
|
xContentControlProps->getPropertyValue("Tag") >>= aTag;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("my tag"), aTag);
|
2022-11-16 21:56:51 -05:00
|
|
|
sal_Int32 nId = 0;
|
|
|
|
xContentControlProps->getPropertyValue("Id") >>= nId;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(2147483647), nId);
|
2022-12-02 10:48:46 -05:00
|
|
|
sal_uInt32 nTabIndex;
|
|
|
|
xContentControlProps->getPropertyValue("TabIndex") >>= nTabIndex;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_uInt32>(4), nTabIndex);
|
2022-11-30 13:09:07 -05:00
|
|
|
OUString aLock;
|
|
|
|
xContentControlProps->getPropertyValue("Lock") >>= aLock;
|
|
|
|
CPPUNIT_ASSERT_EQUAL(OUString("sdtContentLocked"), aLock);
|
2022-10-18 11:33:20 +02:00
|
|
|
}
|
|
|
|
|
2022-07-26 08:50:03 +02:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDropdownContentControlAutostyleExport)
|
|
|
|
{
|
|
|
|
// Given a document with a dropdown content control, and formatting that forms an autostyle in
|
|
|
|
// ODT:
|
2022-10-24 13:43:14 +02:00
|
|
|
loadFromURL(u"content-control-dropdown.docx");
|
2022-07-26 08:50:03 +02:00
|
|
|
|
|
|
|
// When saving that document to ODT, then make sure no assertion failure happens:
|
2022-10-24 13:43:14 +02:00
|
|
|
uno::Reference<frame::XStorable> xStorable(mxComponent, uno::UNO_QUERY);
|
2022-07-26 08:50:03 +02:00
|
|
|
uno::Sequence<beans::PropertyValue> aStoreProps = comphelper::InitPropertySequence({
|
|
|
|
{ "FilterName", uno::Any(OUString("writer8")) },
|
|
|
|
});
|
|
|
|
// Without the accompanying fix in place, this test would have failed, we had duplicated XML
|
|
|
|
// attributes.
|
2022-11-04 17:31:55 +01:00
|
|
|
xStorable->storeToURL(maTempFile.GetURL(), aStoreProps);
|
2022-07-26 08:50:03 +02:00
|
|
|
}
|
|
|
|
|
2023-01-23 20:10:30 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testScaleWidthRedline)
|
|
|
|
{
|
|
|
|
// Given a document with change tracking enabled, one image is part of a delete redline:
|
|
|
|
loadFromURL(u"scale-width-redline.fodt");
|
|
|
|
dispatchCommand(mxComponent, ".uno:TrackChanges", {});
|
|
|
|
dispatchCommand(mxComponent, ".uno:GoToEndOfLine", {});
|
|
|
|
dispatchCommand(mxComponent, ".uno:EndOfParaSel", {});
|
|
|
|
dispatchCommand(mxComponent, ".uno:Delete", {});
|
|
|
|
|
|
|
|
// When saving to ODT:
|
|
|
|
save("writer8");
|
|
|
|
|
|
|
|
// Then make sure that a non-zero size is written to the output:
|
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - Expected: 6.1728in
|
|
|
|
// - Actual : 0in
|
|
|
|
// i.e. the deleted image had zero size, which is incorrect.
|
|
|
|
assertXPath(pXmlDoc, "//draw:frame[@draw:name='Image45']", "width", "6.1728in");
|
|
|
|
}
|
|
|
|
|
2023-01-28 18:43:51 +09:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testThemeExport)
|
|
|
|
{
|
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
|
|
|
|
uno::Reference<drawing::XDrawPageSupplier> xDrawPageSupplier(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<drawing::XDrawPage> xDrawPage = xDrawPageSupplier->getDrawPage();
|
|
|
|
uno::Reference<beans::XPropertySet> xPageProps(xDrawPage, uno::UNO_QUERY);
|
|
|
|
|
2023-02-25 19:12:23 +09:00
|
|
|
auto pTheme = std::make_shared<model::Theme>("My Theme");
|
2023-05-03 22:58:47 +09:00
|
|
|
auto pColorSet = std::make_shared<model::ColorSet>("My Color Scheme");
|
2023-01-28 18:43:51 +09:00
|
|
|
pColorSet->add(model::ThemeColorType::Dark1, 0x101010);
|
|
|
|
pColorSet->add(model::ThemeColorType::Light1, 0x202020);
|
|
|
|
pColorSet->add(model::ThemeColorType::Dark2, 0x303030);
|
|
|
|
pColorSet->add(model::ThemeColorType::Light2, 0x404040);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent1, 0x505050);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent2, 0x606060);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent3, 0x707070);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent4, 0x808080);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent5, 0x909090);
|
|
|
|
pColorSet->add(model::ThemeColorType::Accent6, 0xa0a0a0);
|
|
|
|
pColorSet->add(model::ThemeColorType::Hyperlink, 0xb0b0b0);
|
|
|
|
pColorSet->add(model::ThemeColorType::FollowedHyperlink, 0xc0c0c0);
|
2023-05-03 22:58:47 +09:00
|
|
|
pTheme->setColorSet(pColorSet);
|
2023-01-28 18:43:51 +09:00
|
|
|
|
2023-02-25 19:12:23 +09:00
|
|
|
uno::Reference<util::XTheme> xTheme = model::theme::createXTheme(pTheme);
|
2023-01-28 18:43:51 +09:00
|
|
|
xPageProps->setPropertyValue("Theme", uno::Any(xTheme));
|
|
|
|
|
|
|
|
// Export to ODT:
|
|
|
|
save("writer8");
|
|
|
|
|
|
|
|
// Check if the 12 colors are written in the XML:
|
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("styles.xml");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color", 12);
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[1]", "name",
|
|
|
|
"dk1");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[1]", "color",
|
|
|
|
"#101010");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[2]", "name",
|
|
|
|
"lt1");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[2]", "color",
|
|
|
|
"#202020");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[12]", "name",
|
|
|
|
"folHlink");
|
|
|
|
assertXPath(pXmlDoc, "//office:styles/loext:theme/loext:color-table/loext:color[12]", "color",
|
|
|
|
"#c0c0c0");
|
|
|
|
}
|
|
|
|
|
2023-02-23 08:21:46 +01:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testFloatingTableExport)
|
|
|
|
{
|
|
|
|
// Given a document with a floating table:
|
|
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
|
|
// Insert a table:
|
|
|
|
uno::Sequence<beans::PropertyValue> aArgs = {
|
|
|
|
comphelper::makePropertyValue("Rows", static_cast<sal_Int32>(1)),
|
|
|
|
comphelper::makePropertyValue("Columns", static_cast<sal_Int32>(1)),
|
|
|
|
};
|
|
|
|
dispatchCommand(mxComponent, ".uno:InsertTable", aArgs);
|
|
|
|
// Select it:
|
|
|
|
dispatchCommand(mxComponent, ".uno:SelectAll", {});
|
|
|
|
// Wrap in a fly:
|
|
|
|
aArgs = {
|
|
|
|
comphelper::makePropertyValue("AnchorType", static_cast<sal_uInt16>(0)),
|
|
|
|
};
|
|
|
|
dispatchCommand(mxComponent, ".uno:InsertFrame", aArgs);
|
|
|
|
// Mark it as a floating table:
|
|
|
|
uno::Reference<text::XTextFramesSupplier> xTextFramesSupplier(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xFrame(
|
|
|
|
xTextFramesSupplier->getTextFrames()->getByName("Frame1"), uno::UNO_QUERY);
|
|
|
|
xFrame->setPropertyValue("IsSplitAllowed", uno::Any(true));
|
|
|
|
|
|
|
|
// When saving to ODT:
|
|
|
|
save("writer8");
|
|
|
|
|
|
|
|
// Then make sure we write a floating table, not a textframe containing a table:
|
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - XPath '//draw:frame' no attribute 'may-break-between-pages' exist
|
|
|
|
// i.e. no floating table was exported.
|
|
|
|
assertXPath(pXmlDoc, "//draw:frame", "may-break-between-pages", "true");
|
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testFloatingTableImport)
|
|
|
|
{
|
|
|
|
// Given a document with a floating table (loext:may-break-between-pages="true"), when importing
|
|
|
|
// that document:
|
|
|
|
loadFromURL(u"floattable.fodt");
|
|
|
|
|
|
|
|
// Then make sure that the matching text frame property is set:
|
|
|
|
uno::Reference<text::XTextFramesSupplier> xTextFramesSupplier(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xFrame(
|
|
|
|
xTextFramesSupplier->getTextFrames()->getByName("Frame1"), uno::UNO_QUERY);
|
|
|
|
bool bIsSplitAllowed = false;
|
|
|
|
// Without the accompanying fix in place, this test would have failed, the property was false.
|
|
|
|
xFrame->getPropertyValue("IsSplitAllowed") >>= bIsSplitAllowed;
|
|
|
|
CPPUNIT_ASSERT(bIsSplitAllowed);
|
|
|
|
}
|
|
|
|
|
2023-03-03 18:04:46 +03:00
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testParagraphScopedTabDistance)
|
|
|
|
{
|
|
|
|
// Given a document with paragraph scoped default tab stop distance (loext:tab-stop-distance="0.5cm")
|
|
|
|
loadFromURL(u"paragraph-tab-stop-distance.fodp");
|
|
|
|
|
|
|
|
uno::Reference<drawing::XDrawPagesSupplier> xDoc(mxComponent, uno::UNO_QUERY);
|
|
|
|
uno::Reference<drawing::XDrawPage> xPage(xDoc->getDrawPages()->getByIndex(0),
|
|
|
|
uno::UNO_QUERY_THROW);
|
|
|
|
|
|
|
|
uno::Reference<beans::XPropertySet> xShape(xPage->getByIndex(0), uno::UNO_QUERY);
|
|
|
|
uno::Reference<text::XText> xText
|
|
|
|
= uno::Reference<text::XTextRange>(xShape, uno::UNO_QUERY_THROW)->getText();
|
|
|
|
|
|
|
|
uno::Reference<container::XEnumerationAccess> paraEnumAccess(xText, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> paraEnum(paraEnumAccess->createEnumeration());
|
|
|
|
uno::Reference<text::XTextRange> xParagraph(paraEnum->nextElement(), uno::UNO_QUERY_THROW);
|
|
|
|
|
|
|
|
uno::Reference<container::XEnumerationAccess> runEnumAccess(xParagraph, uno::UNO_QUERY);
|
|
|
|
uno::Reference<container::XEnumeration> runEnum = runEnumAccess->createEnumeration();
|
|
|
|
uno::Reference<text::XTextRange> xRun(runEnum->nextElement(), uno::UNO_QUERY);
|
|
|
|
uno::Reference<beans::XPropertySet> xPropSet(xRun, uno::UNO_QUERY_THROW);
|
|
|
|
|
|
|
|
// Make sure the tab stop default distance is imported correctly
|
|
|
|
// Without the accompanying fix in place, this test would have failed with:
|
|
|
|
// - Expected: 10000
|
|
|
|
// - Actual : 0
|
|
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int32>(10000),
|
|
|
|
xPropSet->getPropertyValue("ParaTabStopDefaultDistance").get<sal_Int32>());
|
|
|
|
|
|
|
|
// Save the imported file to test the export too
|
|
|
|
save("impress8");
|
|
|
|
|
|
|
|
// Then make sure we write the tab-stop-distance
|
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport("content.xml");
|
|
|
|
assertXPath(pXmlDoc, "//style:style[@style:name='P1']/style:paragraph-properties",
|
|
|
|
"tab-stop-distance", "10cm");
|
|
|
|
|
|
|
|
assertXPath(pXmlDoc, "//text:p[@text:style-name='P1']");
|
|
|
|
}
|
|
|
|
|
2020-08-03 21:03:43 +02:00
|
|
|
CPPUNIT_PLUGIN_IMPLEMENT();
|
|
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|