/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::oox::core; using namespace ::oox::drawingml; using namespace ::com::sun::star; using namespace ::com::sun::star::awt; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::text; using namespace ::com::sun::star::drawing; using namespace ::com::sun::star::document; using namespace ::com::sun::star::container; using namespace ::com::sun::star::presentation; namespace oox::ppt { PPTShape::PPTShape( const oox::ppt::ShapeLocation eShapeLocation, const char* pServiceName ) : Shape( pServiceName ) , meShapeLocation( eShapeLocation ) , mbReferenced( false ) , mbHasNoninheritedShapeProperties( false ) { } PPTShape::~PPTShape() { } static const char* lclDebugSubType( sal_Int32 nType ) { switch (nType) { case XML_ctrTitle : return "ctrTitle"; case XML_title : return "title"; case XML_subTitle : return "subTitle"; case XML_obj : return "obj"; case XML_body : return "body"; case XML_dt : return "dt"; case XML_hdr : return "hdr"; case XML_ftr : return "frt"; case XML_sldNum : return "sldNum"; case XML_sldImg : return "sldImg"; } return "unknown - please extend lclDebugSubType"; } namespace { bool ShapeHasNoVisualPropertiesOnImport(const oox::ppt::PPTShape& rPPTShape) { return !rPPTShape.hasNonInheritedShapeProperties() && !rPPTShape.hasShapeStyleRefs() && !rPPTShape.getTextBody()->hasVisualRunProperties() && !rPPTShape.getTextBody()->hasNoninheritedBodyProperties() && !rPPTShape.getTextBody()->hasListStyleOnImport() && !rPPTShape.getTextBody()->hasParagraphProperties(); } } oox::drawingml::TextListStylePtr PPTShape::getSubTypeTextListStyle( const SlidePersist& rSlidePersist, sal_Int32 nSubType ) { oox::drawingml::TextListStylePtr pTextListStyle; SAL_INFO("oox.ppt", "subtype style: " << lclDebugSubType( nSubType ) ); switch( nSubType ) { case XML_ctrTitle : case XML_title : pTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getTitleTextStyle() : rSlidePersist.getTitleTextStyle(); break; case XML_subTitle : case XML_obj : case XML_body : if ( rSlidePersist.isNotesPage() ) pTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getNotesTextStyle() : rSlidePersist.getNotesTextStyle(); else pTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getBodyTextStyle() : rSlidePersist.getBodyTextStyle(); break; } return pTextListStyle; } bool PPTShape::IsPlaceHolderCandidate(const SlidePersist& rSlidePersist) const { if (meShapeLocation != Slide) return false; if (rSlidePersist.isNotesPage()) return false; auto pTextBody = getTextBody(); if (!pTextBody) return false; auto rParagraphs = pTextBody->getParagraphs(); if (rParagraphs.size() != 1) return false; if (rParagraphs.front()->getRuns().size() != 1) return false; // If the placeholder has a shape other than rectangle, // we have to place it in the slide as a CustomShape. if (!mpCustomShapePropertiesPtr->representsDefaultShape()) return false; return ShapeHasNoVisualPropertiesOnImport(*this); } void PPTShape::addShape( oox::core::XmlFilterBase& rFilterBase, const SlidePersist& rSlidePersist, const oox::drawingml::Theme* pTheme, const Reference< XShapes >& rxShapes, basegfx::B2DHomMatrix& aTransformation, ::oox::drawingml::ShapeIdMap* pShapeMap ) { SAL_INFO("oox.ppt","add shape id: " << msId << " location: " << ((meShapeLocation == Master) ? "master" : ((meShapeLocation == Slide) ? "slide" : ((meShapeLocation == Layout) ? "layout" : "other"))) << " subtype: " << mnSubType << " service: " << msServiceName); // only placeholder from layout are being inserted if ( mnSubType && ( meShapeLocation == Master ) ) return; OUString sServiceName( msServiceName ); if (sServiceName.isEmpty()) return; try { oox::drawingml::TextListStylePtr aMasterTextListStyle; Reference xServiceFact(rFilterBase.getModel(), UNO_QUERY_THROW); bool bClearText = false; if (sServiceName != "com.sun.star.drawing.GraphicObjectShape" && sServiceName != "com.sun.star.drawing.OLE2Shape") { static const OUStringLiteral sOutlinerShapeService(u"com.sun.star.presentation.OutlinerShape"); SAL_INFO("oox.ppt","has master: " << std::hex << rSlidePersist.getMasterPersist().get()); switch (mnSubType) { case XML_ctrTitle : case XML_title : { sServiceName = "com.sun.star.presentation.TitleTextShape"; aMasterTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getTitleTextStyle() : rSlidePersist.getTitleTextStyle(); } break; case XML_subTitle : { if ((meShapeLocation == Master) || (meShapeLocation == Layout)) sServiceName = OUString(); else { sServiceName = "com.sun.star.presentation.SubtitleShape"; aMasterTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getBodyTextStyle() : rSlidePersist.getBodyTextStyle(); } } break; case XML_obj : { sServiceName = sOutlinerShapeService; aMasterTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getBodyTextStyle() : rSlidePersist.getBodyTextStyle(); } break; case XML_body : { if (rSlidePersist.isNotesPage()) { sServiceName = "com.sun.star.presentation.NotesShape"; aMasterTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getNotesTextStyle() : rSlidePersist.getNotesTextStyle(); } else { sServiceName = sOutlinerShapeService; aMasterTextListStyle = rSlidePersist.getMasterPersist() ? rSlidePersist.getMasterPersist()->getBodyTextStyle() : rSlidePersist.getBodyTextStyle(); } } break; case XML_dt : if (IsPlaceHolderCandidate(rSlidePersist)) { TextRunPtr& pTextRun = getTextBody()->getParagraphs().front()->getRuns().front(); oox::drawingml::TextField* pTextField = dynamic_cast(pTextRun.get()); if (pTextField) { OUString aType = pTextField->getType(); if ( aType.startsWith("datetime") ) { SvxDateFormat eDateFormat = drawingml::TextField::getLODateFormat(aType); SvxTimeFormat eTimeFormat = drawingml::TextField::getLOTimeFormat(aType); Reference< XPropertySet > xPropertySet( rSlidePersist.getPage(), UNO_QUERY ); if( eDateFormat != SvxDateFormat::AppDefault || eTimeFormat != SvxTimeFormat::AppDefault ) { // DateTimeFormat property looks for the date in 4 LSBs // and looks for time format in the 4 bits after that sal_Int32 nDateTimeFormat = static_cast(eDateFormat) | static_cast(eTimeFormat) << 4; xPropertySet->setPropertyValue( "IsDateTimeVisible", Any(true) ); xPropertySet->setPropertyValue( "IsDateTimeFixed", Any(false) ); xPropertySet->setPropertyValue( "DateTimeFormat", Any(nDateTimeFormat) ); return; } } } } sServiceName = "com.sun.star.presentation.DateTimeShape"; bClearText = true; break; case XML_hdr : sServiceName = "com.sun.star.presentation.HeaderShape"; bClearText = true; break; case XML_ftr : if (IsPlaceHolderCandidate(rSlidePersist)) { const OUString& rFooterText = getTextBody()->toString(); if( !rFooterText.isEmpty() ) { // if it is possible to get the footer as a property the LO way, // get it and discard the shape Reference< XPropertySet > xPropertySet( rSlidePersist.getPage(), UNO_QUERY ); xPropertySet->setPropertyValue( "IsFooterVisible", Any( true ) ); xPropertySet->setPropertyValue( "FooterText", Any(rFooterText) ); return; } } sServiceName = "com.sun.star.presentation.FooterShape"; bClearText = true; break; case XML_sldNum : if (IsPlaceHolderCandidate(rSlidePersist)) { TextRunPtr& pTextRun = getTextBody()->getParagraphs().front()->getRuns().front(); oox::drawingml::TextField* pTextField = dynamic_cast(pTextRun.get()); if (pTextField && pTextField->getType() == "slidenum") { // if it is possible to get the slidenum placeholder as a property // do that and discard the shape Reference xPropertySet(rSlidePersist.getPage(), UNO_QUERY); xPropertySet->setPropertyValue("IsPageNumberVisible", Any(true)); return; } } sServiceName = "com.sun.star.presentation.SlideNumberShape"; bClearText = true; break; case XML_sldImg : sServiceName = "com.sun.star.presentation.PageShape"; break; case XML_chart : if (meShapeLocation == Layout) sServiceName = sOutlinerShapeService; else sServiceName = "com.sun.star.presentation.ChartShape"; break; case XML_tbl : if (meShapeLocation == Layout) sServiceName = sOutlinerShapeService; else sServiceName = "com.sun.star.presentation.TableShape"; break; case XML_pic : if (meShapeLocation == Layout) sServiceName = sOutlinerShapeService; else sServiceName = "com.sun.star.presentation.GraphicObjectShape"; break; case XML_media : if (meShapeLocation == Layout) sServiceName = sOutlinerShapeService; else sServiceName = "com.sun.star.presentation.MediaShape"; break; default: if (mnSubType && meShapeLocation == Layout) sServiceName = sOutlinerShapeService; break; } } // Since it is not possible to represent custom shaped placeholders in Impress // Need to use service name css.drawing.CustomShape if they have a non default shape. // This workaround has the drawback of them not really being processed as placeholders // so it is only done for slide footers... if ((mnSubType == XML_sldNum || mnSubType == XML_dt || mnSubType == XML_ftr) && meShapeLocation == Slide && !mpCustomShapePropertiesPtr->representsDefaultShape()) { sServiceName = "com.sun.star.drawing.CustomShape"; } SAL_INFO("oox.ppt","shape service: " << sServiceName); if (mnSubType && getSubTypeIndex().has() && meShapeLocation == Layout) { oox::drawingml::ShapePtr pPlaceholder = PPTShape::findPlaceholderByIndex( getSubTypeIndex().get(), rSlidePersist.getShapes()->getChildren(), true ); if (!pPlaceholder) pPlaceholder = PPTShape::findPlaceholder( mnSubType, 0, getSubTypeIndex(), rSlidePersist.getShapes()->getChildren(), true ); if (pPlaceholder) { if (maSize.Width == 0 || maSize.Height == 0) { awt::Size aSize = maSize; if (maSize.Width == 0) aSize.Width = pPlaceholder->getSize().Width; if (maSize.Height == 0) aSize.Height = pPlaceholder->getSize().Height; setSize( aSize ); if (maPosition.X == 0 || maPosition.Y == 0) { awt::Point aPosition = maPosition; if (maPosition.X == 0) aPosition.X = pPlaceholder->getPosition().X; if (maPosition.Y == 0) aPosition.Y = pPlaceholder->getPosition().Y; setPosition( aPosition ); } } } } // use placeholder index if possible if (mnSubType && getSubTypeIndex().has() && rSlidePersist.getMasterPersist()) { oox::drawingml::ShapePtr pPlaceholder = PPTShape::findPlaceholderByIndex(getSubTypeIndex().get(), rSlidePersist.getMasterPersist()->getShapes()->getChildren()); // TODO: Check if this is required for non-notes slides as well... if (rSlidePersist.isNotesPage() && pPlaceholder && pPlaceholder->getSubType() != getSubType()) pPlaceholder.reset(); if (pPlaceholder) { SAL_INFO("oox.ppt","found placeholder with index: " << getSubTypeIndex().get() << " and type: " << lclDebugSubType( mnSubType )); PPTShape* pPPTPlaceholder = dynamic_cast< PPTShape* >( pPlaceholder.get() ); TextListStylePtr pNewTextListStyle = std::make_shared(); if (pPlaceholder->getTextBody()) { pNewTextListStyle->apply( pPlaceholder->getTextBody()->getTextListStyle() ); if (pPlaceholder->getMasterTextListStyle()) pNewTextListStyle->apply( *pPlaceholder->getMasterTextListStyle() ); // SAL_INFO("oox.ppt","placeholder body style"); // pPlaceholder->getTextBody()->getTextListStyle().dump(); // SAL_INFO("oox.ppt","master text list style"); // pPlaceholder->getMasterTextListStyle()->dump(); aMasterTextListStyle = pNewTextListStyle; // SAL_INFO("oox.ppt","combined master text list style"); // aMasterTextListStyle->dump(); } if (pPPTPlaceholder && pPPTPlaceholder->mpPlaceholder) { SAL_INFO("oox.ppt","placeholder has parent placeholder: " << pPPTPlaceholder->mpPlaceholder->getId() << " type: " << lclDebugSubType( pPPTPlaceholder->mpPlaceholder->getSubType() ) << " index: " << pPPTPlaceholder->mpPlaceholder->getSubTypeIndex().get() ); SAL_INFO("oox.ppt","has textbody " << (pPPTPlaceholder->mpPlaceholder->getTextBody() != nullptr) ); TextListStylePtr pPlaceholderStyle = getSubTypeTextListStyle( rSlidePersist, pPPTPlaceholder->mpPlaceholder->getSubType() ); if (pPPTPlaceholder->mpPlaceholder->getTextBody()) pNewTextListStyle->apply( pPPTPlaceholder->mpPlaceholder->getTextBody()->getTextListStyle() ); if (pPlaceholderStyle) { pNewTextListStyle->apply( *pPlaceholderStyle ); //pPlaceholderStyle->dump(); } } } else if (!mpPlaceholder) { aMasterTextListStyle.reset(); } SAL_INFO("oox.ppt","placeholder id: " << (pPlaceholder ? pPlaceholder->getId() : "not found")); } if (!sServiceName.isEmpty()) { if (!aMasterTextListStyle) { bool isOther = !getTextBody() && sServiceName != "com.sun.star.drawing.GroupShape"; TextListStylePtr aSlideStyle = isOther ? rSlidePersist.getOtherTextStyle() : rSlidePersist.getDefaultTextStyle(); // Combine from MasterSlide details as well. if (rSlidePersist.getMasterPersist()) { aMasterTextListStyle = isOther ? rSlidePersist.getMasterPersist()->getOtherTextStyle() : rSlidePersist.getMasterPersist()->getDefaultTextStyle(); if (aSlideStyle) aMasterTextListStyle->apply( *aSlideStyle ); } else { aMasterTextListStyle = aSlideStyle; } } if( aMasterTextListStyle && getTextBody() ) { TextListStylePtr aCombinedTextListStyle = std::make_shared(); aCombinedTextListStyle->apply( *aMasterTextListStyle ); if( mpPlaceholder && mpPlaceholder->getTextBody() ) aCombinedTextListStyle->apply( mpPlaceholder->getTextBody()->getTextListStyle() ); aCombinedTextListStyle->apply( getTextBody()->getTextListStyle() ); setMasterTextListStyle( aCombinedTextListStyle ); } else setMasterTextListStyle( aMasterTextListStyle ); Reference< XShape > xShape( createAndInsert( rFilterBase, sServiceName, pTheme, rxShapes, bClearText, bool(mpPlaceholder), aTransformation, getFillProperties() ) ); // if exists and not duplicated, try to use the title text as slide name to help its re-use on UI if (!rSlidePersist.isMasterPage() && rSlidePersist.getPage().is() && (mnSubType == XML_title || mnSubType == XML_ctrTitle)) { try { sal_Int32 nCount = 1; OUString aTitleText; Reference xText(xShape, UNO_QUERY_THROW); aTitleText = xText->getString(); Reference xDPS(rFilterBase.getModel(), uno::UNO_QUERY_THROW); Reference xDrawPages(xDPS->getDrawPages(), uno::UNO_SET_THROW); sal_uInt32 nMaxPages = xDrawPages->getCount(); // just a magic value but we don't want to drop out slide names which are too long if (aTitleText.getLength() > 63) aTitleText = aTitleText.copy(0, 63); bool bUseTitleAsSlideName = !aTitleText.isEmpty(); // check duplicated title name if (bUseTitleAsSlideName) { for (sal_uInt32 nPage = 0; nPage < nMaxPages; ++nPage) { Reference xDrawPage(xDrawPages->getByIndex(nPage), uno::UNO_QUERY); Reference xNamed(xDrawPage, UNO_QUERY_THROW); OUString sRest; if (xNamed->getName().startsWith(aTitleText, &sRest) && (sRest.isEmpty() || (sRest.startsWith(" (") && sRest.endsWith(")") && sRest.copy(2, sRest.getLength() - 3).toInt32() > 0))) nCount++; } Reference xName(rSlidePersist.getPage(), UNO_QUERY_THROW); xName->setName( aTitleText + (nCount == 1 ? OUString("") : " (" + OUString::number(nCount) + ")")); } } catch (uno::Exception&) { } } // Apply text properties on placeholder text inside this placeholder shape if (meShapeLocation == Slide && mpPlaceholder && getTextBody() && getTextBody()->isEmpty()) { Reference < XText > xText(mxShape, UNO_QUERY); if (xText.is()) { TextCharacterProperties aCharStyleProperties; getTextBody()->ApplyStyleEmpty(rFilterBase, xText, aCharStyleProperties, mpMasterTextListStyle); } } if (pShapeMap) { // bnc#705982 - if optional model id reference is // there, use that to obtain target shape if (!msModelId.isEmpty()) { (*pShapeMap)[ msModelId ] = shared_from_this(); } else if (!msId.isEmpty()) { (*pShapeMap)[ msId ] = shared_from_this(); } } // we will be losing whatever information there is in the footer placeholder on master/layout slides // since they should have the "