/* -*- 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 "XMLImageMapContext.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "xexptran.hxx" #include #include #include "XMLStringBufferImportContext.hxx" #include #include #include using namespace ::com::sun::star; using namespace ::xmloff::token; using ::com::sun::star::beans::XPropertySet; using ::com::sun::star::beans::XPropertySetInfo; using ::com::sun::star::container::XIndexContainer; using ::com::sun::star::lang::XMultiServiceFactory; using ::com::sun::star::uno::Reference; using ::com::sun::star::uno::UNO_QUERY; using ::com::sun::star::xml::sax::XAttributeList; using ::com::sun::star::uno::XInterface; using ::com::sun::star::uno::Any; using ::com::sun::star::drawing::PointSequenceSequence; using ::com::sun::star::document::XEventsSupplier; enum XMLImageMapToken: decltype(XML_TOK_UNKNOWN) { XML_TOK_IMAP_URL, XML_TOK_IMAP_X, XML_TOK_IMAP_Y, XML_TOK_IMAP_CENTER_X, XML_TOK_IMAP_CENTER_Y, XML_TOK_IMAP_WIDTH, XML_TOK_IMAP_HEIGHT, XML_TOK_IMAP_POINTS, XML_TOK_IMAP_VIEWBOX, XML_TOK_IMAP_NOHREF, XML_TOK_IMAP_NAME, XML_TOK_IMAP_RADIUS, XML_TOK_IMAP_TARGET }; static SvXMLTokenMapEntry aImageMapObjectTokenMap[] = { { XML_NAMESPACE_XLINK, XML_HREF, XML_TOK_IMAP_URL }, { XML_NAMESPACE_OFFICE, XML_NAME, XML_TOK_IMAP_NAME }, { XML_NAMESPACE_DRAW, XML_NOHREF, XML_TOK_IMAP_NOHREF }, { XML_NAMESPACE_SVG, XML_X, XML_TOK_IMAP_X }, { XML_NAMESPACE_SVG, XML_Y, XML_TOK_IMAP_Y }, { XML_NAMESPACE_SVG, XML_CX, XML_TOK_IMAP_CENTER_X }, { XML_NAMESPACE_SVG, XML_CY, XML_TOK_IMAP_CENTER_Y }, { XML_NAMESPACE_SVG, XML_WIDTH, XML_TOK_IMAP_WIDTH }, { XML_NAMESPACE_SVG, XML_HEIGHT, XML_TOK_IMAP_HEIGHT }, { XML_NAMESPACE_SVG, XML_R, XML_TOK_IMAP_RADIUS }, { XML_NAMESPACE_SVG, XML_VIEWBOX, XML_TOK_IMAP_VIEWBOX }, { XML_NAMESPACE_DRAW, XML_POINTS, XML_TOK_IMAP_POINTS }, { XML_NAMESPACE_OFFICE, XML_TARGET_FRAME_NAME, XML_TOK_IMAP_TARGET }, XML_TOKEN_MAP_END }; class XMLImageMapObjectContext : public SvXMLImportContext { protected: const OUString sBoundary; const OUString sCenter; const OUString sTitle; const OUString sDescription; const OUString sImageMap; const OUString sIsActive; const OUString sName; const OUString sPolygon; const OUString sRadius; const OUString sTarget; const OUString sURL; Reference xImageMap; /// the image map Reference xMapEntry; /// one map-entry (one area) OUString sUrl; OUString sTargt; OUStringBuffer sDescriptionBuffer; OUStringBuffer sTitleBuffer; OUString sNam; bool bIsActive; bool bValid; public: TYPEINFO_OVERRIDE(); XMLImageMapObjectContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, ::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexContainer> xMap, const sal_Char* pServiceName); void StartElement( const ::com::sun::star::uno::Reference< ::com::sun::star::xml::sax::XAttributeList >& xAttrList ) SAL_OVERRIDE; void EndElement() SAL_OVERRIDE; SvXMLImportContext *CreateChildContext( sal_uInt16 nPrefix, const OUString& rLocalName, const ::com::sun::star::uno::Reference< ::com::sun::star::xml::sax::XAttributeList> & xAttrList ) SAL_OVERRIDE; protected: virtual void ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue); virtual void Prepare( ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet> & rPropertySet); }; TYPEINIT1( XMLImageMapObjectContext, SvXMLImportContext ); XMLImageMapObjectContext::XMLImageMapObjectContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, Reference xMap, const sal_Char* pServiceName) : SvXMLImportContext(rImport, nPrefix, rLocalName), sBoundary("Boundary"), sCenter("Center"), sTitle("Title"), sDescription("Description"), sImageMap("ImageMap"), sIsActive("IsActive"), sName("Name"), sPolygon("Polygon"), sRadius("Radius"), sTarget("Target"), sURL("URL"), xImageMap(xMap), bIsActive(true), bValid(false) { DBG_ASSERT(NULL != pServiceName, "Please supply the image map object service name"); Reference xFactory(GetImport().GetModel(),UNO_QUERY); if( xFactory.is() ) { Reference xIfc = xFactory->createInstance( OUString::createFromAscii(pServiceName)); DBG_ASSERT(xIfc.is(), "can't create image map object!"); if( xIfc.is() ) { Reference xPropertySet( xIfc, UNO_QUERY ); xMapEntry = xPropertySet; } // else: can't create service -> ignore } // else: can't even get factory -> ignore } void XMLImageMapObjectContext::StartElement( const Reference& xAttrList ) { SvXMLTokenMap aMap(aImageMapObjectTokenMap); sal_Int16 nLength = xAttrList->getLength(); for(sal_Int16 nAttr = 0; nAttr < nLength; nAttr++) { OUString sLocalName; sal_uInt16 nPrefix = GetImport().GetNamespaceMap(). GetKeyByAttrName( xAttrList->getNameByIndex(nAttr), &sLocalName ); OUString sValue = xAttrList->getValueByIndex(nAttr); ProcessAttribute( (enum XMLImageMapToken)aMap.Get(nPrefix, sLocalName), sValue); } } void XMLImageMapObjectContext::EndElement() { // only create and insert image map object if validity flag is set // (and we actually have an image map) if ( bValid && xImageMap.is() && xMapEntry.is() ) { // set values Prepare( xMapEntry ); // insert into image map Any aAny; aAny <<= xMapEntry; xImageMap->insertByIndex( xImageMap->getCount(), aAny ); } // else: not valid -> don't create and insert } SvXMLImportContext* XMLImageMapObjectContext::CreateChildContext( sal_uInt16 nPrefix, const OUString& rLocalName, const Reference & xAttrList ) { if ( (XML_NAMESPACE_OFFICE == nPrefix) && IsXMLToken(rLocalName, XML_EVENT_LISTENERS) ) { Reference xEvents( xMapEntry, UNO_QUERY ); return new XMLEventsImportContext( GetImport(), nPrefix, rLocalName, xEvents); } else if ( (XML_NAMESPACE_SVG == nPrefix) && IsXMLToken(rLocalName, XML_TITLE) ) { return new XMLStringBufferImportContext( GetImport(), nPrefix, rLocalName, sTitleBuffer); } else if ( (XML_NAMESPACE_SVG == nPrefix) && IsXMLToken(rLocalName, XML_DESC) ) { return new XMLStringBufferImportContext( GetImport(), nPrefix, rLocalName, sDescriptionBuffer); } else return SvXMLImportContext::CreateChildContext(nPrefix, rLocalName, xAttrList); } void XMLImageMapObjectContext::ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) { switch (eToken) { case XML_TOK_IMAP_URL: sUrl = GetImport().GetAbsoluteReference(rValue); break; case XML_TOK_IMAP_TARGET: sTargt = rValue; break; case XML_TOK_IMAP_NOHREF: bIsActive = ! IsXMLToken(rValue, XML_NOHREF); break; case XML_TOK_IMAP_NAME: sNam = rValue; break; default: // do nothing break; } } void XMLImageMapObjectContext::Prepare( Reference & rPropertySet) { rPropertySet->setPropertyValue( sURL, Any( sUrl ) ); rPropertySet->setPropertyValue( sTitle, Any( sTitleBuffer.makeStringAndClear() ) ); rPropertySet->setPropertyValue( sDescription, Any( sDescriptionBuffer.makeStringAndClear() ) ); rPropertySet->setPropertyValue( sTarget, Any( sTargt ) ); rPropertySet->setPropertyValue( sIsActive, Any( bIsActive ) ); rPropertySet->setPropertyValue( sName, Any( sNam ) ); } class XMLImageMapRectangleContext : public XMLImageMapObjectContext { awt::Rectangle aRectangle; bool bXOK; bool bYOK; bool bWidthOK; bool bHeightOK; public: TYPEINFO_OVERRIDE(); XMLImageMapRectangleContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, ::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexContainer> xMap); virtual ~XMLImageMapRectangleContext(); protected: virtual void ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) SAL_OVERRIDE; virtual void Prepare( ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet> & rPropertySet) SAL_OVERRIDE; }; TYPEINIT1(XMLImageMapRectangleContext, XMLImageMapObjectContext); XMLImageMapRectangleContext::XMLImageMapRectangleContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, Reference xMap) : XMLImageMapObjectContext(rImport, nPrefix, rLocalName, xMap, "com.sun.star.image.ImageMapRectangleObject"), bXOK(false), bYOK(false), bWidthOK(false), bHeightOK(false) { } XMLImageMapRectangleContext::~XMLImageMapRectangleContext() { } void XMLImageMapRectangleContext::ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) { sal_Int32 nTmp; switch (eToken) { case XML_TOK_IMAP_X: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aRectangle.X = nTmp; bXOK = true; } break; case XML_TOK_IMAP_Y: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aRectangle.Y = nTmp; bYOK = true; } break; case XML_TOK_IMAP_WIDTH: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aRectangle.Width = nTmp; bWidthOK = true; } break; case XML_TOK_IMAP_HEIGHT: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aRectangle.Height = nTmp; bHeightOK = true; } break; default: XMLImageMapObjectContext::ProcessAttribute(eToken, rValue); } bValid = bHeightOK && bXOK && bYOK && bWidthOK; } void XMLImageMapRectangleContext::Prepare( Reference & rPropertySet) { Any aAny; aAny <<= aRectangle; rPropertySet->setPropertyValue( sBoundary, aAny ); // common properties handled by super class XMLImageMapObjectContext::Prepare(rPropertySet); } class XMLImageMapPolygonContext : public XMLImageMapObjectContext { OUString sViewBoxString; OUString sPointsString; bool bViewBoxOK; bool bPointsOK; public: TYPEINFO_OVERRIDE(); XMLImageMapPolygonContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, ::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexContainer> xMap); virtual ~XMLImageMapPolygonContext(); protected: virtual void ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) SAL_OVERRIDE; virtual void Prepare( ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet> & rPropertySet) SAL_OVERRIDE; }; TYPEINIT1(XMLImageMapPolygonContext, XMLImageMapObjectContext); XMLImageMapPolygonContext::XMLImageMapPolygonContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, Reference xMap) : XMLImageMapObjectContext(rImport, nPrefix, rLocalName, xMap, "com.sun.star.image.ImageMapPolygonObject"), bViewBoxOK(false), bPointsOK(false) { } XMLImageMapPolygonContext::~XMLImageMapPolygonContext() { } void XMLImageMapPolygonContext::ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) { switch (eToken) { case XML_TOK_IMAP_POINTS: sPointsString = rValue; bPointsOK = true; break; case XML_TOK_IMAP_VIEWBOX: sViewBoxString = rValue; bViewBoxOK = true; break; default: XMLImageMapObjectContext::ProcessAttribute(eToken, rValue); break; } bValid = bViewBoxOK && bPointsOK; } void XMLImageMapPolygonContext::Prepare(Reference & rPropertySet) { // process view box SdXMLImExViewBox aViewBox(sViewBoxString, GetImport().GetMM100UnitConverter()); // get polygon sequence basegfx::B2DPolygon aPolygon; if(basegfx::tools::importFromSvgPoints(aPolygon, sPointsString)) { if(aPolygon.count()) { com::sun::star::drawing::PointSequence aPointSequence; uno::Any aAny; basegfx::tools::B2DPolygonToUnoPointSequence(aPolygon, aPointSequence); aAny <<= aPointSequence; rPropertySet->setPropertyValue(sPolygon, aAny); } } // parent properties XMLImageMapObjectContext::Prepare(rPropertySet); } class XMLImageMapCircleContext : public XMLImageMapObjectContext { awt::Point aCenter; sal_Int32 nRadius; bool bXOK; bool bYOK; bool bRadiusOK; public: TYPEINFO_OVERRIDE(); XMLImageMapCircleContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, ::com::sun::star::uno::Reference< ::com::sun::star::container::XIndexContainer> xMap); virtual ~XMLImageMapCircleContext(); protected: virtual void ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) SAL_OVERRIDE; virtual void Prepare( ::com::sun::star::uno::Reference< ::com::sun::star::beans::XPropertySet> & rPropertySet) SAL_OVERRIDE; }; TYPEINIT1(XMLImageMapCircleContext, XMLImageMapObjectContext); XMLImageMapCircleContext::XMLImageMapCircleContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, Reference xMap) : XMLImageMapObjectContext(rImport, nPrefix, rLocalName, xMap, "com.sun.star.image.ImageMapCircleObject") , nRadius(0) , bXOK(false) , bYOK(false) , bRadiusOK(false) { } XMLImageMapCircleContext::~XMLImageMapCircleContext() { } void XMLImageMapCircleContext::ProcessAttribute( enum XMLImageMapToken eToken, const OUString& rValue) { sal_Int32 nTmp; switch (eToken) { case XML_TOK_IMAP_CENTER_X: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aCenter.X = nTmp; bXOK = true; } break; case XML_TOK_IMAP_CENTER_Y: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { aCenter.Y = nTmp; bYOK = true; } break; case XML_TOK_IMAP_RADIUS: if (GetImport().GetMM100UnitConverter().convertMeasureToCore(nTmp, rValue)) { nRadius = nTmp; bRadiusOK = true; } break; default: XMLImageMapObjectContext::ProcessAttribute(eToken, rValue); } bValid = bRadiusOK && bXOK && bYOK; } void XMLImageMapCircleContext::Prepare( Reference & rPropertySet) { // center (x,y) Any aAny; aAny <<= aCenter; rPropertySet->setPropertyValue( sCenter, aAny ); // radius aAny <<= nRadius; rPropertySet->setPropertyValue( sRadius, aAny ); // common properties handled by super class XMLImageMapObjectContext::Prepare(rPropertySet); } TYPEINIT1(XMLImageMapContext, SvXMLImportContext); XMLImageMapContext::XMLImageMapContext( SvXMLImport& rImport, sal_uInt16 nPrefix, const OUString& rLocalName, Reference & rPropertySet) : SvXMLImportContext(rImport, nPrefix, rLocalName), sImageMap("ImageMap"), xPropertySet(rPropertySet) { try { Reference < XPropertySetInfo > xInfo = xPropertySet->getPropertySetInfo(); if( xInfo.is() && xInfo->hasPropertyByName( sImageMap ) ) xPropertySet->getPropertyValue(sImageMap) >>= xImageMap; } catch(const com::sun::star::uno::Exception& e) { uno::Sequence aSeq(0); rImport.SetError( XMLERROR_FLAG_WARNING | XMLERROR_API, aSeq, e.Message, NULL ); } } XMLImageMapContext::~XMLImageMapContext() { } SvXMLImportContext *XMLImageMapContext::CreateChildContext( sal_uInt16 nPrefix, const OUString& rLocalName, const Reference & xAttrList ) { SvXMLImportContext* pContext = NULL; if ( XML_NAMESPACE_DRAW == nPrefix ) { if ( IsXMLToken(rLocalName, XML_AREA_RECTANGLE) ) { pContext = new XMLImageMapRectangleContext( GetImport(), nPrefix, rLocalName, xImageMap); } else if ( IsXMLToken(rLocalName, XML_AREA_POLYGON) ) { pContext = new XMLImageMapPolygonContext( GetImport(), nPrefix, rLocalName, xImageMap); } else if ( IsXMLToken(rLocalName, XML_AREA_CIRCLE) ) { pContext = new XMLImageMapCircleContext( GetImport(), nPrefix, rLocalName, xImageMap); } } else pContext = SvXMLImportContext::CreateChildContext(nPrefix, rLocalName, xAttrList); return pContext; } void XMLImageMapContext::EndElement() { Reference < XPropertySetInfo > xInfo = xPropertySet->getPropertySetInfo(); if( xInfo.is() && xInfo->hasPropertyByName( sImageMap ) ) xPropertySet->setPropertyValue(sImageMap, uno::makeAny( xImageMap ) ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */