3a9027609e
With OPAQUE option enabled, the rectangle need to be drawn, before drawing text. With CLIPPED option enabled, the text needs to be drawn only inside rectangle. In previous implementation, the rectangle read was skipped and it was not drawn (for OPAQUE). For CLIPPED the rectangle was read but Clip was not applied to text drawing This commit fix that issues, and allow to draw rectangle correctly, with BGColor (background color), and add support for Text Clipping according to [MS-WMF] and [MS-EMF] specification. The fix applied to EMR_EXTTEXTOUTW, EMR_EXTTEXTOUTA and EXTTEXTOUT records Change-Id: I0f6248bb9465e6d5f797cddb53f058afb0815a2d Reviewed-on: https://gerrit.libreoffice.org/c/core/+/116072 Tested-by: Jenkins Reviewed-by: Bartosz Kosiorek <gang65@poczta.onet.pl>
317 lines
12 KiB
C++
317 lines
12 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/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include <sal/config.h>
|
|
|
|
#include <string_view>
|
|
|
|
#include <test/xmltesttools.hxx>
|
|
#include <test/bootstrapfixture.hxx>
|
|
#include <vcl/gdimtf.hxx>
|
|
#include <vcl/wmf.hxx>
|
|
#include <mtftools.hxx>
|
|
|
|
using namespace css;
|
|
|
|
class WmfTest : public test::BootstrapFixture, public XmlTestTools
|
|
{
|
|
OUString maDataUrl;
|
|
|
|
OUString getFullUrl(std::u16string_view sFileName)
|
|
{
|
|
return m_directories.getURLFromSrc(maDataUrl) + sFileName;
|
|
}
|
|
|
|
public:
|
|
WmfTest()
|
|
: BootstrapFixture(true, false)
|
|
, maDataUrl("/emfio/qa/cppunit/wmf/data/")
|
|
{
|
|
}
|
|
|
|
void testNonPlaceableWmf();
|
|
void testSine();
|
|
void testEmfProblem();
|
|
void testEmfLineStyles();
|
|
void testWorldTransformFontSize();
|
|
void testTdf93750();
|
|
void testTdf99402();
|
|
void testTdf39894();
|
|
void testETO_PDY();
|
|
|
|
CPPUNIT_TEST_SUITE(WmfTest);
|
|
CPPUNIT_TEST(testNonPlaceableWmf);
|
|
CPPUNIT_TEST(testSine);
|
|
CPPUNIT_TEST(testEmfProblem);
|
|
CPPUNIT_TEST(testEmfLineStyles);
|
|
CPPUNIT_TEST(testWorldTransformFontSize);
|
|
CPPUNIT_TEST(testTdf93750);
|
|
CPPUNIT_TEST(testTdf99402);
|
|
CPPUNIT_TEST(testTdf39894);
|
|
CPPUNIT_TEST(testETO_PDY);
|
|
CPPUNIT_TEST_SUITE_END();
|
|
};
|
|
|
|
void WmfTest::testNonPlaceableWmf()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"visio_import_source.wmf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
dumper.filterAllActionTypes();
|
|
dumper.filterActionType(MetaActionType::POLYLINE, false);
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[1]", "x", "16798");
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[1]", "y", "1003");
|
|
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[2]", "x", "16798");
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[2]", "y", "7507");
|
|
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[3]", "x", "26090");
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[3]", "y", "7507");
|
|
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[4]", "x", "26090");
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[4]", "y", "1003");
|
|
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[5]", "x", "16798");
|
|
assertXPath(pDoc, "/metafile/polyline[1]/point[5]", "y", "1003");
|
|
}
|
|
|
|
void WmfTest::testSine()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"sine_wave.emf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
dumper.filterAllActionTypes();
|
|
dumper.filterActionType(MetaActionType::ISECTRECTCLIPREGION, false);
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion", 0);
|
|
}
|
|
|
|
void WmfTest::testEmfProblem()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"computer_mail.emf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
dumper.filterAllActionTypes();
|
|
dumper.filterActionType(MetaActionType::ISECTRECTCLIPREGION, false);
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion", 2);
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion[1]", "top", "2125");
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion[1]", "left", "1084");
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion[1]", "bottom", "2927");
|
|
assertXPath(pDoc, "/metafile/sectrectclipregion[1]", "right", "2376");
|
|
}
|
|
|
|
void WmfTest::testEmfLineStyles()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"line_styles.emf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
dumper.filterAllActionTypes();
|
|
dumper.filterActionType(MetaActionType::LINE, false);
|
|
dumper.filterActionType(MetaActionType::LINECOLOR, false);
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/line", 4);
|
|
assertXPath(pDoc, "/metafile/linecolor", 5);
|
|
|
|
assertXPath(pDoc, "/metafile/linecolor[1]", "color", "#ffffff");
|
|
assertXPath(pDoc, "/metafile/linecolor[2]", "color", "#00ff00");
|
|
assertXPath(pDoc, "/metafile/linecolor[3]", "color", "#408080");
|
|
assertXPath(pDoc, "/metafile/linecolor[4]", "color", "#ff0000");
|
|
assertXPath(pDoc, "/metafile/linecolor[5]", "color", "#0000ff");
|
|
|
|
assertXPath(pDoc, "/metafile/line[1]", "style", "dash");
|
|
assertXPath(pDoc, "/metafile/line[1]", "dashlen", "528");
|
|
assertXPath(pDoc, "/metafile/line[1]", "dashcount", "1");
|
|
assertXPath(pDoc, "/metafile/line[1]", "dotlen", "176");
|
|
assertXPath(pDoc, "/metafile/line[1]", "dotcount", "0");
|
|
assertXPath(pDoc, "/metafile/line[1]", "distance", "176");
|
|
assertXPath(pDoc, "/metafile/line[1]", "join", "miter");
|
|
assertXPath(pDoc, "/metafile/line[1]", "cap", "butt");
|
|
|
|
assertXPath(pDoc, "/metafile/line[2]", "style", "dash");
|
|
assertXPath(pDoc, "/metafile/line[2]", "dashlen", "528");
|
|
assertXPath(pDoc, "/metafile/line[2]", "dashcount", "0");
|
|
assertXPath(pDoc, "/metafile/line[2]", "dotlen", "176");
|
|
assertXPath(pDoc, "/metafile/line[2]", "dotcount", "1");
|
|
assertXPath(pDoc, "/metafile/line[2]", "distance", "176");
|
|
assertXPath(pDoc, "/metafile/line[2]", "join", "miter");
|
|
assertXPath(pDoc, "/metafile/line[2]", "cap", "butt");
|
|
|
|
assertXPath(pDoc, "/metafile/line[3]", "style", "dash");
|
|
assertXPath(pDoc, "/metafile/line[3]", "dashlen", "528");
|
|
assertXPath(pDoc, "/metafile/line[3]", "dashcount", "1");
|
|
assertXPath(pDoc, "/metafile/line[3]", "dotlen", "176");
|
|
assertXPath(pDoc, "/metafile/line[3]", "dotcount", "1");
|
|
assertXPath(pDoc, "/metafile/line[3]", "distance", "176");
|
|
assertXPath(pDoc, "/metafile/line[3]", "join", "miter");
|
|
assertXPath(pDoc, "/metafile/line[3]", "cap", "butt");
|
|
|
|
assertXPath(pDoc, "/metafile/line[4]", "style", "dash");
|
|
assertXPath(pDoc, "/metafile/line[4]", "dashlen", "528");
|
|
assertXPath(pDoc, "/metafile/line[4]", "dashcount", "1");
|
|
assertXPath(pDoc, "/metafile/line[4]", "dotlen", "176");
|
|
assertXPath(pDoc, "/metafile/line[4]", "dotcount", "2");
|
|
assertXPath(pDoc, "/metafile/line[4]", "distance", "176");
|
|
assertXPath(pDoc, "/metafile/line[4]", "join", "miter");
|
|
assertXPath(pDoc, "/metafile/line[4]", "cap", "butt");
|
|
};
|
|
|
|
void WmfTest::testWorldTransformFontSize()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"image1.emf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
dumper.filterAllActionTypes();
|
|
dumper.filterActionType(MetaActionType::FONT, false);
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/font", 9);
|
|
|
|
assertXPath(pDoc, "/metafile/font[1]", "color", "#595959");
|
|
assertXPath(pDoc, "/metafile/font[1]", "width", "0");
|
|
assertXPath(pDoc, "/metafile/font[1]", "height", "389");
|
|
assertXPath(pDoc, "/metafile/font[1]", "orientation", "0");
|
|
assertXPath(pDoc, "/metafile/font[1]", "weight", "bold");
|
|
|
|
assertXPath(pDoc, "/metafile/font[3]", "color", "#000000");
|
|
assertXPath(pDoc, "/metafile/font[3]", "width", "0");
|
|
assertXPath(pDoc, "/metafile/font[3]", "height", "389");
|
|
assertXPath(pDoc, "/metafile/font[3]", "orientation", "0");
|
|
assertXPath(pDoc, "/metafile/font[3]", "weight", "bold");
|
|
|
|
// World transform should not affect font size. Rotating text for 90 degrees
|
|
// should not exchange font width and height.
|
|
assertXPath(pDoc, "/metafile/font[4]", "color", "#000000");
|
|
assertXPath(pDoc, "/metafile/font[4]", "width", "0");
|
|
assertXPath(pDoc, "/metafile/font[4]", "height", "530");
|
|
assertXPath(pDoc, "/metafile/font[4]", "orientation", "900");
|
|
assertXPath(pDoc, "/metafile/font[4]", "weight", "normal");
|
|
}
|
|
|
|
void WmfTest::testTdf93750()
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(u"tdf93750.emf"), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
assertXPath(pDoc, "/metafile/push[1]/comment[2]", "datasize", "28");
|
|
assertXPath(pDoc, "/metafile/push[1]/comment[3]", "datasize", "72");
|
|
}
|
|
|
|
void WmfTest::testTdf99402()
|
|
{
|
|
// Symbol font should arrive with RTL_TEXTENCODING_SYMBOL encoding,
|
|
// even if charset is OEM_CHARSET/DEFAULT_CHARSET in WMF
|
|
emfio::LOGFONTW logfontw;
|
|
logfontw.lfHeight = 0;
|
|
logfontw.lfWidth = 0;
|
|
logfontw.lfEscapement = 0;
|
|
logfontw.lfWeight = 0;
|
|
logfontw.lfItalic = 0;
|
|
logfontw.lfUnderline = 0;
|
|
logfontw.lfStrikeOut = 0;
|
|
logfontw.lfCharSet = OEM_CHARSET;
|
|
logfontw.lfPitchAndFamily = FF_ROMAN | DEFAULT_PITCH;
|
|
logfontw.alfFaceName = "Symbol";
|
|
|
|
emfio::WinMtfFontStyle fontStyle(logfontw);
|
|
|
|
CPPUNIT_ASSERT_EQUAL(RTL_TEXTENCODING_SYMBOL, fontStyle.aFont.GetCharSet());
|
|
}
|
|
|
|
void WmfTest::testTdf39894()
|
|
{
|
|
OUString files[] = { "tdf39894.wmf", "tdf39894.emf" };
|
|
for (const auto& file : files)
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(file), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
// The x position of the second text must take into account
|
|
// the previous text's last Dx (previously was ~300)
|
|
auto x = getXPath(pDoc, "/metafile/push[2]/textarray[2]", "x");
|
|
CPPUNIT_ASSERT_MESSAGE(file.toUtf8().getStr(), x.toInt32() > 2700);
|
|
}
|
|
}
|
|
|
|
void WmfTest::testETO_PDY()
|
|
{
|
|
OUString files[] = { "ETO_PDY.wmf", "ETO_PDY.emf" };
|
|
for (const auto& file : files)
|
|
{
|
|
SvFileStream aFileStream(getFullUrl(file), StreamMode::READ);
|
|
GDIMetaFile aGDIMetaFile;
|
|
ReadWindowMetafile(aFileStream, aGDIMetaFile);
|
|
|
|
MetafileXmlDump dumper;
|
|
xmlDocUniquePtr pDoc = dumpAndParse(dumper, aGDIMetaFile);
|
|
|
|
CPPUNIT_ASSERT(pDoc);
|
|
|
|
// The y position of following text
|
|
// must be smaller than that of previous
|
|
auto y1 = getXPath(pDoc, "/metafile/push[2]/textarray[1]", "y");
|
|
auto y2 = getXPath(pDoc, "/metafile/push[2]/textarray[2]", "y");
|
|
auto y3 = getXPath(pDoc, "/metafile/push[2]/textarray[3]", "y");
|
|
CPPUNIT_ASSERT_MESSAGE(file.toUtf8().getStr(), y2.toInt32() < y1.toInt32());
|
|
CPPUNIT_ASSERT_MESSAGE(file.toUtf8().getStr(), y3.toInt32() < y2.toInt32());
|
|
}
|
|
}
|
|
|
|
CPPUNIT_TEST_SUITE_REGISTRATION(WmfTest);
|
|
|
|
CPPUNIT_PLUGIN_IMPLEMENT();
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|