mirror of
https://github.com/LibreOffice/core.git
synced 2025-08-02 04:50:01 +00:00

Change-Id: I13da5ca9d07131a340dcf95ca0b980d75ca6e9d0 Reviewed-on: https://gerrit.libreoffice.org/c/core/+/142229 Tested-by: Jenkins Reviewed-by: Xisco Fauli <xiscofauli@libreoffice.org>
879 lines
48 KiB
C++
879 lines
48 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include <test/unoapixml_test.hxx>
|
|
|
|
#include <com/sun/star/beans/XPropertySet.hpp>
|
|
#include <com/sun/star/beans/PropertyValues.hpp>
|
|
#include <com/sun/star/frame/Desktop.hpp>
|
|
#include <com/sun/star/frame/XStorable.hpp>
|
|
#include <com/sun/star/text/XTextDocument.hpp>
|
|
#include <com/sun/star/text/BibliographyDataType.hpp>
|
|
#include <com/sun/star/text/TextContentAnchorType.hpp>
|
|
#include <com/sun/star/style/XStyleFamiliesSupplier.hpp>
|
|
|
|
#include <comphelper/propertysequence.hxx>
|
|
#include <comphelper/propertyvalue.hxx>
|
|
#include <comphelper/sequenceashashmap.hxx>
|
|
#include <unotools/tempfile.hxx>
|
|
#include <unotools/ucbstreamhelper.hxx>
|
|
|
|
using namespace ::com::sun::star;
|
|
|
|
/// Covers xmloff/source/text/ fixes.
|
|
class XmloffStyleTest : public UnoApiXmlTest
|
|
{
|
|
public:
|
|
XmloffStyleTest();
|
|
void registerNamespaces(xmlXPathContextPtr& pXmlXpathCtx) override;
|
|
};
|
|
|
|
XmloffStyleTest::XmloffStyleTest()
|
|
: UnoApiXmlTest("/xmloff/qa/unit/data/")
|
|
{
|
|
}
|
|
|
|
void XmloffStyleTest::registerNamespaces(xmlXPathContextPtr& pXmlXpathCtx)
|
|
{
|
|
XmlTestTools::registerODFNamespaces(pXmlXpathCtx);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testMailMergeInEditeng)
|
|
{
|
|
// Without the accompanying fix in place, this test would have failed, as unexpected
|
|
// <text:database-display> in editeng text aborted the whole import process.
|
|
loadFromURL(u"mail-merge-editeng.odt");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCommentResolved)
|
|
{
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Sequence<beans::PropertyValue> aCommentProps = comphelper::InitPropertySequence({
|
|
{ "Text", uno::Any(OUString("comment")) },
|
|
});
|
|
dispatchCommand(mxComponent, ".uno:InsertAnnotation", aCommentProps);
|
|
uno::Reference<text::XTextDocument> xTextDocument(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);
|
|
uno::Reference<beans::XPropertySet> xField(xPortion->getPropertyValue("TextField"),
|
|
uno::UNO_QUERY);
|
|
xField->setPropertyValue("Resolved", uno::Any(true));
|
|
|
|
saveAndReload("writer8");
|
|
xTextDocument.set(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testBibliographyLocalUrl)
|
|
{
|
|
// 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("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")),
|
|
};
|
|
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");
|
|
// 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:
|
|
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("LocalURL") != aMap.end());
|
|
auto aActual = aMap["LocalURL"].get<OUString>();
|
|
CPPUNIT_ASSERT_EQUAL(OUString("file:///home/me/test.pdf"), aActual);
|
|
}
|
|
|
|
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.
|
|
loadFromURL(u"comment-table-border.fodt");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testParaStyleListLevel)
|
|
{
|
|
// Given a document with style:list-level="...":
|
|
loadFromURL(u"para-style-list-level.fodt");
|
|
|
|
// Then make sure we map that to the paragraph style's numbering level:
|
|
uno::Reference<style::XStyleFamiliesSupplier> xStyleFamiliesSupplier(mxComponent,
|
|
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);
|
|
CPPUNIT_ASSERT_EQUAL(static_cast<sal_Int16>(1), nNumberingLevel);
|
|
|
|
// Test the export as well:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure we save the style's numbering level:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "styles.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContinueNumberingWord)
|
|
{
|
|
// Given a document, which is produced by Word and contains text:continue-numbering="true":
|
|
loadFromURL(u"continue-numbering-word.odt");
|
|
|
|
// Then make sure that the numbering from the 1st para is continued on the 3rd para:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testListId)
|
|
{
|
|
// Given a document with a simple list (no continue-list="..." attribute):
|
|
loadFromURL(u"list-id.fodt");
|
|
|
|
// When storing that document as ODF:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure that unreferenced xml:id="..." attributes are not written:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testClearingBreakExport)
|
|
{
|
|
// Given a document with a clearing break:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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;
|
|
xLineBreakProps->setPropertyValue("Clear", uno::Any(eClear));
|
|
uno::Reference<text::XText> xText = xTextDocument->getText();
|
|
uno::Reference<text::XTextCursor> xCursor = xText->createTextCursor();
|
|
xText->insertTextContent(xCursor, xLineBreak, /*bAbsorb=*/false);
|
|
|
|
// When exporting to ODT:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testClearingBreakImport)
|
|
{
|
|
// Given an ODF document with a clearing break:
|
|
loadFromURL(u"clearing-break.fodt");
|
|
|
|
// Then make sure that the "clear" attribute is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testRelativeWidth)
|
|
{
|
|
// Given a document with an 50% wide text frame:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<style::XStyleFamiliesSupplier> xStyleFamiliesSupplier(mxComponent,
|
|
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).
|
|
xStyle->setPropertyValue("Width", uno::Any(static_cast<sal_Int32>(10000)));
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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("RelativeWidth", uno::Any(static_cast<sal_Int16>(50)));
|
|
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.
|
|
xStyle->setPropertyValue("Width", uno::Any(static_cast<sal_Int32>(20000)));
|
|
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testScaleWidthAndHeight)
|
|
{
|
|
// Given a broken document where both IsSyncHeightToWidth and IsSyncWidthToHeight are set to
|
|
// true:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure that we still export a non-zero size:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContentControlExport)
|
|
{
|
|
// Given a document with a content control around one or more text portions:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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("ShowingPlaceHolder", uno::Any(true));
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
// When exporting to ODT:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testContentControlImport)
|
|
{
|
|
// Given an ODF document with a content control:
|
|
loadFromURL(u"content-control.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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());
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testCheckboxContentControlExport)
|
|
{
|
|
// Given a document with a checkbox content control around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
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"☐")));
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
// When exporting to ODT:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
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:
|
|
loadFromURL(u"content-control-checkbox.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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());
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDropdownContentControlExport)
|
|
{
|
|
// Given a document with a dropdown content control around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
{
|
|
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:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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:
|
|
loadFromURL(u"content-control-dropdown.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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());
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPictureContentControlExport)
|
|
{
|
|
// Given a document with a picture content control around an as-char image:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
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<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:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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:
|
|
loadFromURL(u"content-control-picture.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDateContentControlExport)
|
|
{
|
|
// Given a document with a date content control around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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")));
|
|
xContentControlProps->setPropertyValue("CurrentDate",
|
|
uno::Any(OUString("2022-05-25T00:00:00Z")));
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
// When exporting to ODT:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
assertXPath(pXmlDoc, "//loext:content-control", "current-date", "2022-05-25T00:00:00Z");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDateContentControlImport)
|
|
{
|
|
// Given an ODF document with a date content control:
|
|
loadFromURL(u"content-control-date.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
OUString aCurrentDate;
|
|
xContentControlProps->getPropertyValue("CurrentDate") >>= aCurrentDate;
|
|
CPPUNIT_ASSERT_EQUAL(OUString("2022-05-25T00:00:00Z"), aCurrentDate);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testPlainTextContentControlExport)
|
|
{
|
|
// Given a document with a plain text content control around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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:
|
|
loadFromURL(u"content-control-plain-text.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testComboBoxContentControlExport)
|
|
{
|
|
// Given a document with a combo box content control around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testAliasContentControlExport)
|
|
{
|
|
// Given a document with a content control and its alias around a text portion:
|
|
mxComponent = loadFromDesktop("private:factory/swriter");
|
|
uno::Reference<lang::XMultiServiceFactory> xMSF(mxComponent, uno::UNO_QUERY);
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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")));
|
|
xText->insertTextContent(xCursor, xContentControl, /*bAbsorb=*/true);
|
|
|
|
// When exporting to ODT:
|
|
utl::TempFileNamed aTempFile = save("writer8");
|
|
|
|
// Then make sure the expected markup is used:
|
|
xmlDocUniquePtr pXmlDoc = parseExport(aTempFile.GetURL(), "content.xml");
|
|
// 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");
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testComboBoxContentControlImport)
|
|
{
|
|
// Given an ODF document with a plain-text content control:
|
|
loadFromURL(u"content-control-combo-box.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testAliasContentControlImport)
|
|
{
|
|
// Given an ODF document with a content control and its alias/tag:
|
|
loadFromURL(u"content-control-alias.fodt");
|
|
|
|
// Then make sure that the content control is not lost on import:
|
|
uno::Reference<text::XTextDocument> xTextDocument(mxComponent, uno::UNO_QUERY);
|
|
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);
|
|
}
|
|
|
|
CPPUNIT_TEST_FIXTURE(XmloffStyleTest, testDropdownContentControlAutostyleExport)
|
|
{
|
|
// Given a document with a dropdown content control, and formatting that forms an autostyle in
|
|
// ODT:
|
|
loadFromURL(u"content-control-dropdown.docx");
|
|
|
|
// When saving that document to ODT, then make sure no assertion failure happens:
|
|
uno::Reference<frame::XStorable> xStorable(mxComponent, uno::UNO_QUERY);
|
|
uno::Sequence<beans::PropertyValue> aStoreProps = comphelper::InitPropertySequence({
|
|
{ "FilterName", uno::Any(OUString("writer8")) },
|
|
});
|
|
utl::TempFileNamed aTempFile;
|
|
aTempFile.EnableKillingFile();
|
|
// Without the accompanying fix in place, this test would have failed, we had duplicated XML
|
|
// attributes.
|
|
xStorable->storeToURL(aTempFile.GetURL(), aStoreProps);
|
|
}
|
|
|
|
CPPUNIT_PLUGIN_IMPLEMENT();
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|