1*cdf0e10cSrcweir /************************************************************************* 2*cdf0e10cSrcweir * 3*cdf0e10cSrcweir * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4*cdf0e10cSrcweir * 5*cdf0e10cSrcweir * Copyright 2000, 2010 Oracle and/or its affiliates. 6*cdf0e10cSrcweir * 7*cdf0e10cSrcweir * OpenOffice.org - a multi-platform office productivity suite 8*cdf0e10cSrcweir * 9*cdf0e10cSrcweir * This file is part of OpenOffice.org. 10*cdf0e10cSrcweir * 11*cdf0e10cSrcweir * OpenOffice.org is free software: you can redistribute it and/or modify 12*cdf0e10cSrcweir * it under the terms of the GNU Lesser General Public License version 3 13*cdf0e10cSrcweir * only, as published by the Free Software Foundation. 14*cdf0e10cSrcweir * 15*cdf0e10cSrcweir * OpenOffice.org is distributed in the hope that it will be useful, 16*cdf0e10cSrcweir * but WITHOUT ANY WARRANTY; without even the implied warranty of 17*cdf0e10cSrcweir * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 18*cdf0e10cSrcweir * GNU Lesser General Public License version 3 for more details 19*cdf0e10cSrcweir * (a copy is included in the LICENSE file that accompanied this code). 20*cdf0e10cSrcweir * 21*cdf0e10cSrcweir * You should have received a copy of the GNU Lesser General Public License 22*cdf0e10cSrcweir * version 3 along with OpenOffice.org. If not, see 23*cdf0e10cSrcweir * <http://www.openoffice.org/license.html> 24*cdf0e10cSrcweir * for a copy of the LGPLv3 License. 25*cdf0e10cSrcweir * 26*cdf0e10cSrcweir ************************************************************************/ 27*cdf0e10cSrcweir 28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove 29*cdf0e10cSrcweir #include "precompiled_dtrans.hxx" 30*cdf0e10cSrcweir 31*cdf0e10cSrcweir //------------------------------------------------------------------------ 32*cdf0e10cSrcweir // includes 33*cdf0e10cSrcweir //------------------------------------------------------------------------ 34*cdf0e10cSrcweir #include <osl/diagnose.h> 35*cdf0e10cSrcweir 36*cdf0e10cSrcweir #ifndef _TXDATAOBJECT_HXX_ 37*cdf0e10cSrcweir #include "XTDataObject.hxx" 38*cdf0e10cSrcweir #endif 39*cdf0e10cSrcweir #include <com/sun/star/datatransfer/DataFlavor.hpp> 40*cdf0e10cSrcweir #include "..\misc\ImplHelper.hxx" 41*cdf0e10cSrcweir #include "DTransHelper.hxx" 42*cdf0e10cSrcweir #include "TxtCnvtHlp.hxx" 43*cdf0e10cSrcweir #include <com/sun/star/datatransfer/clipboard/XClipboardEx.hpp> 44*cdf0e10cSrcweir #include "FmtFilter.hxx" 45*cdf0e10cSrcweir 46*cdf0e10cSrcweir #if defined _MSC_VER 47*cdf0e10cSrcweir #pragma warning(push,1) 48*cdf0e10cSrcweir #pragma warning(disable:4917) 49*cdf0e10cSrcweir #endif 50*cdf0e10cSrcweir #include <windows.h> 51*cdf0e10cSrcweir #include <shlobj.h> 52*cdf0e10cSrcweir #if defined _MSC_VER 53*cdf0e10cSrcweir #pragma warning(pop) 54*cdf0e10cSrcweir #endif 55*cdf0e10cSrcweir 56*cdf0e10cSrcweir #ifdef __MINGW32__ 57*cdf0e10cSrcweir #define __uuidof(I) IID_##I 58*cdf0e10cSrcweir #endif 59*cdf0e10cSrcweir 60*cdf0e10cSrcweir //------------------------------------------------------------------------ 61*cdf0e10cSrcweir // namespace directives 62*cdf0e10cSrcweir //------------------------------------------------------------------------ 63*cdf0e10cSrcweir 64*cdf0e10cSrcweir using namespace com::sun::star::datatransfer; 65*cdf0e10cSrcweir using namespace com::sun::star::datatransfer::clipboard; 66*cdf0e10cSrcweir using namespace com::sun::star::uno; 67*cdf0e10cSrcweir using namespace com::sun::star::lang; 68*cdf0e10cSrcweir using namespace rtl; 69*cdf0e10cSrcweir 70*cdf0e10cSrcweir //------------------------------------------------------------------------ 71*cdf0e10cSrcweir // a helper class that will be thrown by the function validateFormatEtc 72*cdf0e10cSrcweir //------------------------------------------------------------------------ 73*cdf0e10cSrcweir 74*cdf0e10cSrcweir class CInvalidFormatEtcException 75*cdf0e10cSrcweir { 76*cdf0e10cSrcweir public: 77*cdf0e10cSrcweir HRESULT m_hr; 78*cdf0e10cSrcweir CInvalidFormatEtcException( HRESULT hr ) : m_hr( hr ) {}; 79*cdf0e10cSrcweir }; 80*cdf0e10cSrcweir 81*cdf0e10cSrcweir //------------------------------------------------------------------------ 82*cdf0e10cSrcweir // ctor 83*cdf0e10cSrcweir //------------------------------------------------------------------------ 84*cdf0e10cSrcweir 85*cdf0e10cSrcweir CXTDataObject::CXTDataObject( const Reference< XMultiServiceFactory >& aServiceManager, 86*cdf0e10cSrcweir const Reference< XTransferable >& aXTransferable ) : 87*cdf0e10cSrcweir m_nRefCnt( 0 ), 88*cdf0e10cSrcweir m_SrvMgr( aServiceManager ), 89*cdf0e10cSrcweir m_XTransferable( aXTransferable ), 90*cdf0e10cSrcweir m_DataFormatTranslator( aServiceManager ), 91*cdf0e10cSrcweir m_bFormatEtcContainerInitialized( sal_False ), 92*cdf0e10cSrcweir m_FormatRegistrar( m_SrvMgr, m_DataFormatTranslator ) 93*cdf0e10cSrcweir { 94*cdf0e10cSrcweir } 95*cdf0e10cSrcweir 96*cdf0e10cSrcweir //------------------------------------------------------------------------ 97*cdf0e10cSrcweir // IUnknown->QueryInterface 98*cdf0e10cSrcweir //------------------------------------------------------------------------ 99*cdf0e10cSrcweir 100*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::QueryInterface( REFIID iid, LPVOID* ppvObject ) 101*cdf0e10cSrcweir { 102*cdf0e10cSrcweir if ( NULL == ppvObject ) 103*cdf0e10cSrcweir return E_INVALIDARG; 104*cdf0e10cSrcweir 105*cdf0e10cSrcweir HRESULT hr = E_NOINTERFACE; 106*cdf0e10cSrcweir 107*cdf0e10cSrcweir *ppvObject = NULL; 108*cdf0e10cSrcweir if ( ( __uuidof( IUnknown ) == iid ) || 109*cdf0e10cSrcweir ( __uuidof( IDataObject ) == iid ) ) 110*cdf0e10cSrcweir { 111*cdf0e10cSrcweir *ppvObject = static_cast< IUnknown* >( this ); 112*cdf0e10cSrcweir ( (LPUNKNOWN)*ppvObject )->AddRef( ); 113*cdf0e10cSrcweir hr = S_OK; 114*cdf0e10cSrcweir } 115*cdf0e10cSrcweir 116*cdf0e10cSrcweir return hr; 117*cdf0e10cSrcweir } 118*cdf0e10cSrcweir 119*cdf0e10cSrcweir //------------------------------------------------------------------------ 120*cdf0e10cSrcweir // IUnknown->AddRef 121*cdf0e10cSrcweir //------------------------------------------------------------------------ 122*cdf0e10cSrcweir 123*cdf0e10cSrcweir STDMETHODIMP_(ULONG) CXTDataObject::AddRef( ) 124*cdf0e10cSrcweir { 125*cdf0e10cSrcweir return static_cast< ULONG >( InterlockedIncrement( &m_nRefCnt ) ); 126*cdf0e10cSrcweir } 127*cdf0e10cSrcweir 128*cdf0e10cSrcweir //------------------------------------------------------------------------ 129*cdf0e10cSrcweir // IUnknown->Release 130*cdf0e10cSrcweir //------------------------------------------------------------------------ 131*cdf0e10cSrcweir 132*cdf0e10cSrcweir STDMETHODIMP_(ULONG) CXTDataObject::Release( ) 133*cdf0e10cSrcweir { 134*cdf0e10cSrcweir ULONG nRefCnt = 135*cdf0e10cSrcweir static_cast< ULONG >( InterlockedDecrement( &m_nRefCnt ) ); 136*cdf0e10cSrcweir 137*cdf0e10cSrcweir if ( 0 == nRefCnt ) 138*cdf0e10cSrcweir delete this; 139*cdf0e10cSrcweir 140*cdf0e10cSrcweir return nRefCnt; 141*cdf0e10cSrcweir } 142*cdf0e10cSrcweir 143*cdf0e10cSrcweir //------------------------------------------------------------------------ 144*cdf0e10cSrcweir // 145*cdf0e10cSrcweir //------------------------------------------------------------------------ 146*cdf0e10cSrcweir 147*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::GetData( LPFORMATETC pFormatetc, LPSTGMEDIUM pmedium ) 148*cdf0e10cSrcweir { 149*cdf0e10cSrcweir if ( !(pFormatetc && pmedium) ) 150*cdf0e10cSrcweir return E_INVALIDARG; 151*cdf0e10cSrcweir 152*cdf0e10cSrcweir try 153*cdf0e10cSrcweir { 154*cdf0e10cSrcweir // prepare data transfer 155*cdf0e10cSrcweir invalidateStgMedium( *pmedium ); 156*cdf0e10cSrcweir validateFormatEtc( pFormatetc ); 157*cdf0e10cSrcweir 158*cdf0e10cSrcweir // handle locale request, because locale is a artificial format for us 159*cdf0e10cSrcweir if ( CF_LOCALE == pFormatetc->cfFormat ) 160*cdf0e10cSrcweir renderLocaleAndSetupStgMedium( *pFormatetc, *pmedium ); 161*cdf0e10cSrcweir else if ( CF_UNICODETEXT == pFormatetc->cfFormat ) 162*cdf0e10cSrcweir renderUnicodeAndSetupStgMedium( *pFormatetc, *pmedium ); 163*cdf0e10cSrcweir else 164*cdf0e10cSrcweir renderAnyDataAndSetupStgMedium( *pFormatetc, *pmedium ); 165*cdf0e10cSrcweir } 166*cdf0e10cSrcweir catch(UnsupportedFlavorException&) 167*cdf0e10cSrcweir { 168*cdf0e10cSrcweir HRESULT hr = DV_E_FORMATETC; 169*cdf0e10cSrcweir 170*cdf0e10cSrcweir if ( m_FormatRegistrar.isSynthesizeableFormat( *pFormatetc ) ) 171*cdf0e10cSrcweir hr = renderSynthesizedFormatAndSetupStgMedium( *pFormatetc, *pmedium ); 172*cdf0e10cSrcweir 173*cdf0e10cSrcweir return hr; 174*cdf0e10cSrcweir } 175*cdf0e10cSrcweir catch( CInvalidFormatEtcException& ex ) 176*cdf0e10cSrcweir { 177*cdf0e10cSrcweir return ex.m_hr; 178*cdf0e10cSrcweir } 179*cdf0e10cSrcweir catch( CStgTransferHelper::CStgTransferException& ex ) 180*cdf0e10cSrcweir { 181*cdf0e10cSrcweir return translateStgExceptionCode( ex.m_hr ); 182*cdf0e10cSrcweir } 183*cdf0e10cSrcweir catch(...) 184*cdf0e10cSrcweir { 185*cdf0e10cSrcweir return E_UNEXPECTED; 186*cdf0e10cSrcweir } 187*cdf0e10cSrcweir 188*cdf0e10cSrcweir return S_OK; 189*cdf0e10cSrcweir } 190*cdf0e10cSrcweir 191*cdf0e10cSrcweir //------------------------------------------------------------------------ 192*cdf0e10cSrcweir // 193*cdf0e10cSrcweir //------------------------------------------------------------------------ 194*cdf0e10cSrcweir 195*cdf0e10cSrcweir // inline 196*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderDataAndSetupStgMedium( 197*cdf0e10cSrcweir const sal_Int8* lpStorage, const FORMATETC& fetc, sal_uInt32 nInitStgSize, 198*cdf0e10cSrcweir sal_uInt32 nBytesToTransfer, STGMEDIUM& stgmedium ) 199*cdf0e10cSrcweir { 200*cdf0e10cSrcweir OSL_PRECOND( !nInitStgSize || nInitStgSize && (nInitStgSize >= nBytesToTransfer), 201*cdf0e10cSrcweir "Memory size less than number of bytes to transfer" ); 202*cdf0e10cSrcweir 203*cdf0e10cSrcweir CStgTransferHelper stgTransfHelper( AUTO_INIT ); 204*cdf0e10cSrcweir 205*cdf0e10cSrcweir // setup storage size 206*cdf0e10cSrcweir if ( nInitStgSize > 0 ) 207*cdf0e10cSrcweir stgTransfHelper.init( nInitStgSize, GHND ); 208*cdf0e10cSrcweir 209*cdf0e10cSrcweir #if OSL_DEBUG_LEVEL > 0 210*cdf0e10cSrcweir sal_uInt32 nBytesWritten = 0; 211*cdf0e10cSrcweir stgTransfHelper.write( lpStorage, nBytesToTransfer, &nBytesWritten ); 212*cdf0e10cSrcweir OSL_ASSERT( nBytesWritten == nBytesToTransfer ); 213*cdf0e10cSrcweir #else 214*cdf0e10cSrcweir stgTransfHelper.write( lpStorage, nBytesToTransfer ); 215*cdf0e10cSrcweir #endif 216*cdf0e10cSrcweir 217*cdf0e10cSrcweir setupStgMedium( fetc, stgTransfHelper, stgmedium ); 218*cdf0e10cSrcweir } 219*cdf0e10cSrcweir 220*cdf0e10cSrcweir //------------------------------------------------------------------------ 221*cdf0e10cSrcweir // 222*cdf0e10cSrcweir //------------------------------------------------------------------------ 223*cdf0e10cSrcweir 224*cdf0e10cSrcweir //inline 225*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderLocaleAndSetupStgMedium( 226*cdf0e10cSrcweir FORMATETC& fetc, STGMEDIUM& stgmedium ) 227*cdf0e10cSrcweir { 228*cdf0e10cSrcweir if ( m_FormatRegistrar.hasSynthesizedLocale( ) ) 229*cdf0e10cSrcweir { 230*cdf0e10cSrcweir LCID lcid = m_FormatRegistrar.getSynthesizedLocale( ); 231*cdf0e10cSrcweir renderDataAndSetupStgMedium( 232*cdf0e10cSrcweir reinterpret_cast< sal_Int8* >( &lcid ), 233*cdf0e10cSrcweir fetc, 234*cdf0e10cSrcweir 0, 235*cdf0e10cSrcweir sizeof( LCID ), 236*cdf0e10cSrcweir stgmedium ); 237*cdf0e10cSrcweir } 238*cdf0e10cSrcweir else 239*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_FORMATETC ); 240*cdf0e10cSrcweir } 241*cdf0e10cSrcweir 242*cdf0e10cSrcweir //------------------------------------------------------------------------ 243*cdf0e10cSrcweir // 244*cdf0e10cSrcweir //------------------------------------------------------------------------ 245*cdf0e10cSrcweir 246*cdf0e10cSrcweir //inline 247*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderUnicodeAndSetupStgMedium( 248*cdf0e10cSrcweir FORMATETC& fetc, STGMEDIUM& stgmedium ) 249*cdf0e10cSrcweir { 250*cdf0e10cSrcweir DataFlavor aFlavor = formatEtcToDataFlavor( fetc ); 251*cdf0e10cSrcweir 252*cdf0e10cSrcweir Any aAny = m_XTransferable->getTransferData( aFlavor ); 253*cdf0e10cSrcweir 254*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 255*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 256*cdf0e10cSrcweir // so we must check the any 257*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 258*cdf0e10cSrcweir { 259*cdf0e10cSrcweir OSL_ENSURE( sal_False, "XTransferable should throw an exception if ask for an unsupported flavor" ); 260*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 261*cdf0e10cSrcweir } 262*cdf0e10cSrcweir 263*cdf0e10cSrcweir OUString aText; 264*cdf0e10cSrcweir aAny >>= aText; 265*cdf0e10cSrcweir 266*cdf0e10cSrcweir sal_uInt32 nBytesToTransfer = aText.getLength( ) * sizeof( sal_Unicode ); 267*cdf0e10cSrcweir 268*cdf0e10cSrcweir // to be sure there is an ending 0 269*cdf0e10cSrcweir sal_uInt32 nRequiredMemSize = nBytesToTransfer + sizeof( sal_Unicode ); 270*cdf0e10cSrcweir 271*cdf0e10cSrcweir renderDataAndSetupStgMedium( 272*cdf0e10cSrcweir reinterpret_cast< const sal_Int8* >( aText.getStr( ) ), 273*cdf0e10cSrcweir fetc, 274*cdf0e10cSrcweir nRequiredMemSize, 275*cdf0e10cSrcweir nBytesToTransfer, 276*cdf0e10cSrcweir stgmedium ); 277*cdf0e10cSrcweir } 278*cdf0e10cSrcweir 279*cdf0e10cSrcweir //------------------------------------------------------------------------ 280*cdf0e10cSrcweir // 281*cdf0e10cSrcweir //------------------------------------------------------------------------ 282*cdf0e10cSrcweir 283*cdf0e10cSrcweir //inline 284*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderAnyDataAndSetupStgMedium( 285*cdf0e10cSrcweir FORMATETC& fetc, STGMEDIUM& stgmedium ) 286*cdf0e10cSrcweir { 287*cdf0e10cSrcweir DataFlavor aFlavor = formatEtcToDataFlavor( fetc ); 288*cdf0e10cSrcweir 289*cdf0e10cSrcweir Any aAny = m_XTransferable->getTransferData( aFlavor ); 290*cdf0e10cSrcweir 291*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 292*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 293*cdf0e10cSrcweir // so we must check the any 294*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 295*cdf0e10cSrcweir { 296*cdf0e10cSrcweir OSL_ENSURE( sal_False, "XTransferable should throw an exception if ask for an unsupported flavor" ); 297*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 298*cdf0e10cSrcweir } 299*cdf0e10cSrcweir 300*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 301*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 302*cdf0e10cSrcweir // so we must check the any 303*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 304*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 305*cdf0e10cSrcweir 306*cdf0e10cSrcweir Sequence< sal_Int8 > clipDataStream; 307*cdf0e10cSrcweir aAny >>= clipDataStream; 308*cdf0e10cSrcweir 309*cdf0e10cSrcweir sal_uInt32 nRequiredMemSize = 0; 310*cdf0e10cSrcweir if ( m_DataFormatTranslator.isOemOrAnsiTextFormat( fetc.cfFormat ) ) 311*cdf0e10cSrcweir nRequiredMemSize = sizeof( sal_Int8 ) * clipDataStream.getLength( ) + 1; 312*cdf0e10cSrcweir 313*cdf0e10cSrcweir // prepare data for transmision 314*cdf0e10cSrcweir if ( CF_DIB == fetc.cfFormat ) 315*cdf0e10cSrcweir clipDataStream = OOBmpToWinDIB( clipDataStream ); 316*cdf0e10cSrcweir 317*cdf0e10cSrcweir if ( CF_METAFILEPICT == fetc.cfFormat ) 318*cdf0e10cSrcweir { 319*cdf0e10cSrcweir stgmedium.tymed = TYMED_MFPICT; 320*cdf0e10cSrcweir stgmedium.hMetaFilePict = OOMFPictToWinMFPict( clipDataStream ); 321*cdf0e10cSrcweir stgmedium.pUnkForRelease = NULL; 322*cdf0e10cSrcweir } 323*cdf0e10cSrcweir else if( CF_ENHMETAFILE == fetc.cfFormat ) 324*cdf0e10cSrcweir { 325*cdf0e10cSrcweir stgmedium.tymed = TYMED_ENHMF; 326*cdf0e10cSrcweir stgmedium.hMetaFilePict = OOMFPictToWinENHMFPict( clipDataStream ); 327*cdf0e10cSrcweir stgmedium.pUnkForRelease = NULL; 328*cdf0e10cSrcweir } 329*cdf0e10cSrcweir else 330*cdf0e10cSrcweir renderDataAndSetupStgMedium( 331*cdf0e10cSrcweir clipDataStream.getArray( ), 332*cdf0e10cSrcweir fetc, 333*cdf0e10cSrcweir nRequiredMemSize, 334*cdf0e10cSrcweir clipDataStream.getLength( ), 335*cdf0e10cSrcweir stgmedium ); 336*cdf0e10cSrcweir } 337*cdf0e10cSrcweir 338*cdf0e10cSrcweir //------------------------------------------------------------------------ 339*cdf0e10cSrcweir // 340*cdf0e10cSrcweir //------------------------------------------------------------------------ 341*cdf0e10cSrcweir 342*cdf0e10cSrcweir HRESULT SAL_CALL CXTDataObject::renderSynthesizedFormatAndSetupStgMedium( FORMATETC& fetc, STGMEDIUM& stgmedium ) 343*cdf0e10cSrcweir { 344*cdf0e10cSrcweir HRESULT hr = S_OK; 345*cdf0e10cSrcweir 346*cdf0e10cSrcweir try 347*cdf0e10cSrcweir { 348*cdf0e10cSrcweir if ( CF_UNICODETEXT == fetc.cfFormat ) 349*cdf0e10cSrcweir // the transferable seems to have only text 350*cdf0e10cSrcweir renderSynthesizedUnicodeAndSetupStgMedium( fetc, stgmedium ); 351*cdf0e10cSrcweir else if ( m_DataFormatTranslator.isOemOrAnsiTextFormat( fetc.cfFormat ) ) 352*cdf0e10cSrcweir // the transferable seems to have only unicode text 353*cdf0e10cSrcweir renderSynthesizedTextAndSetupStgMedium( fetc, stgmedium ); 354*cdf0e10cSrcweir else 355*cdf0e10cSrcweir // the transferable seems to have only text/html 356*cdf0e10cSrcweir renderSynthesizedHtmlAndSetupStgMedium( fetc, stgmedium ); 357*cdf0e10cSrcweir } 358*cdf0e10cSrcweir catch(UnsupportedFlavorException&) 359*cdf0e10cSrcweir { 360*cdf0e10cSrcweir hr = DV_E_FORMATETC; 361*cdf0e10cSrcweir } 362*cdf0e10cSrcweir catch( CInvalidFormatEtcException& ) 363*cdf0e10cSrcweir { 364*cdf0e10cSrcweir OSL_ENSURE( sal_False, "Unexpected exception" ); 365*cdf0e10cSrcweir } 366*cdf0e10cSrcweir catch( CStgTransferHelper::CStgTransferException& ex ) 367*cdf0e10cSrcweir { 368*cdf0e10cSrcweir return translateStgExceptionCode( ex.m_hr ); 369*cdf0e10cSrcweir } 370*cdf0e10cSrcweir catch(...) 371*cdf0e10cSrcweir { 372*cdf0e10cSrcweir hr = E_UNEXPECTED; 373*cdf0e10cSrcweir } 374*cdf0e10cSrcweir 375*cdf0e10cSrcweir return hr; 376*cdf0e10cSrcweir } 377*cdf0e10cSrcweir 378*cdf0e10cSrcweir //------------------------------------------------------------------------ 379*cdf0e10cSrcweir // the transferable must have only text, so we will synthesize unicode text 380*cdf0e10cSrcweir //------------------------------------------------------------------------ 381*cdf0e10cSrcweir 382*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderSynthesizedUnicodeAndSetupStgMedium( FORMATETC& fetc, STGMEDIUM& stgmedium ) 383*cdf0e10cSrcweir { 384*cdf0e10cSrcweir OSL_ASSERT( CF_UNICODETEXT == fetc.cfFormat ); 385*cdf0e10cSrcweir 386*cdf0e10cSrcweir Any aAny = m_XTransferable->getTransferData( m_FormatRegistrar.getRegisteredTextFlavor( ) ); 387*cdf0e10cSrcweir 388*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 389*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 390*cdf0e10cSrcweir // so we must check the any 391*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 392*cdf0e10cSrcweir { 393*cdf0e10cSrcweir OSL_ENSURE( sal_False, "XTransferable should throw an exception if ask for an unsupported flavor" ); 394*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 395*cdf0e10cSrcweir } 396*cdf0e10cSrcweir 397*cdf0e10cSrcweir Sequence< sal_Int8 > aText; 398*cdf0e10cSrcweir aAny >>= aText; 399*cdf0e10cSrcweir 400*cdf0e10cSrcweir CStgTransferHelper stgTransfHelper; 401*cdf0e10cSrcweir 402*cdf0e10cSrcweir MultiByteToWideCharEx( 403*cdf0e10cSrcweir m_FormatRegistrar.getRegisteredTextCodePage( ), 404*cdf0e10cSrcweir reinterpret_cast< char* >( aText.getArray( ) ), 405*cdf0e10cSrcweir aText.getLength( ), 406*cdf0e10cSrcweir stgTransfHelper ); 407*cdf0e10cSrcweir 408*cdf0e10cSrcweir setupStgMedium( fetc, stgTransfHelper, stgmedium ); 409*cdf0e10cSrcweir } 410*cdf0e10cSrcweir 411*cdf0e10cSrcweir //------------------------------------------------------------------------ 412*cdf0e10cSrcweir // the transferable must have only unicode text so we will sythesize text 413*cdf0e10cSrcweir //------------------------------------------------------------------------ 414*cdf0e10cSrcweir 415*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderSynthesizedTextAndSetupStgMedium( FORMATETC& fetc, STGMEDIUM& stgmedium ) 416*cdf0e10cSrcweir { 417*cdf0e10cSrcweir OSL_ASSERT( m_DataFormatTranslator.isOemOrAnsiTextFormat( fetc.cfFormat ) ); 418*cdf0e10cSrcweir 419*cdf0e10cSrcweir DataFlavor aFlavor = formatEtcToDataFlavor( 420*cdf0e10cSrcweir m_DataFormatTranslator.getFormatEtcForClipformat( CF_UNICODETEXT ) ); 421*cdf0e10cSrcweir 422*cdf0e10cSrcweir Any aAny = m_XTransferable->getTransferData( aFlavor ); 423*cdf0e10cSrcweir 424*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 425*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 426*cdf0e10cSrcweir // so we must check the any 427*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 428*cdf0e10cSrcweir { 429*cdf0e10cSrcweir OSL_ENSURE( sal_False, "XTransferable should throw an exception if ask for an unsupported flavor" ); 430*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 431*cdf0e10cSrcweir } 432*cdf0e10cSrcweir 433*cdf0e10cSrcweir OUString aUnicodeText; 434*cdf0e10cSrcweir aAny >>= aUnicodeText; 435*cdf0e10cSrcweir 436*cdf0e10cSrcweir CStgTransferHelper stgTransfHelper; 437*cdf0e10cSrcweir 438*cdf0e10cSrcweir WideCharToMultiByteEx( 439*cdf0e10cSrcweir GetACP( ), 440*cdf0e10cSrcweir reinterpret_cast<LPCWSTR>( aUnicodeText.getStr( ) ), 441*cdf0e10cSrcweir aUnicodeText.getLength( ), 442*cdf0e10cSrcweir stgTransfHelper ); 443*cdf0e10cSrcweir 444*cdf0e10cSrcweir setupStgMedium( fetc, stgTransfHelper, stgmedium ); 445*cdf0e10cSrcweir } 446*cdf0e10cSrcweir 447*cdf0e10cSrcweir //------------------------------------------------------------------------ 448*cdf0e10cSrcweir // 449*cdf0e10cSrcweir //------------------------------------------------------------------------ 450*cdf0e10cSrcweir 451*cdf0e10cSrcweir void SAL_CALL CXTDataObject::renderSynthesizedHtmlAndSetupStgMedium( FORMATETC& fetc, STGMEDIUM& stgmedium ) 452*cdf0e10cSrcweir { 453*cdf0e10cSrcweir OSL_ASSERT( m_DataFormatTranslator.isHTMLFormat( fetc.cfFormat ) ); 454*cdf0e10cSrcweir 455*cdf0e10cSrcweir DataFlavor aFlavor; 456*cdf0e10cSrcweir 457*cdf0e10cSrcweir // creating a DataFlavor on the fly 458*cdf0e10cSrcweir aFlavor.MimeType = OUString::createFromAscii( "text/html" ); 459*cdf0e10cSrcweir aFlavor.DataType = getCppuType( (Sequence< sal_Int8 >*)0 ); 460*cdf0e10cSrcweir 461*cdf0e10cSrcweir Any aAny = m_XTransferable->getTransferData( aFlavor ); 462*cdf0e10cSrcweir 463*cdf0e10cSrcweir // unfortunately not all transferables fulfill the 464*cdf0e10cSrcweir // spec. an do throw an UnsupportedFlavorException 465*cdf0e10cSrcweir // so we must check the any 466*cdf0e10cSrcweir if ( !aAny.hasValue( ) ) 467*cdf0e10cSrcweir { 468*cdf0e10cSrcweir OSL_ENSURE( sal_False, "XTransferable should throw an exception if ask for an unsupported flavor" ); 469*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 470*cdf0e10cSrcweir } 471*cdf0e10cSrcweir 472*cdf0e10cSrcweir Sequence< sal_Int8 > aTextHtmlSequence; 473*cdf0e10cSrcweir aAny >>= aTextHtmlSequence; 474*cdf0e10cSrcweir 475*cdf0e10cSrcweir Sequence< sal_Int8 > aHTMLFormatSequence = TextHtmlToHTMLFormat( aTextHtmlSequence ); 476*cdf0e10cSrcweir 477*cdf0e10cSrcweir sal_uInt32 nBytesToTransfer = aHTMLFormatSequence.getLength( ); 478*cdf0e10cSrcweir 479*cdf0e10cSrcweir renderDataAndSetupStgMedium( 480*cdf0e10cSrcweir reinterpret_cast< const sal_Int8* >( aHTMLFormatSequence.getArray( ) ), 481*cdf0e10cSrcweir fetc, 482*cdf0e10cSrcweir 0, 483*cdf0e10cSrcweir nBytesToTransfer, 484*cdf0e10cSrcweir stgmedium ); 485*cdf0e10cSrcweir } 486*cdf0e10cSrcweir 487*cdf0e10cSrcweir //------------------------------------------------------------------------ 488*cdf0e10cSrcweir // IDataObject->EnumFormatEtc 489*cdf0e10cSrcweir //------------------------------------------------------------------------ 490*cdf0e10cSrcweir 491*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::EnumFormatEtc( 492*cdf0e10cSrcweir DWORD dwDirection, IEnumFORMATETC** ppenumFormatetc ) 493*cdf0e10cSrcweir { 494*cdf0e10cSrcweir if ( NULL == ppenumFormatetc ) 495*cdf0e10cSrcweir return E_INVALIDARG; 496*cdf0e10cSrcweir 497*cdf0e10cSrcweir if ( DATADIR_SET == dwDirection ) 498*cdf0e10cSrcweir return E_NOTIMPL; 499*cdf0e10cSrcweir 500*cdf0e10cSrcweir *ppenumFormatetc = NULL; 501*cdf0e10cSrcweir 502*cdf0e10cSrcweir InitializeFormatEtcContainer( ); 503*cdf0e10cSrcweir 504*cdf0e10cSrcweir HRESULT hr; 505*cdf0e10cSrcweir if ( DATADIR_GET == dwDirection ) 506*cdf0e10cSrcweir { 507*cdf0e10cSrcweir *ppenumFormatetc = new CEnumFormatEtc( this, m_FormatEtcContainer ); 508*cdf0e10cSrcweir if ( NULL != *ppenumFormatetc ) 509*cdf0e10cSrcweir static_cast< LPUNKNOWN >( *ppenumFormatetc )->AddRef( ); 510*cdf0e10cSrcweir 511*cdf0e10cSrcweir hr = ( NULL != *ppenumFormatetc ) ? S_OK : E_OUTOFMEMORY; 512*cdf0e10cSrcweir } 513*cdf0e10cSrcweir else 514*cdf0e10cSrcweir hr = E_INVALIDARG; 515*cdf0e10cSrcweir 516*cdf0e10cSrcweir return hr; 517*cdf0e10cSrcweir } 518*cdf0e10cSrcweir 519*cdf0e10cSrcweir //------------------------------------------------------------------------ 520*cdf0e10cSrcweir // IDataObject->QueryGetData 521*cdf0e10cSrcweir //------------------------------------------------------------------------ 522*cdf0e10cSrcweir 523*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::QueryGetData( LPFORMATETC pFormatetc ) 524*cdf0e10cSrcweir { 525*cdf0e10cSrcweir if ( (NULL == pFormatetc) || IsBadReadPtr( pFormatetc, sizeof( FORMATETC ) ) ) 526*cdf0e10cSrcweir return E_INVALIDARG; 527*cdf0e10cSrcweir 528*cdf0e10cSrcweir InitializeFormatEtcContainer( ); 529*cdf0e10cSrcweir 530*cdf0e10cSrcweir return m_FormatEtcContainer.hasFormatEtc( *pFormatetc ) ? S_OK : S_FALSE; 531*cdf0e10cSrcweir } 532*cdf0e10cSrcweir 533*cdf0e10cSrcweir //------------------------------------------------------------------------ 534*cdf0e10cSrcweir // IDataObject->GetDataHere 535*cdf0e10cSrcweir //------------------------------------------------------------------------ 536*cdf0e10cSrcweir 537*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::GetDataHere( LPFORMATETC, LPSTGMEDIUM ) 538*cdf0e10cSrcweir { 539*cdf0e10cSrcweir return E_NOTIMPL; 540*cdf0e10cSrcweir } 541*cdf0e10cSrcweir 542*cdf0e10cSrcweir //------------------------------------------------------------------------ 543*cdf0e10cSrcweir // IDataObject->GetCanonicalFormatEtc 544*cdf0e10cSrcweir //------------------------------------------------------------------------ 545*cdf0e10cSrcweir 546*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::GetCanonicalFormatEtc( LPFORMATETC, LPFORMATETC ) 547*cdf0e10cSrcweir { 548*cdf0e10cSrcweir return E_NOTIMPL; 549*cdf0e10cSrcweir } 550*cdf0e10cSrcweir 551*cdf0e10cSrcweir //------------------------------------------------------------------------ 552*cdf0e10cSrcweir // IDataObject->SetData 553*cdf0e10cSrcweir //------------------------------------------------------------------------ 554*cdf0e10cSrcweir 555*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::SetData( LPFORMATETC, LPSTGMEDIUM, BOOL ) 556*cdf0e10cSrcweir { 557*cdf0e10cSrcweir return E_NOTIMPL; 558*cdf0e10cSrcweir } 559*cdf0e10cSrcweir 560*cdf0e10cSrcweir //------------------------------------------------------------------------ 561*cdf0e10cSrcweir // IDataObject->DAdvise 562*cdf0e10cSrcweir //------------------------------------------------------------------------ 563*cdf0e10cSrcweir 564*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::DAdvise( LPFORMATETC, DWORD, LPADVISESINK, DWORD * ) 565*cdf0e10cSrcweir { 566*cdf0e10cSrcweir return E_NOTIMPL; 567*cdf0e10cSrcweir } 568*cdf0e10cSrcweir 569*cdf0e10cSrcweir //------------------------------------------------------------------------ 570*cdf0e10cSrcweir // IDataObject->DUnadvise 571*cdf0e10cSrcweir //------------------------------------------------------------------------ 572*cdf0e10cSrcweir 573*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::DUnadvise( DWORD ) 574*cdf0e10cSrcweir { 575*cdf0e10cSrcweir return E_NOTIMPL; 576*cdf0e10cSrcweir } 577*cdf0e10cSrcweir 578*cdf0e10cSrcweir //------------------------------------------------------------------------ 579*cdf0e10cSrcweir // IDataObject->EnumDAdvise 580*cdf0e10cSrcweir //------------------------------------------------------------------------ 581*cdf0e10cSrcweir 582*cdf0e10cSrcweir STDMETHODIMP CXTDataObject::EnumDAdvise( LPENUMSTATDATA * ) 583*cdf0e10cSrcweir { 584*cdf0e10cSrcweir return E_NOTIMPL; 585*cdf0e10cSrcweir } 586*cdf0e10cSrcweir 587*cdf0e10cSrcweir //------------------------------------------------------------------------ 588*cdf0e10cSrcweir // for our convenience 589*cdf0e10cSrcweir //------------------------------------------------------------------------ 590*cdf0e10cSrcweir 591*cdf0e10cSrcweir CXTDataObject::operator IDataObject*( ) 592*cdf0e10cSrcweir { 593*cdf0e10cSrcweir return static_cast< IDataObject* >( this ); 594*cdf0e10cSrcweir } 595*cdf0e10cSrcweir 596*cdf0e10cSrcweir //------------------------------------------------------------------------ 597*cdf0e10cSrcweir // 598*cdf0e10cSrcweir //------------------------------------------------------------------------ 599*cdf0e10cSrcweir 600*cdf0e10cSrcweir inline 601*cdf0e10cSrcweir DataFlavor SAL_CALL CXTDataObject::formatEtcToDataFlavor( const FORMATETC& aFormatEtc ) const 602*cdf0e10cSrcweir { 603*cdf0e10cSrcweir DataFlavor aFlavor; 604*cdf0e10cSrcweir 605*cdf0e10cSrcweir if ( m_FormatRegistrar.hasSynthesizedLocale( ) ) 606*cdf0e10cSrcweir aFlavor = 607*cdf0e10cSrcweir m_DataFormatTranslator.getDataFlavorFromFormatEtc( aFormatEtc, m_FormatRegistrar.getSynthesizedLocale( ) ); 608*cdf0e10cSrcweir else 609*cdf0e10cSrcweir aFlavor = m_DataFormatTranslator.getDataFlavorFromFormatEtc( aFormatEtc ); 610*cdf0e10cSrcweir 611*cdf0e10cSrcweir if ( !aFlavor.MimeType.getLength( ) ) 612*cdf0e10cSrcweir throw UnsupportedFlavorException( ); 613*cdf0e10cSrcweir 614*cdf0e10cSrcweir return aFlavor; 615*cdf0e10cSrcweir } 616*cdf0e10cSrcweir 617*cdf0e10cSrcweir //------------------------------------------------------------------------ 618*cdf0e10cSrcweir // 619*cdf0e10cSrcweir //------------------------------------------------------------------------ 620*cdf0e10cSrcweir 621*cdf0e10cSrcweir inline 622*cdf0e10cSrcweir void CXTDataObject::validateFormatEtc( LPFORMATETC lpFormatEtc ) const 623*cdf0e10cSrcweir { 624*cdf0e10cSrcweir OSL_ASSERT( lpFormatEtc ); 625*cdf0e10cSrcweir 626*cdf0e10cSrcweir if ( lpFormatEtc->lindex != -1 ) 627*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_LINDEX ); 628*cdf0e10cSrcweir 629*cdf0e10cSrcweir if ( !(lpFormatEtc->dwAspect & DVASPECT_CONTENT) && 630*cdf0e10cSrcweir !(lpFormatEtc->dwAspect & DVASPECT_SHORTNAME) ) 631*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_DVASPECT ); 632*cdf0e10cSrcweir 633*cdf0e10cSrcweir if ( !(lpFormatEtc->tymed & TYMED_HGLOBAL) && 634*cdf0e10cSrcweir !(lpFormatEtc->tymed & TYMED_ISTREAM) && 635*cdf0e10cSrcweir !(lpFormatEtc->tymed & TYMED_MFPICT) && 636*cdf0e10cSrcweir !(lpFormatEtc->tymed & TYMED_ENHMF) ) 637*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_TYMED ); 638*cdf0e10cSrcweir 639*cdf0e10cSrcweir if ( lpFormatEtc->cfFormat == CF_METAFILEPICT && 640*cdf0e10cSrcweir !(lpFormatEtc->tymed & TYMED_MFPICT) ) 641*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_TYMED ); 642*cdf0e10cSrcweir 643*cdf0e10cSrcweir if ( lpFormatEtc->cfFormat == CF_ENHMETAFILE && 644*cdf0e10cSrcweir !(lpFormatEtc->tymed & TYMED_ENHMF) ) 645*cdf0e10cSrcweir throw CInvalidFormatEtcException( DV_E_TYMED ); 646*cdf0e10cSrcweir } 647*cdf0e10cSrcweir 648*cdf0e10cSrcweir //------------------------------------------------------------------------ 649*cdf0e10cSrcweir // 650*cdf0e10cSrcweir //------------------------------------------------------------------------ 651*cdf0e10cSrcweir 652*cdf0e10cSrcweir //inline 653*cdf0e10cSrcweir void SAL_CALL CXTDataObject::setupStgMedium( const FORMATETC& fetc, 654*cdf0e10cSrcweir CStgTransferHelper& stgTransHlp, 655*cdf0e10cSrcweir STGMEDIUM& stgmedium ) 656*cdf0e10cSrcweir { 657*cdf0e10cSrcweir stgmedium.pUnkForRelease = NULL; 658*cdf0e10cSrcweir 659*cdf0e10cSrcweir if ( fetc.cfFormat == CF_METAFILEPICT ) 660*cdf0e10cSrcweir { 661*cdf0e10cSrcweir stgmedium.tymed = TYMED_MFPICT; 662*cdf0e10cSrcweir stgmedium.hMetaFilePict = static_cast< HMETAFILEPICT >( stgTransHlp.getHGlobal( ) ); 663*cdf0e10cSrcweir } 664*cdf0e10cSrcweir else if ( fetc.cfFormat == CF_ENHMETAFILE ) 665*cdf0e10cSrcweir { 666*cdf0e10cSrcweir stgmedium.tymed = TYMED_ENHMF; 667*cdf0e10cSrcweir stgmedium.hEnhMetaFile = static_cast< HENHMETAFILE >( stgTransHlp.getHGlobal( ) ); 668*cdf0e10cSrcweir } 669*cdf0e10cSrcweir else if ( fetc.tymed & TYMED_HGLOBAL ) 670*cdf0e10cSrcweir { 671*cdf0e10cSrcweir stgmedium.tymed = TYMED_HGLOBAL; 672*cdf0e10cSrcweir stgmedium.hGlobal = stgTransHlp.getHGlobal( ); 673*cdf0e10cSrcweir } 674*cdf0e10cSrcweir else if ( fetc.tymed & TYMED_ISTREAM ) 675*cdf0e10cSrcweir { 676*cdf0e10cSrcweir stgmedium.tymed = TYMED_ISTREAM; 677*cdf0e10cSrcweir stgTransHlp.getIStream( &stgmedium.pstm ); 678*cdf0e10cSrcweir } 679*cdf0e10cSrcweir else 680*cdf0e10cSrcweir OSL_ASSERT( sal_False ); 681*cdf0e10cSrcweir } 682*cdf0e10cSrcweir 683*cdf0e10cSrcweir //------------------------------------------------------------------------ 684*cdf0e10cSrcweir // 685*cdf0e10cSrcweir //------------------------------------------------------------------------ 686*cdf0e10cSrcweir 687*cdf0e10cSrcweir inline 688*cdf0e10cSrcweir void SAL_CALL CXTDataObject::invalidateStgMedium( STGMEDIUM& stgmedium ) const 689*cdf0e10cSrcweir { 690*cdf0e10cSrcweir stgmedium.tymed = TYMED_NULL; 691*cdf0e10cSrcweir } 692*cdf0e10cSrcweir 693*cdf0e10cSrcweir //------------------------------------------------------------------------ 694*cdf0e10cSrcweir // 695*cdf0e10cSrcweir //------------------------------------------------------------------------ 696*cdf0e10cSrcweir 697*cdf0e10cSrcweir inline 698*cdf0e10cSrcweir HRESULT SAL_CALL CXTDataObject::translateStgExceptionCode( HRESULT hr ) const 699*cdf0e10cSrcweir { 700*cdf0e10cSrcweir HRESULT hrTransl; 701*cdf0e10cSrcweir 702*cdf0e10cSrcweir switch( hr ) 703*cdf0e10cSrcweir { 704*cdf0e10cSrcweir case STG_E_MEDIUMFULL: 705*cdf0e10cSrcweir hrTransl = hr; 706*cdf0e10cSrcweir break; 707*cdf0e10cSrcweir 708*cdf0e10cSrcweir default: 709*cdf0e10cSrcweir hrTransl = E_UNEXPECTED; 710*cdf0e10cSrcweir break; 711*cdf0e10cSrcweir } 712*cdf0e10cSrcweir 713*cdf0e10cSrcweir return hrTransl; 714*cdf0e10cSrcweir } 715*cdf0e10cSrcweir 716*cdf0e10cSrcweir //------------------------------------------------------------------------ 717*cdf0e10cSrcweir // 718*cdf0e10cSrcweir //------------------------------------------------------------------------ 719*cdf0e10cSrcweir 720*cdf0e10cSrcweir inline void SAL_CALL CXTDataObject::InitializeFormatEtcContainer( ) 721*cdf0e10cSrcweir { 722*cdf0e10cSrcweir if ( !m_bFormatEtcContainerInitialized ) 723*cdf0e10cSrcweir { 724*cdf0e10cSrcweir m_FormatRegistrar.RegisterFormats( m_XTransferable, m_FormatEtcContainer ); 725*cdf0e10cSrcweir m_bFormatEtcContainerInitialized = sal_True; 726*cdf0e10cSrcweir } 727*cdf0e10cSrcweir } 728*cdf0e10cSrcweir 729*cdf0e10cSrcweir //============================================================================ 730*cdf0e10cSrcweir // CEnumFormatEtc 731*cdf0e10cSrcweir //============================================================================ 732*cdf0e10cSrcweir 733*cdf0e10cSrcweir //---------------------------------------------------------------------------- 734*cdf0e10cSrcweir // ctor 735*cdf0e10cSrcweir //---------------------------------------------------------------------------- 736*cdf0e10cSrcweir 737*cdf0e10cSrcweir CEnumFormatEtc::CEnumFormatEtc( LPUNKNOWN lpUnkOuter, const CFormatEtcContainer& aFormatEtcContainer ) : 738*cdf0e10cSrcweir m_nRefCnt( 0 ), 739*cdf0e10cSrcweir m_lpUnkOuter( lpUnkOuter ), 740*cdf0e10cSrcweir m_FormatEtcContainer( aFormatEtcContainer ) 741*cdf0e10cSrcweir { 742*cdf0e10cSrcweir Reset( ); 743*cdf0e10cSrcweir } 744*cdf0e10cSrcweir 745*cdf0e10cSrcweir //---------------------------------------------------------------------------- 746*cdf0e10cSrcweir // IUnknown->QueryInterface 747*cdf0e10cSrcweir //---------------------------------------------------------------------------- 748*cdf0e10cSrcweir 749*cdf0e10cSrcweir STDMETHODIMP CEnumFormatEtc::QueryInterface( REFIID iid, LPVOID* ppvObject ) 750*cdf0e10cSrcweir { 751*cdf0e10cSrcweir if ( NULL == ppvObject ) 752*cdf0e10cSrcweir return E_INVALIDARG; 753*cdf0e10cSrcweir 754*cdf0e10cSrcweir HRESULT hr = E_NOINTERFACE; 755*cdf0e10cSrcweir 756*cdf0e10cSrcweir *ppvObject = NULL; 757*cdf0e10cSrcweir 758*cdf0e10cSrcweir if ( ( __uuidof( IUnknown ) == iid ) || 759*cdf0e10cSrcweir ( __uuidof( IEnumFORMATETC ) == iid ) ) 760*cdf0e10cSrcweir { 761*cdf0e10cSrcweir *ppvObject = static_cast< IUnknown* >( this ); 762*cdf0e10cSrcweir static_cast< LPUNKNOWN >( *ppvObject )->AddRef( ); 763*cdf0e10cSrcweir hr = S_OK; 764*cdf0e10cSrcweir } 765*cdf0e10cSrcweir 766*cdf0e10cSrcweir return hr; 767*cdf0e10cSrcweir } 768*cdf0e10cSrcweir 769*cdf0e10cSrcweir //---------------------------------------------------------------------------- 770*cdf0e10cSrcweir // IUnknown->AddRef 771*cdf0e10cSrcweir //---------------------------------------------------------------------------- 772*cdf0e10cSrcweir 773*cdf0e10cSrcweir STDMETHODIMP_(ULONG) CEnumFormatEtc::AddRef( ) 774*cdf0e10cSrcweir { 775*cdf0e10cSrcweir // keep the dataobject alive 776*cdf0e10cSrcweir m_lpUnkOuter->AddRef( ); 777*cdf0e10cSrcweir return InterlockedIncrement( &m_nRefCnt ); 778*cdf0e10cSrcweir } 779*cdf0e10cSrcweir 780*cdf0e10cSrcweir //---------------------------------------------------------------------------- 781*cdf0e10cSrcweir // IUnknown->Release 782*cdf0e10cSrcweir //---------------------------------------------------------------------------- 783*cdf0e10cSrcweir 784*cdf0e10cSrcweir STDMETHODIMP_(ULONG) CEnumFormatEtc::Release( ) 785*cdf0e10cSrcweir { 786*cdf0e10cSrcweir // release the outer dataobject 787*cdf0e10cSrcweir m_lpUnkOuter->Release( ); 788*cdf0e10cSrcweir 789*cdf0e10cSrcweir ULONG nRefCnt = InterlockedDecrement( &m_nRefCnt ); 790*cdf0e10cSrcweir if ( 0 == nRefCnt ) 791*cdf0e10cSrcweir delete this; 792*cdf0e10cSrcweir 793*cdf0e10cSrcweir return nRefCnt; 794*cdf0e10cSrcweir } 795*cdf0e10cSrcweir 796*cdf0e10cSrcweir //---------------------------------------------------------------------------- 797*cdf0e10cSrcweir // IEnumFORMATETC->Next 798*cdf0e10cSrcweir //---------------------------------------------------------------------------- 799*cdf0e10cSrcweir 800*cdf0e10cSrcweir STDMETHODIMP CEnumFormatEtc::Next( ULONG nRequested, LPFORMATETC lpDest, ULONG* lpFetched ) 801*cdf0e10cSrcweir { 802*cdf0e10cSrcweir if ( ( nRequested < 1 ) || 803*cdf0e10cSrcweir (( nRequested > 1 ) && ( NULL == lpFetched )) || 804*cdf0e10cSrcweir IsBadWritePtr( lpDest, sizeof( FORMATETC ) * nRequested ) ) 805*cdf0e10cSrcweir return E_INVALIDARG; 806*cdf0e10cSrcweir 807*cdf0e10cSrcweir sal_uInt32 nFetched = m_FormatEtcContainer.nextFormatEtc( lpDest, nRequested ); 808*cdf0e10cSrcweir 809*cdf0e10cSrcweir if ( NULL != lpFetched ) 810*cdf0e10cSrcweir *lpFetched = nFetched; 811*cdf0e10cSrcweir 812*cdf0e10cSrcweir return (nFetched == nRequested) ? S_OK : S_FALSE; 813*cdf0e10cSrcweir } 814*cdf0e10cSrcweir 815*cdf0e10cSrcweir //---------------------------------------------------------------------------- 816*cdf0e10cSrcweir // IEnumFORMATETC->Skip 817*cdf0e10cSrcweir //---------------------------------------------------------------------------- 818*cdf0e10cSrcweir 819*cdf0e10cSrcweir STDMETHODIMP CEnumFormatEtc::Skip( ULONG celt ) 820*cdf0e10cSrcweir { 821*cdf0e10cSrcweir return m_FormatEtcContainer.skipFormatEtc( celt ) ? S_OK : S_FALSE; 822*cdf0e10cSrcweir } 823*cdf0e10cSrcweir 824*cdf0e10cSrcweir //---------------------------------------------------------------------------- 825*cdf0e10cSrcweir // IEnumFORMATETC->Reset 826*cdf0e10cSrcweir //---------------------------------------------------------------------------- 827*cdf0e10cSrcweir 828*cdf0e10cSrcweir STDMETHODIMP CEnumFormatEtc::Reset( ) 829*cdf0e10cSrcweir { 830*cdf0e10cSrcweir m_FormatEtcContainer.beginEnumFormatEtc( ); 831*cdf0e10cSrcweir return S_OK; 832*cdf0e10cSrcweir } 833*cdf0e10cSrcweir 834*cdf0e10cSrcweir //---------------------------------------------------------------------------- 835*cdf0e10cSrcweir // IEnumFORMATETC->Clone 836*cdf0e10cSrcweir //---------------------------------------------------------------------------- 837*cdf0e10cSrcweir 838*cdf0e10cSrcweir STDMETHODIMP CEnumFormatEtc::Clone( IEnumFORMATETC** ppenum ) 839*cdf0e10cSrcweir { 840*cdf0e10cSrcweir if ( NULL == ppenum ) 841*cdf0e10cSrcweir return E_INVALIDARG; 842*cdf0e10cSrcweir 843*cdf0e10cSrcweir *ppenum = new CEnumFormatEtc( m_lpUnkOuter, m_FormatEtcContainer ); 844*cdf0e10cSrcweir if ( NULL != ppenum ) 845*cdf0e10cSrcweir static_cast< LPUNKNOWN >( *ppenum )->AddRef( ); 846*cdf0e10cSrcweir 847*cdf0e10cSrcweir return ( NULL != *ppenum ) ? S_OK : E_OUTOFMEMORY; 848*cdf0e10cSrcweir } 849