1777 lines
63 KiB
C++
1777 lines
63 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 <com/sun/star/datatransfer/UnsupportedFlavorException.hpp>
|
|
#include <com/sun/star/lang/IllegalArgumentException.hpp>
|
|
#include <com/sun/star/lang/DisposedException.hpp>
|
|
#include <com/sun/star/embed/WrongStateException.hpp>
|
|
#include <com/sun/star/embed/UnreachableStateException.hpp>
|
|
#include <com/sun/star/embed/EmbedStates.hpp>
|
|
#include <com/sun/star/ucb/XSimpleFileAccess.hpp>
|
|
#include <com/sun/star/beans/XPropertySet.hpp>
|
|
#include <com/sun/star/io/TempFile.hpp>
|
|
#include <com/sun/star/io/XTruncate.hpp>
|
|
#include <com/sun/star/io/IOException.hpp>
|
|
#include <com/sun/star/awt/XRequestCallback.hpp>
|
|
|
|
#include "platform.h"
|
|
#include <comphelper/multicontainer2.hxx>
|
|
#include <comphelper/mimeconfighelper.hxx>
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <comphelper/servicehelper.hxx>
|
|
#include <comphelper/windowserrorstring.hxx>
|
|
#include <osl/diagnose.h>
|
|
#include <osl/file.hxx>
|
|
#include <rtl/ref.hxx>
|
|
#include <o3tl/char16_t2wchar_t.hxx>
|
|
#include <o3tl/unit_conversion.hxx>
|
|
#include <systools/win32/comtools.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <vcl/threadex.hxx>
|
|
|
|
#include "graphconvert.hxx"
|
|
#include "olecomponent.hxx"
|
|
#include "olepersist.hxx"
|
|
#include "olewrapclient.hxx"
|
|
#include "advisesink.hxx"
|
|
#include <oleembobj.hxx>
|
|
#include "mtnotification.hxx"
|
|
#include <memory>
|
|
#include <string>
|
|
|
|
using namespace ::com::sun::star;
|
|
using namespace ::comphelper;
|
|
#define MAX_ENUM_ELE 20
|
|
#define FORMATS_NUM 3
|
|
|
|
FORMATETC const pFormatTemplates[FORMATS_NUM] = {
|
|
{ CF_ENHMETAFILE, nullptr, 0, -1, TYMED_ENHMF },
|
|
{ CF_METAFILEPICT, nullptr, 0, -1, TYMED_MFPICT },
|
|
{ CF_BITMAP, nullptr, 0, -1, TYMED_GDI } };
|
|
|
|
|
|
// We have at least one single-threaded apartment (STA) in the process (the VCL Main thread, which
|
|
// is the GUI thread), and a multithreaded apartment (MTA) for most of other threads. OLE objects
|
|
// may be created in either: in interactive mode, this typically happens in the STA; when serving
|
|
// external requests, this may be either in STA (when explicit "OnMainThread" argument is passed to
|
|
// loadComponentFromURL, and the instantiation of the object happens during the load), or in MTA
|
|
// (the thread actually serving the incoming calls).
|
|
//
|
|
// The objects typically can only be used in the apartment where they were instantiated. This means
|
|
// that e.g. a call to IOleObject::Close will fail, if it is performed in a different thread, when
|
|
// it was started in the main thread. And vice versa, opening a document in a handler thread, then
|
|
// trying to interact with the OLE object in GUI would fail.
|
|
//
|
|
// To handle this, several workarounds were implemented in the past; the mentioned "OnMainThread"
|
|
// argument is one of these, allowing open document requests be processed not in the handler threads
|
|
// that received the request, but in the main thread which will then be used for interaction. Also
|
|
// OleComponent::GetExtent was changed to check if the first call to IDataObject::GetData failed
|
|
// with RPC_E_WRONG_THREAD, and then retry in the main thread.
|
|
//
|
|
// But ultimately every call to the OLE object needs such checks. E.g., failing to close the object
|
|
// keeps the server running, effectively leaking resources, until it crashes/freezes after multiple
|
|
// iterations.
|
|
//
|
|
// Currently, OleComponentNative_Impl is implemented using IGlobalInterfaceTable, which allows to
|
|
// register an object in process-global instance of the table from the thread that instantiated the
|
|
// object, and obtain a "cookie" (a number); and then use that cookie from any thread to access that
|
|
// object. The global table will do its magic to provide either the original object (when it is
|
|
// requested from the same apartment as used for its registration), or a "proxy", which will marshal
|
|
// all calls to the proper thread, transparently for the caller. This implementation should obsolete
|
|
// the previous workarounds (in theory).
|
|
//
|
|
// m_pGlobalTable is the reference to the global table.
|
|
// The storage object gets registered in the global table immediately when it's created.
|
|
// But the OLE object itself can't be registered immediately, before it is run: an attempt to call
|
|
// RegisterInterfaceInGlobal with such a newly created OLE object fails with CO_E_OBJNOTCONNECTED.
|
|
// Thus, the initial reference to the OLE object (which at this stage seems to be apartment-neutral)
|
|
// is stored to m_pObj. Only when it is run, it is registered in the global table.
|
|
//
|
|
// Indeed, the implicit change of the thread is a blocking operation, which opens a wonderful new
|
|
// opportunities for shiny deadlocks. Thus, precautions are needed to avoid them.
|
|
//
|
|
// When the OLE object is accessed by m_pObj (should be only in initialization code!), no measures
|
|
// are taken to change locking. But when it is accessed by getObj() - which may return the proxy -
|
|
// the calls are guarded by a SolarMutexReleaser, to allow the other thread do its job.
|
|
//
|
|
// There are at least two other mutexes in play here. One is in OleEmbeddedObject, that holds the
|
|
// OleComponent. The calls to OleComponent's methods are also wrapped there into unlock/lock pairs
|
|
// (see OleEmbeddedObject::changeState). The other is in OleComponent itself. For now, I see no
|
|
// deadlocks caused by that mutex, so no unlock/lock is introduced for that. It may turn out to be
|
|
// required eventually.
|
|
class OleComponentNative_Impl
|
|
{
|
|
public:
|
|
sal::systools::COMReference< IUnknown > m_pObj;
|
|
uno::Sequence< datatransfer::DataFlavor > m_aSupportedGraphFormats;
|
|
|
|
// The getters may return a proxy, that redirects the calls to another thread.
|
|
// Thus, calls to methods of returned objects must be inside solar mutex releaser.
|
|
auto getStorage() const { return getInterface<IStorage>(m_nStorage); }
|
|
auto getObj() const { return m_nOleObject ? getInterface<IUnknown>(m_nOleObject) : m_pObj; }
|
|
template <typename T>
|
|
auto get() const { return getObj().QueryInterface<T>(sal::systools::COM_QUERY); }
|
|
|
|
void registerStorage(IStorage* pStorage) { registerInterface(pStorage, m_nStorage); }
|
|
void registerObj() { registerInterface(m_pObj.get(), m_nOleObject); }
|
|
|
|
bool IsStorageRegistered() const { return m_nStorage != 0; }
|
|
|
|
OleComponentNative_Impl()
|
|
: m_pGlobalTable(CLSID_StdGlobalInterfaceTable, nullptr, CLSCTX_INPROC_SERVER)
|
|
{
|
|
// TODO: Extend format list
|
|
m_aSupportedGraphFormats = {
|
|
|
|
datatransfer::DataFlavor(
|
|
"application/x-openoffice-emf;windows_formatname=\"Image EMF\"",
|
|
"Windows Enhanced Metafile",
|
|
cppu::UnoType<uno::Sequence< sal_Int8 >>::get() ),
|
|
|
|
datatransfer::DataFlavor(
|
|
"application/x-openoffice-wmf;windows_formatname=\"Image WMF\"",
|
|
"Windows Metafile",
|
|
cppu::UnoType<uno::Sequence< sal_Int8 >>::get() ),
|
|
|
|
datatransfer::DataFlavor(
|
|
"application/x-openoffice-bitmap;windows_formatname=\"Bitmap\"",
|
|
"Bitmap",
|
|
cppu::UnoType<uno::Sequence< sal_Int8 >>::get() ),
|
|
|
|
datatransfer::DataFlavor(
|
|
"image/png",
|
|
"PNG",
|
|
cppu::UnoType<uno::Sequence< sal_Int8 >>::get() ),
|
|
|
|
datatransfer::DataFlavor(
|
|
"application/x-openoffice-gdimetafile;windows_formatname=\"GDIMetaFile\"",
|
|
"GDIMetafile",
|
|
cppu::UnoType<uno::Sequence< sal_Int8 >>::get() )
|
|
};
|
|
}
|
|
|
|
~OleComponentNative_Impl()
|
|
{
|
|
if (m_nOleObject)
|
|
m_pGlobalTable->RevokeInterfaceFromGlobal(m_nOleObject);
|
|
if (m_nStorage)
|
|
m_pGlobalTable->RevokeInterfaceFromGlobal(m_nStorage);
|
|
}
|
|
|
|
bool ConvertDataForFlavor( const STGMEDIUM& aMedium,
|
|
const datatransfer::DataFlavor& aFlavor,
|
|
uno::Any& aResult );
|
|
|
|
bool GraphicalFlavor( const datatransfer::DataFlavor& aFlavor );
|
|
|
|
uno::Sequence< datatransfer::DataFlavor > GetFlavorsForAspects( sal_uInt32 nSupportedAspects );
|
|
|
|
sal::systools::COMReference<IStorage> CreateNewStorage(const OUString& url);
|
|
|
|
private:
|
|
sal::systools::COMReference<IGlobalInterfaceTable> m_pGlobalTable;
|
|
DWORD m_nStorage = 0;
|
|
DWORD m_nOleObject = 0;
|
|
|
|
template <typename T> sal::systools::COMReference<T> getInterface(DWORD cookie) const
|
|
{
|
|
sal::systools::COMReference<T> result;
|
|
HRESULT hr = m_pGlobalTable->GetInterfaceFromGlobal(cookie, IID_PPV_ARGS(&result));
|
|
SAL_WARN_IF(FAILED(hr), "embeddedobj.ole",
|
|
"GetInterfaceFromGlobal failed: is cookie " << cookie << " not registered?");
|
|
return result;
|
|
}
|
|
|
|
template <typename T> void registerInterface(T* pInterface, DWORD& cookie)
|
|
{
|
|
if (cookie != 0) // E.g., on subsequent RunObject calls
|
|
return;
|
|
HRESULT hr = m_pGlobalTable->RegisterInterfaceInGlobal(pInterface, __uuidof(T), &cookie);
|
|
SAL_WARN_IF(FAILED(hr), "embeddedobj.ole", "RegisterInterfaceInGlobal failed");
|
|
}
|
|
};
|
|
|
|
static DWORD GetAspectFromFlavor( const datatransfer::DataFlavor& aFlavor )
|
|
{
|
|
if ( aFlavor.MimeType.indexOf( ";Aspect=THUMBNAIL" ) != -1 )
|
|
return DVASPECT_THUMBNAIL;
|
|
else if ( aFlavor.MimeType.indexOf( ";Aspect=ICON" ) != -1 )
|
|
return DVASPECT_ICON;
|
|
else if ( aFlavor.MimeType.indexOf( ";Aspect=DOCPRINT" ) != -1 )
|
|
return DVASPECT_DOCPRINT;
|
|
else
|
|
return DVASPECT_CONTENT;
|
|
}
|
|
|
|
|
|
static OUString GetFlavorSuffixFromAspect( DWORD nAsp )
|
|
{
|
|
OUString aResult;
|
|
|
|
if ( nAsp == DVASPECT_THUMBNAIL )
|
|
aResult = ";Aspect=THUMBNAIL";
|
|
else if ( nAsp == DVASPECT_ICON )
|
|
aResult = ";Aspect=ICON";
|
|
else if ( nAsp == DVASPECT_DOCPRINT )
|
|
aResult = ";Aspect=DOCPRINT";
|
|
|
|
// no suffix for DVASPECT_CONTENT
|
|
|
|
return aResult;
|
|
}
|
|
|
|
|
|
bool OleComponentNative_Impl::ConvertDataForFlavor( const STGMEDIUM& aMedium,
|
|
const datatransfer::DataFlavor& aFlavor,
|
|
uno::Any& aResult )
|
|
{
|
|
bool bAnyIsReady = false;
|
|
|
|
// try to convert data from Medium format to specified Flavor format
|
|
if ( aFlavor.DataType == cppu::UnoType<uno::Sequence< sal_Int8 >>::get() )
|
|
{
|
|
// first the GDI-metafile must be generated
|
|
|
|
std::unique_ptr<sal_Int8[]> pBuf;
|
|
sal_uInt32 nBufSize = 0;
|
|
OUString aFormat;
|
|
|
|
if ( aMedium.tymed == TYMED_MFPICT ) // Win Metafile
|
|
{
|
|
aFormat = "image/x-wmf";
|
|
METAFILEPICT* pMF = static_cast<METAFILEPICT*>(GlobalLock( aMedium.hMetaFilePict ));
|
|
if ( pMF )
|
|
{
|
|
nBufSize = GetMetaFileBitsEx( pMF->hMF, 0, nullptr ) + 22;
|
|
pBuf.reset(new sal_Int8[nBufSize]);
|
|
|
|
|
|
// TODO/LATER: the unit size must be calculated correctly
|
|
*reinterpret_cast<long*>( pBuf.get() ) = 0x9ac6cdd7L;
|
|
*reinterpret_cast<short*>( pBuf.get()+6 ) = SHORT(0);
|
|
*reinterpret_cast<short*>( pBuf.get()+8 ) = SHORT(0);
|
|
*reinterpret_cast<short*>( pBuf.get()+10 ) = static_cast<SHORT>(pMF->xExt);
|
|
*reinterpret_cast<short*>( pBuf.get()+12 ) = static_cast<SHORT>(pMF->yExt);
|
|
*reinterpret_cast<short*>( pBuf.get()+14 ) = USHORT(2540);
|
|
|
|
|
|
if ( nBufSize && nBufSize == GetMetaFileBitsEx( pMF->hMF, nBufSize - 22, pBuf.get() + 22 ) )
|
|
{
|
|
if ( aFlavor.MimeType.match( "application/x-openoffice-wmf;windows_formatname=\"Image WMF\"" ) )
|
|
{
|
|
aResult <<= uno::Sequence< sal_Int8 >( pBuf.get(), nBufSize );
|
|
bAnyIsReady = true;
|
|
}
|
|
}
|
|
|
|
GlobalUnlock( aMedium.hMetaFilePict );
|
|
}
|
|
}
|
|
else if ( aMedium.tymed == TYMED_ENHMF ) // Enh Metafile
|
|
{
|
|
aFormat = "image/x-emf";
|
|
nBufSize = GetEnhMetaFileBits( aMedium.hEnhMetaFile, 0, nullptr );
|
|
pBuf.reset(new sal_Int8[nBufSize]);
|
|
if ( nBufSize && nBufSize == GetEnhMetaFileBits( aMedium.hEnhMetaFile, nBufSize, reinterpret_cast<LPBYTE>(pBuf.get()) ) )
|
|
{
|
|
if ( aFlavor.MimeType.match( "application/x-openoffice-emf;windows_formatname=\"Image EMF\"" ) )
|
|
{
|
|
aResult <<= uno::Sequence< sal_Int8 >( pBuf.get(), nBufSize );
|
|
bAnyIsReady = true;
|
|
}
|
|
}
|
|
}
|
|
else if ( aMedium.tymed == TYMED_GDI ) // Bitmap
|
|
{
|
|
aFormat = "image/x-MS-bmp";
|
|
|
|
// Find out size of buffer: deprecated GetBitmapBits does not have a mode to return
|
|
// required buffer size
|
|
BITMAP aBmp;
|
|
GetObjectW(aMedium.hBitmap, sizeof(aBmp), &aBmp);
|
|
nBufSize = aBmp.bmWidthBytes * aBmp.bmHeight;
|
|
|
|
pBuf.reset(new sal_Int8[nBufSize]);
|
|
if ( nBufSize && nBufSize == sal::static_int_cast< ULONG >( GetBitmapBits( aMedium.hBitmap, nBufSize, pBuf.get() ) ) )
|
|
{
|
|
if ( aFlavor.MimeType.match( "application/x-openoffice-bitmap;windows_formatname=\"Bitmap\"" ) )
|
|
{
|
|
aResult <<= uno::Sequence< sal_Int8 >( pBuf.get(), nBufSize );
|
|
bAnyIsReady = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( pBuf && !bAnyIsReady )
|
|
{
|
|
for (auto const& supportedFormat : m_aSupportedGraphFormats)
|
|
if ( aFlavor.MimeType.match( supportedFormat.MimeType )
|
|
&& aFlavor.DataType == supportedFormat.DataType
|
|
&& aFlavor.DataType == cppu::UnoType<uno::Sequence< sal_Int8 >>::get() )
|
|
{
|
|
bAnyIsReady = ConvertBufferToFormat( pBuf.get(), nBufSize, aFormat, aResult );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return bAnyIsReady;
|
|
}
|
|
|
|
|
|
bool OleComponentNative_Impl::GraphicalFlavor( const datatransfer::DataFlavor& aFlavor )
|
|
{
|
|
// Actually all the required graphical formats must be supported
|
|
for (auto const& supportedFormat : m_aSupportedGraphFormats)
|
|
if ( aFlavor.MimeType.match( supportedFormat.MimeType )
|
|
&& aFlavor.DataType == supportedFormat.DataType )
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
static bool GetClassIDFromSequence_Impl( uno::Sequence< sal_Int8 > const & aSeq, CLSID& aResult )
|
|
{
|
|
if ( aSeq.getLength() == 16 )
|
|
{
|
|
aResult.Data1 = ( ( ( ( ( static_cast<sal_uInt8>(aSeq[0]) << 8 ) + static_cast<sal_uInt8>(aSeq[1]) ) << 8 ) + static_cast<sal_uInt8>(aSeq[2]) ) << 8 ) + static_cast<sal_uInt8>(aSeq[3]);
|
|
aResult.Data2 = ( static_cast<sal_uInt8>(aSeq[4]) << 8 ) + static_cast<sal_uInt8>(aSeq[5]);
|
|
aResult.Data3 = ( static_cast<sal_uInt8>(aSeq[6]) << 8 ) + static_cast<sal_uInt8>(aSeq[7]);
|
|
for( int nInd = 0; nInd < 8; nInd++ )
|
|
aResult.Data4[nInd] = static_cast<sal_uInt8>(aSeq[nInd+8]);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
static OUString WinAccToVcl_Impl( const sal_Unicode* pStr )
|
|
{
|
|
OUString aResult;
|
|
|
|
if( pStr )
|
|
{
|
|
while ( *pStr )
|
|
{
|
|
if ( *pStr == '&' )
|
|
{
|
|
aResult += "~";
|
|
while( *( ++pStr ) == '&' );
|
|
}
|
|
else
|
|
{
|
|
aResult += OUStringChar( *pStr );
|
|
pStr++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return aResult;
|
|
}
|
|
|
|
|
|
OleComponent::OleComponent( const uno::Reference< uno::XComponentContext >& xContext, OleEmbeddedObject* pUnoOleObject )
|
|
: m_pInterfaceContainer( nullptr )
|
|
, m_bDisposed( false )
|
|
, m_bModified( false )
|
|
, m_pNativeImpl( std::make_unique<OleComponentNative_Impl>() )
|
|
, m_pUnoOleObject( pUnoOleObject )
|
|
, m_pOleWrapClientSite( nullptr )
|
|
, m_pImplAdviseSink( nullptr )
|
|
, m_xContext( xContext )
|
|
, m_bOleInitialized( false )
|
|
, m_bWorkaroundActive( false )
|
|
{
|
|
OSL_ENSURE( m_pUnoOleObject, "No owner object is provided!" );
|
|
|
|
HRESULT hr = OleInitialize( nullptr );
|
|
if ( hr == S_OK || hr == S_FALSE )
|
|
m_bOleInitialized = true;
|
|
else
|
|
{
|
|
SAL_WARN("embeddedobj.ole", "OleComponent ctor: OleInitialize() failed with 0x"
|
|
<< OUString::number(static_cast<sal_uInt32>(hr), 16) << ": "
|
|
<< WindowsErrorStringFromHRESULT(hr));
|
|
}
|
|
|
|
m_pOleWrapClientSite = new OleWrapperClientSite( this );
|
|
m_pOleWrapClientSite->AddRef();
|
|
|
|
m_pImplAdviseSink = new OleWrapperAdviseSink( this );
|
|
m_pImplAdviseSink->AddRef();
|
|
|
|
}
|
|
|
|
|
|
OleComponent::~OleComponent()
|
|
{
|
|
OSL_ENSURE( !m_pOleWrapClientSite && !m_pImplAdviseSink && !m_pInterfaceContainer && !m_bOleInitialized,
|
|
"The object was not closed successfully! DISASTER is possible!" );
|
|
|
|
if ( m_pOleWrapClientSite || m_pImplAdviseSink || m_pInterfaceContainer || m_bOleInitialized )
|
|
{
|
|
osl_atomic_increment(&m_refCount);
|
|
try {
|
|
Dispose();
|
|
} catch( const uno::Exception& ) {}
|
|
}
|
|
}
|
|
|
|
void OleComponent::Dispose()
|
|
{
|
|
if ( m_bDisposed )
|
|
return;
|
|
|
|
// Call CloseObject() without m_aMutex locked, since it will call
|
|
// IOleObject::Close(), which can call event listeners, which can run on a
|
|
// different thread.
|
|
CloseObject();
|
|
|
|
osl::MutexGuard aGuard(m_aMutex);
|
|
if ( m_pOleWrapClientSite )
|
|
{
|
|
m_pOleWrapClientSite->disconnectOleComponent();
|
|
m_pOleWrapClientSite->Release();
|
|
m_pOleWrapClientSite = nullptr;
|
|
}
|
|
|
|
if ( m_pImplAdviseSink )
|
|
{
|
|
m_pImplAdviseSink->disconnectOleComponent();
|
|
m_pImplAdviseSink->Release();
|
|
m_pImplAdviseSink = nullptr;
|
|
}
|
|
|
|
if ( m_pInterfaceContainer )
|
|
{
|
|
lang::EventObject aEvent( static_cast< ::cppu::OWeakObject* >( this ) );
|
|
m_pInterfaceContainer->disposeAndClear( aEvent );
|
|
|
|
delete m_pInterfaceContainer;
|
|
m_pInterfaceContainer = nullptr;
|
|
}
|
|
|
|
if ( m_bOleInitialized )
|
|
{
|
|
// since the disposing can happen not only from main thread but also from a clipboard
|
|
// the deinitialization might lead to a disaster, SO7 does not deinitialize OLE at all
|
|
// so currently the same approach is selected as workaround
|
|
// OleUninitialize();
|
|
m_bOleInitialized = false;
|
|
}
|
|
|
|
m_bDisposed = true;
|
|
}
|
|
|
|
|
|
void OleComponent::disconnectEmbeddedObject()
|
|
{
|
|
// must not be called from destructor of UNO OLE object!!!
|
|
osl::MutexGuard aGuard( m_aMutex );
|
|
m_pUnoOleObject = nullptr;
|
|
}
|
|
|
|
|
|
OUString OleComponent::getTempURL() const
|
|
{
|
|
OSL_ENSURE( m_pUnoOleObject, "Unexpected object absence!" );
|
|
if ( m_pUnoOleObject )
|
|
return m_pUnoOleObject->CreateTempURLEmpty_Impl();
|
|
else
|
|
return GetNewTempFileURL_Impl(m_xContext);
|
|
}
|
|
|
|
|
|
sal::systools::COMReference<IStorage> OleComponentNative_Impl::CreateNewStorage(const OUString& url)
|
|
{
|
|
if (IsStorageRegistered())
|
|
throw io::IOException(); // TODO:the object is already initialized
|
|
// TODO: in future a global memory could be used instead of file.
|
|
|
|
// write the stream to the temporary file
|
|
if (url.isEmpty())
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
// open an IStorage based on the temporary file
|
|
OUString aTempFilePath;
|
|
if (osl::FileBase::getSystemPathFromFileURL(url, aTempFilePath) != osl::FileBase::E_None)
|
|
throw uno::RuntimeException(); // TODO: something dangerous happened
|
|
|
|
sal::systools::COMReference<IStorage> pStorage;
|
|
HRESULT hr = StgCreateDocfile( o3tl::toW(aTempFilePath.getStr()), STGM_CREATE | STGM_READWRITE | STGM_TRANSACTED | STGM_DELETEONRELEASE, 0, &pStorage );
|
|
if (FAILED(hr) || !pStorage)
|
|
throw io::IOException(); // TODO: transport error code?
|
|
registerStorage(pStorage);
|
|
return pStorage;
|
|
}
|
|
|
|
|
|
uno::Sequence< datatransfer::DataFlavor > OleComponentNative_Impl::GetFlavorsForAspects( sal_uInt32 nSupportedAspects )
|
|
{
|
|
uno::Sequence< datatransfer::DataFlavor > aResult;
|
|
for ( sal_uInt32 nAsp = 1; nAsp <= 8; nAsp *= 2 )
|
|
if ( ( nSupportedAspects & nAsp ) == nAsp )
|
|
{
|
|
OUString aAspectSuffix = GetFlavorSuffixFromAspect( nAsp );
|
|
|
|
sal_Int32 nLength = aResult.getLength();
|
|
aResult.realloc( nLength + m_aSupportedGraphFormats.getLength() );
|
|
auto pResult = aResult.getArray();
|
|
|
|
for ( sal_Int32 nInd = 0; nInd < m_aSupportedGraphFormats.getLength(); nInd++ )
|
|
{
|
|
pResult[nLength + nInd].MimeType = m_aSupportedGraphFormats[nInd].MimeType + aAspectSuffix;
|
|
pResult[nLength + nInd].HumanPresentableName = m_aSupportedGraphFormats[nInd].HumanPresentableName;
|
|
pResult[nLength + nInd].DataType = m_aSupportedGraphFormats[nInd].DataType;
|
|
}
|
|
}
|
|
|
|
return aResult;
|
|
}
|
|
|
|
|
|
void OleComponent::RetrieveObjectDataFlavors_Impl()
|
|
{
|
|
if (!m_pNativeImpl->m_pObj)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
if ( !m_aDataFlavors.getLength() )
|
|
{
|
|
if (auto pDataObject = m_pNativeImpl->get<IDataObject>())
|
|
{
|
|
HRESULT hr;
|
|
sal::systools::COMReference< IEnumFORMATETC > pFormatEnum;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pDataObject->EnumFormatEtc(DATADIR_GET, &pFormatEnum);
|
|
}
|
|
if ( SUCCEEDED( hr ) && pFormatEnum )
|
|
{
|
|
FORMATETC pElem[ MAX_ENUM_ELE ];
|
|
ULONG nNum = 0;
|
|
|
|
// if it is possible to retrieve at least one supported graphical format for an aspect
|
|
// this format can be converted to other supported formats
|
|
sal_uInt32 nSupportedAspects = 0;
|
|
do
|
|
{
|
|
HRESULT hr2 = pFormatEnum->Next( MAX_ENUM_ELE, pElem, &nNum );
|
|
if( hr2 == S_OK || hr2 == S_FALSE )
|
|
{
|
|
for( sal_uInt32 nInd = 0; nInd < FORMATS_NUM; nInd++ )
|
|
{
|
|
if ( pElem[nInd].cfFormat == pFormatTemplates[nInd].cfFormat
|
|
&& pElem[nInd].tymed == pFormatTemplates[nInd].tymed )
|
|
nSupportedAspects |= pElem[nInd].dwAspect;
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
while( nNum == MAX_ENUM_ELE );
|
|
|
|
m_aDataFlavors = m_pNativeImpl->GetFlavorsForAspects( nSupportedAspects );
|
|
}
|
|
}
|
|
|
|
if ( !m_aDataFlavors.getLength() )
|
|
{
|
|
// TODO:
|
|
// for any reason the object could not provide this information
|
|
// try to get access to the cached representation
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void OleComponent::InitializeObject_Impl()
|
|
// There will be no static objects!
|
|
{
|
|
if ( !m_pNativeImpl->m_pObj )
|
|
throw embed::WrongStateException();
|
|
|
|
// the linked object will be detected here
|
|
OSL_ENSURE( m_pUnoOleObject, "Unexpected object absence!" );
|
|
if ( m_pUnoOleObject )
|
|
m_pUnoOleObject->SetObjectIsLink_Impl( m_pNativeImpl->m_pObj.QueryInterface<IOleLink>(sal::systools::COM_QUERY).is() );
|
|
|
|
auto pViewObject2(m_pNativeImpl->m_pObj.QueryInterface<IViewObject2>(sal::systools::COM_QUERY));
|
|
if (!pViewObject2)
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
// remove all the caches
|
|
if ( sal::systools::COMReference< IOleCache > pIOleCache{ m_pNativeImpl->m_pObj, sal::systools::COM_QUERY } )
|
|
{
|
|
IEnumSTATDATA* pEnumSD = nullptr;
|
|
HRESULT hr2 = pIOleCache->EnumCache( &pEnumSD );
|
|
|
|
if ( SUCCEEDED( hr2 ) && pEnumSD )
|
|
{
|
|
pEnumSD->Reset();
|
|
STATDATA aSD;
|
|
DWORD nNum;
|
|
while( SUCCEEDED( pEnumSD->Next( 1, &aSD, &nNum ) ) && nNum == 1 )
|
|
hr2 = pIOleCache->Uncache( aSD.dwConnection );
|
|
}
|
|
|
|
// No IDataObject implementation, caching must be used instead
|
|
DWORD nConn;
|
|
FORMATETC aFormat = { 0, nullptr, DVASPECT_CONTENT, -1, TYMED_MFPICT };
|
|
hr2 = pIOleCache->Cache( &aFormat, ADVFCACHE_ONSAVE, &nConn );
|
|
}
|
|
|
|
auto pOleObject(m_pNativeImpl->m_pObj.QueryInterface<IOleObject>(sal::systools::COM_QUERY));
|
|
if (!pOleObject)
|
|
throw uno::RuntimeException(); // Static objects are not supported, they should be inserted as graphics
|
|
|
|
DWORD nOLEMiscFlags(0);
|
|
pOleObject->GetMiscStatus(DVASPECT_CONTENT, reinterpret_cast<DWORD*>(&nOLEMiscFlags));
|
|
// TODO: use other misc flags also
|
|
// the object should have drawable aspect even in case it supports only iconic representation
|
|
// if ( nOLEMiscFlags & OLEMISC_ONLYICONIC )
|
|
|
|
pOleObject->SetClientSite(m_pOleWrapClientSite);
|
|
|
|
// the only need in this registration is workaround for close notification
|
|
DWORD nAdvConn(0);
|
|
pOleObject->Advise(m_pImplAdviseSink, reinterpret_cast<DWORD*>(&nAdvConn));
|
|
pViewObject2->SetAdvise(DVASPECT_CONTENT, 0, m_pImplAdviseSink);
|
|
|
|
OleSetContainedObject(pOleObject, TRUE);
|
|
}
|
|
|
|
namespace
|
|
{
|
|
HRESULT OleLoadSeh(LPSTORAGE pIStorage, IUnknown** ppObj)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
// tdf#119039: there is a nasty bug in OleLoad, that may call an unpaired
|
|
// IUnknown::Release on pIStorage on STG_E_FILENOTFOUND: see
|
|
// https://developercommunity.visualstudio.com/t/10144795
|
|
// Workaround it here to avoid crash in smart COM pointer destructor that
|
|
// would try to release already released object. Since we don't know if
|
|
// the bug appears each time STG_E_FILENOTFOUND is returned, this might
|
|
// potentially leak the storage object.
|
|
if (pIStorage)
|
|
pIStorage->AddRef();
|
|
|
|
__try {
|
|
hr = OleLoad(pIStorage, IID_IUnknown, nullptr, IID_PPV_ARGS_Helper(ppObj));
|
|
} __except( EXCEPTION_EXECUTE_HANDLER ) {
|
|
hr = E_FAIL;
|
|
}
|
|
if (pIStorage && hr != STG_E_FILENOTFOUND)
|
|
pIStorage->Release();
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
void OleComponent::LoadEmbeddedObject( const OUString& aTempURL )
|
|
{
|
|
if ( !aTempURL.getLength() )
|
|
throw lang::IllegalArgumentException(); // TODO
|
|
|
|
if (m_pNativeImpl->IsStorageRegistered())
|
|
throw io::IOException(); // TODO the object is already initialized or wrong initialization is done
|
|
|
|
// open an IStorage based on the temporary file
|
|
OUString aFilePath;
|
|
if (osl::FileBase::getSystemPathFromFileURL(aTempURL, aFilePath) != ::osl::FileBase::E_None)
|
|
throw uno::RuntimeException(); // TODO: something dangerous happened
|
|
|
|
sal::systools::COMReference<IStorage> pStorage;
|
|
HRESULT hr = StgOpenStorage(o3tl::toW(aFilePath.getStr()), nullptr,
|
|
STGM_READWRITE | STGM_TRANSACTED, // | STGM_DELETEONRELEASE,
|
|
nullptr, 0, &pStorage);
|
|
if (FAILED(hr) || !pStorage)
|
|
throw io::IOException(); // TODO: transport error code?
|
|
|
|
m_pNativeImpl->registerStorage(pStorage);
|
|
|
|
hr = OleLoadSeh(pStorage, &m_pNativeImpl->m_pObj);
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException();
|
|
|
|
InitializeObject_Impl();
|
|
}
|
|
|
|
|
|
void OleComponent::CreateObjectFromClipboard()
|
|
{
|
|
auto pStorage(m_pNativeImpl->CreateNewStorage(getTempURL()));
|
|
if (!pStorage)
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
IDataObject * pDO = nullptr;
|
|
HRESULT hr = OleGetClipboard( &pDO );
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException();
|
|
|
|
hr = OleQueryCreateFromData(pDO);
|
|
if (S_OK == hr)
|
|
{
|
|
hr = OleCreateFromData( pDO,
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW, // OLERENDER_FORMAT
|
|
nullptr, // &aFormat,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException();
|
|
}
|
|
else
|
|
{
|
|
// Static objects are not supported
|
|
pDO->Release();
|
|
}
|
|
|
|
InitializeObject_Impl();
|
|
}
|
|
|
|
|
|
void OleComponent::CreateNewEmbeddedObject( const uno::Sequence< sal_Int8 >& aSeqCLSID )
|
|
{
|
|
CLSID aClsID;
|
|
|
|
if ( !GetClassIDFromSequence_Impl( aSeqCLSID, aClsID ) )
|
|
throw lang::IllegalArgumentException(); // TODO
|
|
|
|
auto pStorage(m_pNativeImpl->CreateNewStorage(getTempURL()));
|
|
if (!pStorage)
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
// FORMATETC aFormat = { CF_METAFILEPICT, NULL, nAspect, -1, TYMED_MFPICT }; // for OLE..._DRAW should be NULL
|
|
|
|
HRESULT hr = OleCreate( aClsID,
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW, // OLERENDER_FORMAT
|
|
nullptr, // &aFormat,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
InitializeObject_Impl();
|
|
|
|
// TODO: getExtent???
|
|
}
|
|
|
|
|
|
void OleComponent::CreateObjectFromData( const uno::Reference< datatransfer::XTransferable >& )
|
|
// Static objects are not supported, they should be inserted as graphics
|
|
{
|
|
// TODO: May be this call is useless since there are no static objects
|
|
// and nonstatic objects will be created based on OLEstorage ( stream ).
|
|
// ???
|
|
|
|
// OleQueryCreateFromData...
|
|
}
|
|
|
|
|
|
void OleComponent::CreateObjectFromFile( const OUString& aFileURL )
|
|
{
|
|
auto pStorage(m_pNativeImpl->CreateNewStorage(getTempURL()));
|
|
if (!pStorage)
|
|
throw uno::RuntimeException(); // TODO:
|
|
|
|
OUString aFilePath;
|
|
if ( ::osl::FileBase::getSystemPathFromFileURL( aFileURL, aFilePath ) != ::osl::FileBase::E_None )
|
|
throw uno::RuntimeException(); // TODO: something dangerous happened
|
|
|
|
HRESULT hr = OleCreateFromFile( CLSID_NULL,
|
|
o3tl::toW(aFilePath.getStr()),
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW, // OLERENDER_FORMAT
|
|
nullptr,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
InitializeObject_Impl();
|
|
}
|
|
|
|
|
|
void OleComponent::CreateLinkFromFile( const OUString& aFileURL )
|
|
{
|
|
auto pStorage(m_pNativeImpl->CreateNewStorage(getTempURL()));
|
|
if (!pStorage)
|
|
throw uno::RuntimeException(); // TODO:
|
|
|
|
OUString aFilePath;
|
|
if ( ::osl::FileBase::getSystemPathFromFileURL( aFileURL, aFilePath ) != ::osl::FileBase::E_None )
|
|
throw uno::RuntimeException(); // TODO: something dangerous happened
|
|
|
|
HRESULT hr = OleCreateLinkToFile( o3tl::toW(aFilePath.getStr()),
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW, // OLERENDER_FORMAT
|
|
nullptr,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
if (FAILED(hr))
|
|
throw uno::RuntimeException(); // TODO
|
|
|
|
InitializeObject_Impl();
|
|
}
|
|
|
|
|
|
void OleComponent::InitEmbeddedCopyOfLink( rtl::Reference<OleComponent> const & pOleLinkComponent )
|
|
{
|
|
if (!pOleLinkComponent)
|
|
throw lang::IllegalArgumentException(); // TODO
|
|
|
|
auto pOleLinkComponentObj(pOleLinkComponent->m_pNativeImpl->getObj());
|
|
if (!pOleLinkComponentObj)
|
|
throw lang::IllegalArgumentException();
|
|
|
|
// the object must be already disconnected from the temporary URL
|
|
auto pStorage(m_pNativeImpl->CreateNewStorage(getTempURL()));
|
|
|
|
SolarMutexReleaser releaser;
|
|
|
|
auto pDataObject(pOleLinkComponentObj.QueryInterface<IDataObject>(sal::systools::COM_QUERY));
|
|
if ( pDataObject && SUCCEEDED( OleQueryCreateFromData( pDataObject ) ) )
|
|
{
|
|
if (!pStorage)
|
|
throw uno::RuntimeException(); // TODO:
|
|
|
|
OleCreateFromData( pDataObject,
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW,
|
|
nullptr,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
}
|
|
|
|
if ( !m_pNativeImpl->m_pObj )
|
|
{
|
|
auto pOleLink(pOleLinkComponentObj.QueryInterface<IOleLink>(sal::systools::COM_QUERY));
|
|
if ( !pOleLink )
|
|
throw io::IOException(); // TODO: the object doesn't support IOleLink
|
|
|
|
sal::systools::COMReference< IMoniker > pMoniker;
|
|
HRESULT hr = pOleLink->GetSourceMoniker( &pMoniker );
|
|
if ( FAILED( hr ) || !pMoniker )
|
|
throw io::IOException(); // TODO: can not retrieve moniker
|
|
|
|
// In case of file moniker life is easy : )
|
|
DWORD aMonType = 0;
|
|
hr = pMoniker->IsSystemMoniker( &aMonType );
|
|
if ( SUCCEEDED( hr ) && aMonType == MKSYS_FILEMONIKER )
|
|
{
|
|
sal::systools::COMReference< IMalloc > pMalloc;
|
|
hr = CoGetMalloc( 1, &pMalloc ); // if fails there will be a memory leak
|
|
OSL_ENSURE(SUCCEEDED(hr) && pMalloc, "CoGetMalloc() failed!");
|
|
|
|
LPOLESTR pOleStr = nullptr;
|
|
hr = pOleLink->GetSourceDisplayName( &pOleStr );
|
|
if ( SUCCEEDED( hr ) && pOleStr )
|
|
{
|
|
std::wstring aFilePath( pOleStr );
|
|
if ( pMalloc )
|
|
pMalloc->Free( pOleStr );
|
|
|
|
hr = OleCreateFromFile( CLSID_NULL,
|
|
aFilePath.c_str(),
|
|
IID_IUnknown,
|
|
OLERENDER_DRAW, // OLERENDER_FORMAT
|
|
nullptr,
|
|
nullptr,
|
|
pStorage,
|
|
IID_PPV_ARGS_Helper(&m_pNativeImpl->m_pObj) );
|
|
}
|
|
}
|
|
|
|
// in case of other moniker types the only way is to get storage
|
|
if ( !m_pNativeImpl->m_pObj )
|
|
{
|
|
sal::systools::COMReference< IBindCtx > pBindCtx;
|
|
hr = CreateBindCtx( 0, &pBindCtx );
|
|
if ( SUCCEEDED( hr ) && pBindCtx )
|
|
{
|
|
sal::systools::COMReference< IStorage > pObjectStorage;
|
|
hr = pMoniker->BindToStorage(pBindCtx, nullptr, IID_PPV_ARGS(&pObjectStorage));
|
|
if ( SUCCEEDED( hr ) && pObjectStorage )
|
|
{
|
|
hr = pObjectStorage->CopyTo(0, nullptr, nullptr, pStorage);
|
|
if ( SUCCEEDED( hr ) )
|
|
hr = OleLoadSeh(pStorage, &m_pNativeImpl->m_pObj);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
InitializeObject_Impl();
|
|
}
|
|
|
|
|
|
void OleComponent::RunObject()
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
OSL_ENSURE(pOleObject, "The pointer can not be set to NULL here!");
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
if (!OleIsRunning(pOleObject))
|
|
{
|
|
HRESULT hr = OleRun( m_pNativeImpl->m_pObj );
|
|
|
|
if ( FAILED( hr ) )
|
|
{
|
|
OUString error = WindowsErrorStringFromHRESULT(hr);
|
|
if ( hr == REGDB_E_CLASSNOTREG )
|
|
{
|
|
if (auto pOleObj
|
|
= m_pNativeImpl->m_pObj.QueryInterface<IOleObject>(sal::systools::COM_QUERY))
|
|
{
|
|
LPOLESTR lpUserType = nullptr;
|
|
if (SUCCEEDED(pOleObj->GetUserType(USERCLASSTYPE_FULL, &lpUserType)))
|
|
{
|
|
error += OUString::Concat("\n") + o3tl::toU(lpUserType);
|
|
sal::systools::COMReference<IMalloc> pMalloc;
|
|
hr = CoGetMalloc(1, &pMalloc); // if fails there will be a memory leak
|
|
SAL_WARN_IF(FAILED(hr) || !pMalloc, "embeddedobj.ole", "CoGetMalloc() failed");
|
|
if (pMalloc)
|
|
pMalloc->Free(lpUserType);
|
|
}
|
|
}
|
|
throw embed::UnreachableStateException(
|
|
error, getXWeak(), -1,
|
|
css::embed::EmbedStates::RUNNING); // the object server is not installed
|
|
}
|
|
else
|
|
throw io::IOException(error, getXWeak());
|
|
}
|
|
// Only now, when the object is activated, it can be registered in the global table;
|
|
// before this point, RegisterInterfaceInGlobal would return CO_E_OBJNOTCONNECTED
|
|
m_pNativeImpl->registerObj();
|
|
}
|
|
}
|
|
|
|
|
|
awt::Size OleComponent::CalculateWithFactor( const awt::Size& aSize,
|
|
const awt::Size& aMultiplier,
|
|
const awt::Size& aDivisor )
|
|
{
|
|
awt::Size aResult;
|
|
|
|
sal_Int64 nWidth = static_cast<sal_Int64>(aSize.Width) * static_cast<sal_Int64>(aMultiplier.Width) / static_cast<sal_Int64>(aDivisor.Width);
|
|
sal_Int64 nHeight = static_cast<sal_Int64>(aSize.Height) * static_cast<sal_Int64>(aMultiplier.Height) / static_cast<sal_Int64>(aDivisor.Height);
|
|
OSL_ENSURE( nWidth < SAL_MAX_INT32 && nWidth > SAL_MIN_INT32
|
|
&& nHeight < SAL_MAX_INT32 && nHeight > SAL_MIN_INT32,
|
|
"Unacceptable result size!" );
|
|
|
|
aResult.Width = static_cast<sal_Int32>(nWidth);
|
|
aResult.Height = static_cast<sal_Int32>(nHeight);
|
|
|
|
return aResult;
|
|
}
|
|
|
|
|
|
void OleComponent::CloseObject()
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (pOleObject && OleIsRunning(pOleObject))
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
HRESULT hr = pOleObject->Close(OLECLOSE_NOSAVE); // must be saved before
|
|
SAL_WARN_IF(FAILED(hr), "embeddedobj.ole", "IOleObject::Close failed");
|
|
}
|
|
}
|
|
|
|
|
|
uno::Sequence< embed::VerbDescriptor > OleComponent::GetVerbList()
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
if( !m_aVerbList.getLength() )
|
|
{
|
|
sal::systools::COMReference< IEnumOLEVERB > pEnum;
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pOleObject->EnumVerbs(&pEnum);
|
|
}
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
OLEVERB szEle[ MAX_ENUM_ELE ];
|
|
ULONG nNum = 0;
|
|
sal_Int32 nSeqSize = 0;
|
|
|
|
do
|
|
{
|
|
hr = pEnum->Next(MAX_ENUM_ELE, szEle, &nNum);
|
|
if( hr == S_OK || hr == S_FALSE )
|
|
{
|
|
m_aVerbList.realloc( nSeqSize += nNum );
|
|
auto pVerbList = m_aVerbList.getArray();
|
|
for( sal_uInt32 nInd = 0; nInd < nNum; nInd++ )
|
|
{
|
|
pVerbList[nSeqSize-nNum+nInd].VerbID = szEle[ nInd ].lVerb;
|
|
pVerbList[nSeqSize-nNum+nInd].VerbName = WinAccToVcl_Impl( o3tl::toU(szEle[ nInd ].lpszVerbName) );
|
|
pVerbList[nSeqSize-nNum+nInd].VerbFlags = szEle[ nInd ].fuFlags;
|
|
pVerbList[nSeqSize-nNum+nInd].VerbAttributes = szEle[ nInd ].grfAttribs;
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
while( nNum == MAX_ENUM_ELE );
|
|
}
|
|
}
|
|
|
|
return m_aVerbList;
|
|
}
|
|
|
|
|
|
void OleComponent::ExecuteVerb( sal_Int32 nVerbID )
|
|
{
|
|
RunObject();
|
|
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO
|
|
|
|
SolarMutexReleaser releaser;
|
|
|
|
// TODO: probably extents should be set here and stored in aRect
|
|
// TODO: probably the parent window also should be set
|
|
HRESULT hr = pOleObject->DoVerb(nVerbID, nullptr, m_pOleWrapClientSite, 0, nullptr, nullptr);
|
|
|
|
if ( FAILED( hr ) )
|
|
throw io::IOException(); // TODO
|
|
}
|
|
|
|
|
|
void OleComponent::SetHostName( const OUString& aEmbDocName )
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
SolarMutexReleaser releaser;
|
|
pOleObject->SetHostNames(L"app name", o3tl::toW(aEmbDocName.getStr()));
|
|
}
|
|
|
|
|
|
void OleComponent::SetExtent( const awt::Size& aVisAreaSize, sal_Int64 nAspect )
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
DWORD nMSAspect = static_cast<DWORD>(nAspect); // first 32 bits are for MS aspects
|
|
|
|
SIZEL aSize = { aVisAreaSize.Width, aVisAreaSize.Height };
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pOleObject->SetExtent(nMSAspect, &aSize);
|
|
}
|
|
|
|
if ( FAILED( hr ) )
|
|
{
|
|
// TODO/LATER: is it correct? In future user code probably should be ready for the exception.
|
|
// if the object is running but not activated, RPC_E_SERVER_DIED error code is returned by OLE package
|
|
// in this case just do nothing
|
|
// Also Visio returns E_FAIL on resize if it is in running state
|
|
// if ( hr != RPC_E_SERVER_DIED )
|
|
throw io::IOException(); // TODO
|
|
}
|
|
}
|
|
|
|
|
|
awt::Size OleComponent::GetExtent( sal_Int64 nAspect )
|
|
{
|
|
if (!m_pNativeImpl->m_pObj)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
DWORD nMSAspect = static_cast<DWORD>(nAspect); // first 32 bits are for MS aspects
|
|
awt::Size aSize;
|
|
bool bGotSize = false;
|
|
|
|
if ( nMSAspect == DVASPECT_CONTENT )
|
|
{
|
|
// Try to get the size from the replacement image first
|
|
if (auto pDataObject = m_pNativeImpl->get<IDataObject>())
|
|
{
|
|
STGMEDIUM aMedium;
|
|
FORMATETC aFormat = pFormatTemplates[1]; // use windows metafile format
|
|
aFormat.dwAspect = nMSAspect;
|
|
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pDataObject->GetData(&aFormat, &aMedium);
|
|
}
|
|
|
|
if (hr == RPC_E_WRONG_THREAD)
|
|
{
|
|
// Assume that the OLE object was loaded on the main thread.
|
|
vcl::solarthread::syncExecute([this, &hr, &pDataObject, &aFormat, &aMedium]() {
|
|
// Make sure that the current state is embed::EmbedStates::RUNNING.
|
|
RunObject();
|
|
// Now try again on the correct thread.
|
|
hr = pDataObject->GetData(&aFormat, &aMedium);
|
|
});
|
|
}
|
|
|
|
if ( SUCCEEDED( hr ) && aMedium.tymed == TYMED_MFPICT ) // Win Metafile
|
|
{
|
|
METAFILEPICT* pMF = static_cast<METAFILEPICT*>(GlobalLock( aMedium.hMetaFilePict ));
|
|
if ( pMF )
|
|
{
|
|
// the object uses 0.01 mm as unit, so the metafile size should be converted to object unit
|
|
o3tl::Length eFrom = o3tl::Length::mm100;
|
|
switch( pMF->mm )
|
|
{
|
|
case MM_HIENGLISH:
|
|
eFrom = o3tl::Length::in1000;
|
|
break;
|
|
|
|
case MM_LOENGLISH:
|
|
eFrom = o3tl::Length::in100;
|
|
break;
|
|
|
|
case MM_LOMETRIC:
|
|
eFrom = o3tl::Length::mm10;
|
|
break;
|
|
|
|
case MM_TWIPS:
|
|
eFrom = o3tl::Length::twip;
|
|
break;
|
|
|
|
case MM_ISOTROPIC:
|
|
case MM_ANISOTROPIC:
|
|
case MM_HIMETRIC:
|
|
// do nothing
|
|
break;
|
|
}
|
|
|
|
sal_Int64 nX = o3tl::convert(abs( pMF->xExt ), eFrom, o3tl::Length::mm100);
|
|
sal_Int64 nY = o3tl::convert(abs( pMF->yExt ), eFrom, o3tl::Length::mm100);
|
|
if ( nX < SAL_MAX_INT32 && nY < SAL_MAX_INT32 )
|
|
{
|
|
aSize.Width = static_cast<sal_Int32>(nX);
|
|
aSize.Height = static_cast<sal_Int32>(nY);
|
|
bGotSize = true;
|
|
}
|
|
else
|
|
OSL_FAIL( "Unexpected size is provided!" );
|
|
}
|
|
}
|
|
else if (!SUCCEEDED(hr))
|
|
{
|
|
SAL_WARN("embeddedobj.ole", " OleComponent::GetExtent: GetData() failed");
|
|
}
|
|
// i113605, to release storage medium
|
|
if ( SUCCEEDED( hr ) )
|
|
::ReleaseStgMedium(&aMedium);
|
|
}
|
|
}
|
|
|
|
if ( !bGotSize )
|
|
throw lang::IllegalArgumentException();
|
|
|
|
return aSize;
|
|
}
|
|
|
|
|
|
awt::Size OleComponent::GetCachedExtent( sal_Int64 nAspect )
|
|
{
|
|
auto pViewObject2(m_pNativeImpl->get<IViewObject2>());
|
|
if (!pViewObject2)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
DWORD nMSAspect = static_cast<DWORD>(nAspect); // first 32 bits are for MS aspects
|
|
SIZEL aSize;
|
|
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pViewObject2->GetExtent(nMSAspect, -1, nullptr, &aSize);
|
|
}
|
|
|
|
if ( FAILED( hr ) )
|
|
{
|
|
// TODO/LATER: is it correct?
|
|
// if there is no appropriate cache for the aspect, OLE_E_BLANK error code is returned
|
|
// if ( hr == OLE_E_BLANK )
|
|
// throw lang::IllegalArgumentException();
|
|
//else
|
|
// throw io::IOException(); // TODO
|
|
|
|
SAL_WARN("embeddedobj.ole", " OleComponent::GetCachedExtent: GetExtent() failed");
|
|
throw lang::IllegalArgumentException();
|
|
}
|
|
|
|
return awt::Size( aSize.cx, aSize.cy );
|
|
}
|
|
|
|
|
|
awt::Size OleComponent::GetRecommendedExtent( sal_Int64 nAspect )
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
DWORD nMSAspect = static_cast<DWORD>(nAspect); // first 32 bits are for MS aspects
|
|
SIZEL aSize;
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pOleObject->GetExtent(nMSAspect, &aSize);
|
|
}
|
|
if ( FAILED( hr ) )
|
|
{
|
|
SAL_WARN("embeddedobj.ole", " OleComponent::GetRecommendedExtent: GetExtent() failed");
|
|
throw lang::IllegalArgumentException();
|
|
}
|
|
|
|
return awt::Size( aSize.cx, aSize.cy );
|
|
}
|
|
|
|
|
|
sal_Int64 OleComponent::GetMiscStatus( sal_Int64 nAspect )
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
DWORD nResult = 0;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
pOleObject->GetMiscStatus(static_cast<DWORD>(nAspect), &nResult);
|
|
}
|
|
return static_cast<sal_Int64>(nResult); // first 32 bits are for MS flags
|
|
}
|
|
|
|
|
|
uno::Sequence< sal_Int8 > OleComponent::GetCLSID()
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
GUID aCLSID;
|
|
HRESULT hr;
|
|
{
|
|
SolarMutexReleaser releaser;
|
|
hr = pOleObject->GetUserClassID(&aCLSID);
|
|
}
|
|
if ( FAILED( hr ) )
|
|
throw io::IOException(); // TODO:
|
|
|
|
return MimeConfigurationHelper::GetSequenceClassID( aCLSID.Data1, aCLSID.Data2, aCLSID.Data3,
|
|
aCLSID.Data4[0], aCLSID.Data4[1],
|
|
aCLSID.Data4[2], aCLSID.Data4[3],
|
|
aCLSID.Data4[4], aCLSID.Data4[5],
|
|
aCLSID.Data4[6], aCLSID.Data4[7] );
|
|
}
|
|
|
|
|
|
bool OleComponent::IsDirty()
|
|
{
|
|
if ( IsWorkaroundActive() )
|
|
return true;
|
|
|
|
auto pPersistStorage(m_pNativeImpl->get<IPersistStorage>());
|
|
if ( !pPersistStorage )
|
|
throw io::IOException(); // TODO
|
|
|
|
SolarMutexReleaser releaser;
|
|
HRESULT hr = pPersistStorage->IsDirty();
|
|
return ( hr != S_FALSE );
|
|
}
|
|
|
|
|
|
void OleComponent::StoreOwnTmpIfNecessary()
|
|
{
|
|
auto pOleObject(m_pNativeImpl->get<IOleObject>());
|
|
if (!pOleObject)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
auto pPersistStorage(m_pNativeImpl->get<IPersistStorage>());
|
|
if ( !pPersistStorage )
|
|
throw io::IOException(); // TODO
|
|
|
|
SolarMutexReleaser releaser;
|
|
|
|
if ( m_bWorkaroundActive || pPersistStorage->IsDirty() != S_FALSE )
|
|
{
|
|
auto pStorage(m_pNativeImpl->getStorage());
|
|
HRESULT hr = OleSave(pPersistStorage, pStorage, TRUE);
|
|
if ( FAILED( hr ) )
|
|
{
|
|
// Till now was required only for AcrobatReader7.0.8
|
|
GUID aCLSID;
|
|
hr = pOleObject->GetUserClassID(&aCLSID);
|
|
if ( FAILED( hr ) )
|
|
{
|
|
SAL_WARN("embeddedobj.ole", "OleComponent::StoreOwnTmpIfNecessary: GetUserClassID() failed");
|
|
throw io::IOException(); // TODO
|
|
}
|
|
|
|
hr = WriteClassStg(pStorage, aCLSID);
|
|
if ( FAILED( hr ) )
|
|
throw io::IOException(); // TODO
|
|
|
|
// the result of the following call is not checked because some objects, for example AcrobatReader7.0.8
|
|
// return error even in case the saving was done correctly
|
|
hr = pPersistStorage->Save(pStorage, TRUE);
|
|
|
|
// another workaround for AcrobatReader7.0.8 object, this object might think that it is not changed
|
|
// when it has been created from file, although it must be saved
|
|
m_bWorkaroundActive = true;
|
|
}
|
|
|
|
hr = pStorage->Commit(STGC_DEFAULT);
|
|
if ( FAILED( hr ) )
|
|
throw io::IOException(); // TODO
|
|
|
|
hr = pPersistStorage->SaveCompleted( nullptr );
|
|
if ( FAILED( hr ) && hr != E_UNEXPECTED )
|
|
throw io::IOException(); // TODO
|
|
|
|
}
|
|
}
|
|
|
|
|
|
bool OleComponent::SaveObject_Impl()
|
|
{
|
|
bool bResult = false;
|
|
OleEmbeddedObject* pLockObject = nullptr;
|
|
|
|
{
|
|
osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_pUnoOleObject )
|
|
{
|
|
pLockObject = m_pUnoOleObject;
|
|
pLockObject->acquire();
|
|
}
|
|
}
|
|
|
|
if ( pLockObject )
|
|
{
|
|
bResult = pLockObject->SaveObject_Impl();
|
|
pLockObject->release();
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
|
|
bool OleComponent::OnShowWindow_Impl( bool bShow )
|
|
{
|
|
bool bResult = false;
|
|
OleEmbeddedObject* pLockObject = nullptr;
|
|
|
|
{
|
|
osl::MutexGuard aGuard( m_aMutex );
|
|
|
|
if ( m_pUnoOleObject )
|
|
{
|
|
pLockObject = m_pUnoOleObject;
|
|
pLockObject->acquire();
|
|
}
|
|
}
|
|
|
|
if ( pLockObject )
|
|
{
|
|
bResult = pLockObject->OnShowWindow_Impl( bShow );
|
|
pLockObject->release();
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
|
|
void OleComponent::OnViewChange_Impl( sal_uInt32 dwAspect )
|
|
{
|
|
// TODO: check if it is enough or may be saving notifications are required for Visio2000
|
|
::rtl::Reference< OleEmbeddedObject > xLockObject;
|
|
|
|
{
|
|
osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_pUnoOleObject )
|
|
xLockObject = m_pUnoOleObject;
|
|
}
|
|
|
|
if ( xLockObject.is() )
|
|
{
|
|
uno::Reference < awt::XRequestCallback > xRequestCallback(
|
|
m_xContext->getServiceManager()->createInstanceWithContext("com.sun.star.awt.AsyncCallback", m_xContext),
|
|
uno::UNO_QUERY );
|
|
xRequestCallback->addCallback( new MainThreadNotificationRequest( xLockObject, OLECOMP_ONVIEWCHANGE, dwAspect ), uno::Any() );
|
|
}
|
|
}
|
|
|
|
|
|
void OleComponent::OnClose_Impl()
|
|
{
|
|
::rtl::Reference< OleEmbeddedObject > xLockObject;
|
|
|
|
{
|
|
osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_pUnoOleObject )
|
|
xLockObject = m_pUnoOleObject;
|
|
}
|
|
|
|
if ( xLockObject.is() )
|
|
{
|
|
uno::Reference < awt::XRequestCallback > xRequestCallback(
|
|
m_xContext->getServiceManager()->createInstanceWithContext("com.sun.star.awt.AsyncCallback", m_xContext),
|
|
uno::UNO_QUERY );
|
|
xRequestCallback->addCallback( new MainThreadNotificationRequest( xLockObject, OLECOMP_ONCLOSE ), uno::Any() );
|
|
}
|
|
}
|
|
|
|
// XCloseable
|
|
|
|
void SAL_CALL OleComponent::close( sal_Bool bDeliverOwnership )
|
|
{
|
|
uno::Reference< uno::XInterface > xSelfHold;
|
|
{
|
|
osl::MutexGuard aGuard(m_aMutex);
|
|
if (m_bDisposed)
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
xSelfHold.set(static_cast<::cppu::OWeakObject*>(this));
|
|
lang::EventObject aSource(static_cast<::cppu::OWeakObject*>(this));
|
|
|
|
if (m_pInterfaceContainer)
|
|
{
|
|
comphelper::OInterfaceContainerHelper2* pContainer
|
|
= m_pInterfaceContainer->getContainer(cppu::UnoType<util::XCloseListener>::get());
|
|
if (pContainer != nullptr)
|
|
{
|
|
comphelper::OInterfaceIteratorHelper2 pIterator(*pContainer);
|
|
while (pIterator.hasMoreElements())
|
|
{
|
|
try
|
|
{
|
|
static_cast<util::XCloseListener*>(pIterator.next())
|
|
->queryClosing(aSource, bDeliverOwnership);
|
|
}
|
|
catch (const uno::RuntimeException&)
|
|
{
|
|
pIterator.remove();
|
|
}
|
|
}
|
|
}
|
|
|
|
pContainer
|
|
= m_pInterfaceContainer->getContainer(cppu::UnoType<util::XCloseListener>::get());
|
|
if (pContainer != nullptr)
|
|
{
|
|
comphelper::OInterfaceIteratorHelper2 pCloseIterator(*pContainer);
|
|
while (pCloseIterator.hasMoreElements())
|
|
{
|
|
try
|
|
{
|
|
static_cast<util::XCloseListener*>(pCloseIterator.next())
|
|
->notifyClosing(aSource);
|
|
}
|
|
catch (const uno::RuntimeException&)
|
|
{
|
|
pCloseIterator.remove();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Dispose();
|
|
}
|
|
|
|
|
|
void SAL_CALL OleComponent::addCloseListener( const uno::Reference< util::XCloseListener >& xListener )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( !m_pInterfaceContainer )
|
|
m_pInterfaceContainer = new comphelper::OMultiTypeInterfaceContainerHelper2( m_aMutex );
|
|
|
|
m_pInterfaceContainer->addInterface( cppu::UnoType<util::XCloseListener>::get(), xListener );
|
|
}
|
|
|
|
|
|
void SAL_CALL OleComponent::removeCloseListener( const uno::Reference< util::XCloseListener >& xListener )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( m_pInterfaceContainer )
|
|
m_pInterfaceContainer->removeInterface( cppu::UnoType<util::XCloseListener>::get(),
|
|
xListener );
|
|
}
|
|
|
|
// XTransferable
|
|
|
|
uno::Any SAL_CALL OleComponent::getTransferData( const datatransfer::DataFlavor& aFlavor )
|
|
{
|
|
::osl::ResettableMutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if (!m_pNativeImpl->m_pObj)
|
|
throw embed::WrongStateException(); // TODO: the object is in wrong state
|
|
|
|
uno::Any aResult;
|
|
bool bSupportedFlavor = false;
|
|
|
|
if ( m_pNativeImpl->GraphicalFlavor( aFlavor ) )
|
|
{
|
|
DWORD nRequestedAspect = GetAspectFromFlavor( aFlavor );
|
|
// if own icon is set and icon aspect is requested the own icon can be returned directly
|
|
|
|
auto pDataObject(m_pNativeImpl->get<IDataObject>());
|
|
if ( !pDataObject )
|
|
throw io::IOException(); // TODO: transport error code
|
|
|
|
// The following optimization does not make much sense currently just because
|
|
// only one aspect is supported, and only three formats for the aspect are supported
|
|
// and moreover it is not guaranteed that the once returned format will be supported further
|
|
// example - i52106
|
|
// TODO/LATER: bring the optimization back when other aspects are supported
|
|
|
|
// FORMATETC* pFormatEtc = m_pNativeImpl->GetSupportedFormatForAspect( nRequestedAspect );
|
|
// if ( pFormatEtc )
|
|
// {
|
|
// STGMEDIUM aMedium;
|
|
// hr = pDataObject->GetData( pFormatEtc, &aMedium );
|
|
// if ( SUCCEEDED( hr ) )
|
|
// bSupportedFlavor = m_pNativeImpl->ConvertDataForFlavor( aMedium, aFlavor, aResult );
|
|
// }
|
|
// else
|
|
{
|
|
// the supported format of the application is still not found, find one
|
|
for ( sal_Int32 nInd = 0; nInd < FORMATS_NUM; nInd++ )
|
|
{
|
|
STGMEDIUM aMedium;
|
|
FORMATETC aFormat = pFormatTemplates[nInd];
|
|
aFormat.dwAspect = nRequestedAspect;
|
|
|
|
HRESULT hr;
|
|
{
|
|
osl::ResettableMutexGuardScopedReleaser own_releaser(aGuard);
|
|
SolarMutexReleaser releaser;
|
|
hr = pDataObject->GetData(&aFormat, &aMedium);
|
|
}
|
|
if ( SUCCEEDED( hr ) )
|
|
{
|
|
bSupportedFlavor = m_pNativeImpl->ConvertDataForFlavor( aMedium, aFlavor, aResult );
|
|
::ReleaseStgMedium(&aMedium); // i113605, to release storage medium
|
|
if ( bSupportedFlavor )
|
|
{
|
|
// TODO/LATER: bring the optimization back when other aspects are supported
|
|
// m_pNativeImpl->AddSupportedFormat( aFormat );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If the replacement could not be retrieved, the cached representation should be used
|
|
// currently it is not necessary to retrieve it here, so it is implemented in the object itself
|
|
}
|
|
// TODO: Investigate if there is already the format name
|
|
// and whether this format is really required
|
|
else if ( aFlavor.DataType == cppu::UnoType<io::XInputStream>::get()
|
|
&& aFlavor.MimeType == "application/x-openoffice-contentstream" )
|
|
{
|
|
// allow to retrieve stream-representation of the object persistence
|
|
bSupportedFlavor = true;
|
|
uno::Reference < io::XStream > xTempFileStream(
|
|
io::TempFile::create(m_xContext),
|
|
uno::UNO_QUERY_THROW );
|
|
|
|
uno::Reference< io::XOutputStream > xTempOutStream = xTempFileStream->getOutputStream();
|
|
uno::Reference< io::XInputStream > xTempInStream = xTempFileStream->getInputStream();
|
|
if ( !(xTempOutStream.is() && xTempInStream.is()) )
|
|
throw io::IOException(); // TODO:
|
|
|
|
OSL_ENSURE( m_pUnoOleObject, "Unexpected object absence!" );
|
|
if ( !m_pUnoOleObject )
|
|
throw uno::RuntimeException();
|
|
|
|
m_pUnoOleObject->StoreObjectToStream(xTempOutStream, aGuard);
|
|
|
|
xTempOutStream->closeOutput();
|
|
xTempOutStream.clear();
|
|
|
|
aResult <<= xTempInStream;
|
|
}
|
|
|
|
if ( !bSupportedFlavor )
|
|
throw datatransfer::UnsupportedFlavorException();
|
|
|
|
return aResult;
|
|
}
|
|
|
|
|
|
uno::Sequence< datatransfer::DataFlavor > SAL_CALL OleComponent::getTransferDataFlavors()
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
RetrieveObjectDataFlavors_Impl();
|
|
|
|
return m_aDataFlavors;
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL OleComponent::isDataFlavorSupported( const datatransfer::DataFlavor& aFlavor )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
RetrieveObjectDataFlavors_Impl();
|
|
|
|
for (auto const& supportedFormat : m_aDataFlavors)
|
|
if ( supportedFormat.MimeType.equals( aFlavor.MimeType ) && supportedFormat.DataType == aFlavor.DataType )
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void SAL_CALL OleComponent::dispose()
|
|
{
|
|
try
|
|
{
|
|
close( true );
|
|
}
|
|
catch ( const uno::Exception& )
|
|
{
|
|
}
|
|
}
|
|
|
|
void SAL_CALL OleComponent::addEventListener( const uno::Reference< lang::XEventListener >& xListener )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( !m_pInterfaceContainer )
|
|
m_pInterfaceContainer = new comphelper::OMultiTypeInterfaceContainerHelper2( m_aMutex );
|
|
|
|
m_pInterfaceContainer->addInterface( cppu::UnoType<lang::XEventListener>::get(), xListener );
|
|
}
|
|
|
|
|
|
void SAL_CALL OleComponent::removeEventListener( const uno::Reference< lang::XEventListener >& xListener )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( m_pInterfaceContainer )
|
|
m_pInterfaceContainer->removeInterface( cppu::UnoType<lang::XEventListener>::get(),
|
|
xListener );
|
|
}
|
|
|
|
sal_Int64 SAL_CALL OleComponent::getSomething( const css::uno::Sequence< sal_Int8 >& aIdentifier )
|
|
{
|
|
try
|
|
{
|
|
uno::Sequence < sal_Int8 > aCLSID = GetCLSID();
|
|
if ( MimeConfigurationHelper::ClassIDsEqual( aIdentifier, aCLSID ) )
|
|
return comphelper::getSomething_cast(m_pNativeImpl->m_pObj.get());
|
|
}
|
|
catch ( const uno::Exception& )
|
|
{
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
sal_Bool SAL_CALL OleComponent::isModified()
|
|
{
|
|
return m_bModified;
|
|
}
|
|
|
|
void SAL_CALL OleComponent::setModified( sal_Bool bModified )
|
|
{
|
|
m_bModified = bModified;
|
|
|
|
if ( bModified && m_pInterfaceContainer )
|
|
{
|
|
comphelper::OInterfaceContainerHelper2* pContainer =
|
|
m_pInterfaceContainer->getContainer( cppu::UnoType<util::XModifyListener>::get());
|
|
if ( pContainer != nullptr )
|
|
{
|
|
comphelper::OInterfaceIteratorHelper2 pIterator( *pContainer );
|
|
while ( pIterator.hasMoreElements() )
|
|
{
|
|
try
|
|
{
|
|
lang::EventObject aEvent( static_cast<util::XModifiable*>(this) );
|
|
static_cast<util::XModifyListener*>(pIterator.next())->modified( aEvent );
|
|
}
|
|
catch( const uno::RuntimeException& )
|
|
{
|
|
pIterator.remove();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SAL_CALL OleComponent::addModifyListener( const css::uno::Reference < css::util::XModifyListener >& xListener )
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( !m_pInterfaceContainer )
|
|
m_pInterfaceContainer = new comphelper::OMultiTypeInterfaceContainerHelper2( m_aMutex );
|
|
|
|
m_pInterfaceContainer->addInterface( cppu::UnoType<util::XModifyListener>::get(), xListener );
|
|
}
|
|
|
|
void SAL_CALL OleComponent::removeModifyListener( const css::uno::Reference < css::util::XModifyListener >& xListener)
|
|
{
|
|
::osl::MutexGuard aGuard( m_aMutex );
|
|
if ( m_bDisposed )
|
|
throw lang::DisposedException(); // TODO
|
|
|
|
if ( m_pInterfaceContainer )
|
|
m_pInterfaceContainer->removeInterface( cppu::UnoType<util::XModifyListener>::get(),
|
|
xListener );
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|