Kaydet (Commit) 29dfcc75 authored tarafından Armin Le Grand's avatar Armin Le Grand Kaydeden (comit) Thorsten Behrens

tdf#94088 add import of HTML inline graphics

Related: fdo#63211 for saving. This one adds the import side, plus
fallbacks to use the new FillStyle attributes in HTML im/export in a
SvxBrushItem. Also added graphic import for inline graphics. Comment
markers inserted at places where functionality may be added in the
future when the new FillStyle attributes would be more used in this
content. Unit test checks PageBackground and ParagraphBaground import.

Change-Id: I3f198677db553ad198e0add3162603a4735398f1
Reviewed-on: https://gerrit.libreoffice.org/20129Reviewed-by: 's avatarThorsten Behrens <Thorsten.Behrens@CIB.de>
Tested-by: 's avatarThorsten Behrens <Thorsten.Behrens@CIB.de>
üst 9e36549c
......@@ -3904,7 +3904,6 @@ void SvxBrushItem::PurgeMedium() const
DELETEZ( pImpl->pStream );
}
const GraphicObject* SvxBrushItem::GetGraphicObject(OUString const & referer) const
{
if ( bLoadAgain && !maStrLink.isEmpty() && !pImpl->pGraphicObject )
......@@ -3913,26 +3912,53 @@ const GraphicObject* SvxBrushItem::GetGraphicObject(OUString const & referer) co
if (SvtSecurityOptions().isUntrustedReferer(referer)) {
return nullptr;
}
// tdf#94088 prepare graphic and state
Graphic aGraphic;
bool bGraphicLoaded = false;
// try to create stream directly from given URL
pImpl->pStream = utl::UcbStreamHelper::CreateStream( maStrLink, STREAM_STD_READ );
// tdf#94088 if we have a stream, try to load it directly as graphic
if( pImpl->pStream && !pImpl->pStream->GetError() )
{
Graphic aGraphic;
int nRes;
pImpl->pStream->Seek( STREAM_SEEK_TO_BEGIN );
nRes = GraphicFilter::GetGraphicFilter().
ImportGraphic( aGraphic, maStrLink, *pImpl->pStream,
GRFILTER_FORMAT_DONTKNOW, nullptr, GraphicFilterImportFlags::DontSetLogsizeForJpeg );
if( nRes != GRFILTER_OK )
if (GRFILTER_OK == GraphicFilter::GetGraphicFilter().ImportGraphic( aGraphic, maStrLink, *pImpl->pStream,
GRFILTER_FORMAT_DONTKNOW, nullptr, GraphicFilterImportFlags::DontSetLogsizeForJpeg ))
{
bLoadAgain = false;
bGraphicLoaded = true;
}
else
}
// tdf#94088 if no succeeded, try if the string (which is not epty) contains
// a 'data:' scheme url and try to load that (embedded graphics)
if(!bGraphicLoaded)
{
INetURLObject aGraphicURL( maStrLink );
if( INetProtocol::Data == aGraphicURL.GetProtocol() )
{
pImpl->pGraphicObject = new GraphicObject;
pImpl->pGraphicObject->SetGraphic( aGraphic );
const_cast < SvxBrushItem*> (this)->ApplyGraphicTransparency_Impl();
}
std::unique_ptr<SvMemoryStream> const pStream(aGraphicURL.getData());
if (pStream)
{
if (GRFILTER_OK == GraphicFilter::GetGraphicFilter().ImportGraphic(aGraphic, "", *pStream))
{
bGraphicLoaded = true;
// tdf#94088 delete the no longer needed data scheme URL which
// is potentially pretty // large, containing a base64 encoded copy of the graphic
const_cast< SvxBrushItem* >(this)->maStrLink.clear();
}
}
}
}
// tdf#94088 when we got a graphic, set it
if(bGraphicLoaded && GRAPHIC_NONE != aGraphic.GetType())
{
pImpl->pGraphicObject = new GraphicObject;
pImpl->pGraphicObject->SetGraphic( aGraphic );
const_cast < SvxBrushItem*> (this)->ApplyGraphicTransparency_Impl();
}
else
{
......
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=windows-1252"/>
<title></title>
<meta name="generator" content="LibreOfficeDev 5.1.0.0.alpha1 (Windows)"/>
<meta name="created" content="2015-11-27T14:43:09.574000000"/>
<meta name="changed" content="2015-11-27T14:52:03.499000000"/>
<style type="text/css">
@page { margin: 2cm }
p { margin-bottom: 0.25cm; line-height: 120% }
</style>
</head>
<body lang="de-DE" background="" dir="ltr">
<p style="margin-bottom: 0cm; line-height: 100%; background: url('') repeat scroll ">
Er h&ouml;rte leise Schritte hinter sich. Das bedeutete nichts Gutes.
Wer w&uuml;rde ihm schon folgen, sp&auml;t in der Nacht und dazu noch
in dieser engen Gasse mitten im &uuml;bel beleumundeten Hafenviertel?
Gerade jetzt, wo er das Ding seines Lebens gedreht hatte und mit der
Beute verschwinden wollte! Hatte einer seiner zahllosen Kollegen
dieselbe Idee gehabt, ihn beobachtet und abgewartet, um ihn nun um
die Fr&uuml;chte seiner Arbeit zu erleichtern? Oder geh&ouml;rten die
Schritte hinter ihm zu einem der unz&auml;hligen Gesetzesh&uuml;ter
dieser Stadt, und die st&auml;hlerne Acht um seine Handgelenke w&uuml;rde
gleich zuschnappen? Er konnte die Aufforderung stehen zu bleiben
schon h&ouml;ren. Gehetzt sah er sich um. Pl&ouml;tzlich erblickte er
den schmalen Durchgang. Blitzartig drehte er sich nach rechts und
verschwand zwischen den beiden Geb&auml;uden. Beinahe w&auml;re er
dabei &uuml;ber den umgest&uuml;rzten M&uuml;lleimer gefallen, der
mitten im Weg lag. Er versuchte, sich in der Dunkelheit seinen Weg zu
ertasten und erstarrte: Anscheinend gab es keinen anderen Ausweg aus
diesem kleinen Hof als den Durchgang, durch den er gekommen war. Die
Schritte wurden lauter und lauter, er sah eine dunkle Gestalt um die
Ecke biegen. Fieberhaft irrten seine Augen durch die n&auml;chtliche
Dunkelheit und suchten einen Ausweg. War jetzt wirklich alles vorbei,
waren alle M&uuml;he und alle Vorbereitungen umsonst? Er presste sich
ganz eng an die Wand hinter ihm und hoffte, der Verfolger w&uuml;rde
ihn &uuml;bersehen, als pl&ouml;tzlich neben ihm mit kaum
wahrnehmbarem Quietschen eine T&uuml;r im n&auml;chtlichen Wind hin
und her schwang. K&ouml;nnte dieses der flehentlich herbeigesehnte
Ausweg aus seinem Dilemma sein? Langsam bewegte er sich auf die
offene T&uuml;r zu, immer dicht an die Mauer gepresst. W&uuml;rde
diese T&uuml;r seine Rettung werden?</p>
</body>
</html>
\ No newline at end of file
......@@ -11,6 +11,8 @@
#include <com/sun/star/graphic/XGraphic.hpp>
#include <com/sun/star/graphic/GraphicType.hpp>
#include <com/sun/star/drawing/FillStyle.hpp>
#include <com/sun/star/drawing/BitmapMode.hpp>
#include <vcl/GraphicNativeTransform.hxx>
#include <sfx2/linkmgr.hxx>
......@@ -84,6 +86,46 @@ DECLARE_HTMLIMPORT_TEST(testInlinedImage, "inlined_image.html")
}
}
DECLARE_HTMLIMPORT_TEST(testInlinedImagesPageAndParagraph, "PageAndParagraphFilled.html")
{
SwXTextDocument* pTextDoc = dynamic_cast<SwXTextDocument *>(mxComponent.get());
CPPUNIT_ASSERT(pTextDoc);
// The document contains embedded pictures inlined for PageBackground and
// ParagraphBackground, check for their existance after import
SwDoc* pDoc = pTextDoc->GetDocShell()->GetDoc();
SwEditShell* pEditShell = pDoc->GetEditShell();
CPPUNIT_ASSERT(pEditShell);
// images are not linked, check for zero links
const sfx2::LinkManager& rLinkManager = pEditShell->GetLinkManager();
CPPUNIT_ASSERT_EQUAL(size_t(0), rLinkManager.GetLinks().size());
// get the pageStyle where the PageBackgroundFill is defined. Caution: for
// HTML mode this is *not* called 'Default Style', but 'HTML'. Name is empty
// due to being loaded embedded. BitmapMode is repeat.
uno::Reference<beans::XPropertySet> xPageProperties1(getStyles("PageStyles")->getByName("HTML"), uno::UNO_QUERY);
CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_BITMAP, getProperty<drawing::FillStyle>(xPageProperties1, "FillStyle"));
CPPUNIT_ASSERT_EQUAL(OUString(""), getProperty<OUString>(xPageProperties1, "FillBitmapName"));
CPPUNIT_ASSERT_EQUAL(drawing::BitmapMode_REPEAT, getProperty<drawing::BitmapMode>(xPageProperties1, "FillBitmapMode"));
// we should have one paragraph
const int nParagraphs = getParagraphs();
CPPUNIT_ASSERT_EQUAL(1, nParagraphs);
if(nParagraphs)
{
// get the paragraph
uno::Reference< text::XTextRange > xPara = getParagraph( 0 );
uno::Reference< beans::XPropertySet > xParagraphProperties( xPara, uno::UNO_QUERY);
// check for Bitmap FillStyle, name empty, repeat
CPPUNIT_ASSERT_EQUAL(drawing::FillStyle_BITMAP, getProperty<drawing::FillStyle>(xParagraphProperties, "FillStyle"));
CPPUNIT_ASSERT_EQUAL(OUString(""), getProperty<OUString>(xParagraphProperties, "FillBitmapName"));
CPPUNIT_ASSERT_EQUAL(drawing::BitmapMode_REPEAT, getProperty<drawing::BitmapMode>(xParagraphProperties, "FillBitmapMode"));
}
}
CPPUNIT_PLUGIN_IMPLEMENT();
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
......@@ -1239,7 +1239,17 @@ bool CSS1Expression::GetURL( OUString& rURL ) const
if( aValue.getLength() > 5 )
{
rURL = aValue.copy( 4, aValue.getLength() - 5 );
rURL = comphelper::string::strip(rURL, ' ');
// tdf#94088 original stripped only spaces, but there may also be
// double quotes in CSS style URLs, so be prepared to spaces followed
// by a single quote followed by spaces
const sal_Unicode aSpace(' ');
const sal_Unicode aSingleQuote('\'');
rURL = comphelper::string::strip(rURL, aSpace);
rURL = comphelper::string::strip(rURL, aSingleQuote);
rURL = comphelper::string::strip(rURL, aSpace);
bRet = true;
}
......
......@@ -61,6 +61,7 @@
#include <editeng/lrspitem.hxx>
#include <editeng/protitem.hxx>
#include <editeng/flstitem.hxx>
#include <svx/unobrushitemhelper.hxx>
#include <frmatr.hxx>
#include <charatr.hxx>
......@@ -2870,6 +2871,20 @@ void SwHTMLParser::_SetAttr( bool bChkEnd, bool bBeforeTable,
OSL_ENSURE( false,
"LRSpace ueber mehrere Absaetze gesetzt!" );
// no break (shouldn't reach this point anyway)
// tdf#94088 expand RES_BACKGROUND to the new fill attribute
// definitions in the range [XATTR_FILL_FIRST .. XATTR_FILL_LAST].
// This is the right place in the future if the adapted fill attributes
// may be handled more directly in HTML import to handle them.
case RES_BACKGROUND:
{
const SvxBrushItem& rBrush = static_cast< SvxBrushItem& >(*pAttr->pItem);
SfxItemSet aNewSet(m_pDoc->GetAttrPool(), XATTR_FILL_FIRST, XATTR_FILL_LAST);
setSvxBrushItemAsFillAttributesToTargetSet(rBrush, aNewSet);
m_pDoc->getIDocumentContentOperations().InsertItemSet(*pAttrPam, aNewSet, SetAttrMode::DONTREPLACE);
break;
}
default:
// ggfs. ein Bookmark anspringen
......
......@@ -20,6 +20,9 @@
#include <svl/itemiter.hxx>
#include <svl/whiter.hxx>
// tdf#94088 SdrAllFillAttributesHelper needed
#include <svx/unobrushitemhelper.hxx>
#include "shellio.hxx"
#include "wrt_fn.hxx"
#include "node.hxx"
......@@ -54,6 +57,11 @@ Writer& Out_SfxItemSet( const SwAttrFnTab pTab, Writer& rWrt,
}
const SfxPoolItem* pItem(nullptr);
FnAttrOut pOut;
// tdf#94088 check if any FillAttribute is used [XATTR_FILL_FIRST .. XATTR_FILL_LAST]
// while processing the items
bool bFillItemUsed = false;
if( !bDeep || !pSet->GetParent() )
{
OSL_ENSURE( rSet.Count(), "It has been handled already, right?" );
......@@ -61,8 +69,17 @@ Writer& Out_SfxItemSet( const SwAttrFnTab pTab, Writer& rWrt,
pItem = aIter.GetCurItem();
do {
// pTab only covers POOLATTR_BEGIN..POOLATTR_END.
if( pItem->Which() <= POOLATTR_END && nullptr != ( pOut = pTab[ pItem->Which() - RES_CHRATR_BEGIN] ))
if( pItem->Which() <= POOLATTR_END )
{
if( nullptr != ( pOut = pTab[ pItem->Which() - RES_CHRATR_BEGIN]) )
{
(*pOut)( rWrt, *pItem );
}
}
else if(XATTR_FILLSTYLE == pItem->Which())
{
bFillItemUsed = true;
}
} while( !aIter.IsAtEnd() && nullptr != ( pItem = aIter.NextItem() ) );
}
else
......@@ -76,11 +93,35 @@ Writer& Out_SfxItemSet( const SwAttrFnTab pTab, Writer& rWrt,
*pItem != rPool.GetDefaultItem( nWhich )
|| ( pSet->GetParent() &&
*pItem != pSet->GetParent()->Get( nWhich ))
)) && nullptr != ( pOut = pTab[ nWhich - RES_CHRATR_BEGIN] ))
)))
{
if( nullptr != ( pOut = pTab[ nWhich - RES_CHRATR_BEGIN] ))
{
(*pOut)( rWrt, *pItem );
}
else if(XATTR_FILLSTYLE == pItem->Which())
{
bFillItemUsed = true;
}
}
nWhich = aIter.NextWhich();
}
}
if(bFillItemUsed)
{
// tdf#94088 if used, construct a SvxBrushItem and export it using the
// existing mechanisms.
// This is the right place in the future if the adapted fill attributes
// may be handled more directly in HTML export to handle them.
const SvxBrushItem aSvxBrushItem = getSvxBrushItemFromSourceSet(*pSet, RES_BACKGROUND, bDeep);
if( nullptr != ( pOut = pTab[RES_BACKGROUND - RES_CHRATR_BEGIN] ))
{
(*pOut)( rWrt, aSvxBrushItem );
}
}
return rWrt;
}
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment