xref: /trunk/main/svtools/source/misc/transfer2.cxx (revision cdf0e10c4e3984b49a9502b011690b615761d4a3)
1 /*************************************************************************
2  *
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * Copyright 2000, 2010 Oracle and/or its affiliates.
6  *
7  * OpenOffice.org - a multi-platform office productivity suite
8  *
9  * This file is part of OpenOffice.org.
10  *
11  * OpenOffice.org is free software: you can redistribute it and/or modify
12  * it under the terms of the GNU Lesser General Public License version 3
13  * only, as published by the Free Software Foundation.
14  *
15  * OpenOffice.org is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU Lesser General Public License version 3 for more details
19  * (a copy is included in the LICENSE file that accompanied this code).
20  *
21  * You should have received a copy of the GNU Lesser General Public License
22  * version 3 along with OpenOffice.org.  If not, see
23  * <http://www.openoffice.org/license.html>
24  * for a copy of the LGPLv3 License.
25  *
26  ************************************************************************/
27 
28 // MARKER(update_precomp.py): autogen include statement, do not remove
29 #include "precompiled_svtools.hxx"
30 #include <vos/mutex.hxx>
31 #include <tools/debug.hxx>
32 #include <tools/urlobj.hxx>
33 #include <unotools/ucbstreamhelper.hxx>
34 #include <sot/exchange.hxx>
35 #include <sot/storage.hxx>
36 #include <vcl/bitmap.hxx>
37 #include <vcl/gdimtf.hxx>
38 #include <vcl/graph.hxx>
39 #include <vcl/svapp.hxx>
40 #include <vcl/window.hxx>
41 #include <comphelper/processfactory.hxx>
42 #include <com/sun/star/datatransfer/dnd/XDropTargetDragContext.hpp>
43 #include "svl/urlbmk.hxx"
44 #include <svtools/inetimg.hxx>
45 #include <svtools/imap.hxx>
46 #include <svtools/transfer.hxx>
47 
48 // --------------
49 // - Namespaces -
50 // --------------
51 
52 using namespace ::com::sun::star::uno;
53 using namespace ::com::sun::star::lang;
54 using namespace ::com::sun::star::io;
55 using namespace ::com::sun::star::datatransfer;
56 using namespace ::com::sun::star::datatransfer::clipboard;
57 using namespace ::com::sun::star::datatransfer::dnd;
58 
59 // -----------------------------------------
60 // - DragSourceHelper::DragGestureListener -
61 // -----------------------------------------
62 
63 DragSourceHelper::DragGestureListener::DragGestureListener( DragSourceHelper& rDragSourceHelper ) :
64     mrParent( rDragSourceHelper )
65 {
66 }
67 
68 // -----------------------------------------------------------------------------
69 
70 DragSourceHelper::DragGestureListener::~DragGestureListener()
71 {
72 }
73 
74 // -----------------------------------------------------------------------------
75 
76 void SAL_CALL DragSourceHelper::DragGestureListener::disposing( const EventObject& ) throw( RuntimeException )
77 {
78 }
79 
80 // -----------------------------------------------------------------------------
81 
82 void SAL_CALL DragSourceHelper::DragGestureListener::dragGestureRecognized( const DragGestureEvent& rDGE ) throw( RuntimeException )
83 {
84     const ::vos::OGuard aGuard( Application::GetSolarMutex() );
85 
86     const Point aPtPixel( rDGE.DragOriginX, rDGE.DragOriginY );
87     mrParent.StartDrag( rDGE.DragAction, aPtPixel );
88 }
89 
90 // --------------------
91 // - DragSourceHelper -
92 // --------------------
93 
94 DragSourceHelper::DragSourceHelper( Window* pWindow ) :
95     mxDragGestureRecognizer( pWindow->GetDragGestureRecognizer() )
96 {
97     if( mxDragGestureRecognizer.is() )
98     {
99         mxDragGestureListener = new DragSourceHelper::DragGestureListener( *this );
100         mxDragGestureRecognizer->addDragGestureListener( mxDragGestureListener );
101     }
102 }
103 
104 // -----------------------------------------------------------------------------
105 
106 DragSourceHelper::~DragSourceHelper()
107 {
108     if( mxDragGestureRecognizer.is()  )
109         mxDragGestureRecognizer->removeDragGestureListener( mxDragGestureListener );
110 }
111 
112 // -----------------------------------------------------------------------------
113 
114 void DragSourceHelper::StartDrag( sal_Int8, const Point& )
115 {
116 }
117 
118 // ----------------------------------------
119 // - DropTargetHelper::DropTargetListener -
120 // ----------------------------------------
121 
122 DropTargetHelper::DropTargetListener::DropTargetListener( DropTargetHelper& rDropTargetHelper ) :
123     mrParent( rDropTargetHelper ),
124     mpLastDragOverEvent( NULL )
125 {
126 }
127 
128 // -----------------------------------------------------------------------------
129 
130 DropTargetHelper::DropTargetListener::~DropTargetListener()
131 {
132     delete mpLastDragOverEvent;
133 }
134 
135 // -----------------------------------------------------------------------------
136 
137 void SAL_CALL DropTargetHelper::DropTargetListener::disposing( const EventObject& ) throw( RuntimeException )
138 {
139 }
140 
141 // -----------------------------------------------------------------------------
142 
143 void SAL_CALL DropTargetHelper::DropTargetListener::drop( const DropTargetDropEvent& rDTDE ) throw( RuntimeException )
144 {
145     const ::vos::OGuard aGuard( Application::GetSolarMutex() );
146 
147     try
148     {
149         AcceptDropEvent  aAcceptEvent;
150         ExecuteDropEvent aExecuteEvt( rDTDE.DropAction & ~DNDConstants::ACTION_DEFAULT, Point( rDTDE.LocationX, rDTDE.LocationY ), rDTDE );
151         sal_Int8         nRet = DNDConstants::ACTION_NONE;
152 
153         aExecuteEvt.mbDefault = ( ( rDTDE.DropAction & DNDConstants::ACTION_DEFAULT ) != 0 );
154 
155         // in case of a default action, call ::AcceptDrop first and use the returned
156         // accepted action as the execute action in the call to ::ExecuteDrop
157         aAcceptEvent.mnAction = aExecuteEvt.mnAction;
158         aAcceptEvent.maPosPixel = aExecuteEvt.maPosPixel;
159         (DropTargetEvent&)( aAcceptEvent.maDragEvent ) = (DropTargetEvent&) rDTDE;
160         ( (DropTargetDragEvent&)( aAcceptEvent.maDragEvent ) ).DropAction = rDTDE.DropAction;
161         ( (DropTargetDragEvent&)( aAcceptEvent.maDragEvent ) ).LocationX = rDTDE.LocationX;
162         ( (DropTargetDragEvent&)( aAcceptEvent.maDragEvent ) ).LocationY = rDTDE.LocationY;
163         ( (DropTargetDragEvent&)( aAcceptEvent.maDragEvent ) ).SourceActions = rDTDE.SourceActions;
164         aAcceptEvent.mbLeaving = sal_False;
165         aAcceptEvent.mbDefault = aExecuteEvt.mbDefault;
166 
167         nRet = mrParent.AcceptDrop( aAcceptEvent );
168 
169         if( DNDConstants::ACTION_NONE != nRet )
170         {
171             rDTDE.Context->acceptDrop( nRet );
172 
173             if( aExecuteEvt.mbDefault )
174                 aExecuteEvt.mnAction = nRet;
175 
176             nRet = mrParent.ExecuteDrop( aExecuteEvt );
177         }
178 
179         rDTDE.Context->dropComplete( DNDConstants::ACTION_NONE != nRet );
180 
181         if( mpLastDragOverEvent )
182         {
183             delete mpLastDragOverEvent;
184             mpLastDragOverEvent = NULL;
185         }
186     }
187     catch( const ::com::sun::star::uno::Exception& )
188     {
189     }
190 }
191 
192 // -----------------------------------------------------------------------------
193 
194 void SAL_CALL DropTargetHelper::DropTargetListener::dragEnter( const DropTargetDragEnterEvent& rDTDEE ) throw( RuntimeException )
195 {
196     const ::vos::OGuard aGuard( Application::GetSolarMutex() );
197 
198     try
199     {
200         mrParent.ImplBeginDrag( rDTDEE.SupportedDataFlavors );
201     }
202     catch( const ::com::sun::star::uno::Exception& )
203     {
204     }
205 
206     dragOver( rDTDEE );
207 }
208 
209 // -----------------------------------------------------------------------------
210 
211 void SAL_CALL DropTargetHelper::DropTargetListener::dragOver( const DropTargetDragEvent& rDTDE ) throw( RuntimeException )
212 {
213     const ::vos::OGuard aGuard( Application::GetSolarMutex() );
214 
215     try
216     {
217         if( mpLastDragOverEvent )
218             delete mpLastDragOverEvent;
219 
220         mpLastDragOverEvent = new AcceptDropEvent( rDTDE.DropAction & ~DNDConstants::ACTION_DEFAULT, Point( rDTDE.LocationX, rDTDE.LocationY ), rDTDE );
221         mpLastDragOverEvent->mbDefault = ( ( rDTDE.DropAction & DNDConstants::ACTION_DEFAULT ) != 0 );
222 
223         const sal_Int8 nRet = mrParent.AcceptDrop( *mpLastDragOverEvent );
224 
225         if( DNDConstants::ACTION_NONE == nRet )
226             rDTDE.Context->rejectDrag();
227         else
228             rDTDE.Context->acceptDrag( nRet );
229     }
230     catch( const ::com::sun::star::uno::Exception& )
231     {
232     }
233 }
234 
235 // -----------------------------------------------------------------------------
236 
237 void SAL_CALL DropTargetHelper::DropTargetListener::dragExit( const DropTargetEvent& ) throw( RuntimeException )
238 {
239     const ::vos::OGuard aGuard( Application::GetSolarMutex() );
240 
241     try
242     {
243         if( mpLastDragOverEvent )
244         {
245             mpLastDragOverEvent->mbLeaving = sal_True;
246             mrParent.AcceptDrop( *mpLastDragOverEvent );
247             delete mpLastDragOverEvent;
248             mpLastDragOverEvent = NULL;
249         }
250 
251         mrParent.ImplEndDrag();
252     }
253     catch( const ::com::sun::star::uno::Exception& )
254     {
255     }
256 }
257 
258 
259 // -----------------------------------------------------------------------------
260 
261 void SAL_CALL DropTargetHelper::DropTargetListener::dropActionChanged( const DropTargetDragEvent& ) throw( RuntimeException )
262 {
263 }
264 
265 // --------------------
266 // - DropTargetHelper -
267 // --------------------
268 
269 DropTargetHelper::DropTargetHelper( Window* pWindow ) :
270     mxDropTarget( pWindow->GetDropTarget() ),
271     mpFormats( new DataFlavorExVector )
272 {
273     ImplConstruct();
274 }
275 
276 // -----------------------------------------------------------------------------
277 
278 DropTargetHelper::DropTargetHelper( const Reference< XDropTarget >& rxDropTarget ) :
279     mxDropTarget( rxDropTarget ),
280     mpFormats( new DataFlavorExVector )
281 {
282     ImplConstruct();
283 }
284 
285 // -----------------------------------------------------------------------------
286 
287 DropTargetHelper::~DropTargetHelper()
288 {
289     if( mxDropTarget.is() )
290         mxDropTarget->removeDropTargetListener( mxDropTargetListener );
291 
292     delete mpFormats;
293 }
294 
295 // -----------------------------------------------------------------------------
296 
297 void DropTargetHelper::ImplConstruct()
298 {
299     if( mxDropTarget.is() )
300     {
301         mxDropTargetListener = new DropTargetHelper::DropTargetListener( *this );
302         mxDropTarget->addDropTargetListener( mxDropTargetListener );
303         mxDropTarget->setActive( sal_True );
304     }
305 }
306 
307 // -----------------------------------------------------------------------------
308 
309 void DropTargetHelper::ImplBeginDrag( const Sequence< DataFlavor >& rSupportedDataFlavors )
310 {
311     mpFormats->clear();
312     TransferableDataHelper::FillDataFlavorExVector( rSupportedDataFlavors, *mpFormats );
313 }
314 
315 // -----------------------------------------------------------------------------
316 
317 void DropTargetHelper::ImplEndDrag()
318 {
319     mpFormats->clear();
320 }
321 
322 // -----------------------------------------------------------------------------
323 
324 sal_Int8 DropTargetHelper::AcceptDrop( const AcceptDropEvent& )
325 {
326     return( DNDConstants::ACTION_NONE );
327 }
328 
329 // -----------------------------------------------------------------------------
330 
331 sal_Int8 DropTargetHelper::ExecuteDrop( const ExecuteDropEvent& )
332 {
333     return( DNDConstants::ACTION_NONE );
334 }
335 
336 // -----------------------------------------------------------------------------
337 
338 sal_Bool DropTargetHelper::IsDropFormatSupported( SotFormatStringId nFormat )
339 {
340     DataFlavorExVector::iterator    aIter( mpFormats->begin() ), aEnd( mpFormats->end() );
341     sal_Bool                        bRet = sal_False;
342 
343     while( aIter != aEnd )
344     {
345         if( nFormat == (*aIter++).mnSotId )
346         {
347             bRet = sal_True;
348             aIter = aEnd;
349         }
350     }
351 
352     return bRet;
353 }
354 
355 // -----------------------------------------------------------------------------
356 
357 sal_Bool DropTargetHelper::IsDropFormatSupported( const DataFlavor& rFlavor )
358 {
359     DataFlavorExVector::iterator    aIter( mpFormats->begin() ), aEnd( mpFormats->end() );
360     sal_Bool                        bRet = sal_False;
361 
362     while( aIter != aEnd )
363     {
364         if( TransferableDataHelper::IsEqual( rFlavor, *aIter++ ) )
365         {
366             bRet = sal_True;
367             aIter = aEnd;
368         }
369     }
370 
371     return bRet;
372 }
373 
374 // -----------------------------------------------------------------------------
375 // TransferDataContainer
376 // -----------------------------------------------------------------------------
377 
378 struct TDataCntnrEntry_Impl
379 {
380     ::com::sun::star::uno::Any aAny;
381     SotFormatStringId nId;
382 };
383 
384 // -----------------------------------------------------------------------------
385 
386 typedef ::std::list< TDataCntnrEntry_Impl > TDataCntnrEntryList;
387 
388 // -----------------------------------------------------------------------------
389 
390 struct TransferDataContainer_Impl
391 {
392     TDataCntnrEntryList aFmtList;
393     Link aFinshedLnk;
394     INetBookmark* pBookmk;
395     Graphic* pGrf;
396 
397     TransferDataContainer_Impl()
398         : pBookmk( 0 ), pGrf( 0 )
399     {
400     }
401 
402     ~TransferDataContainer_Impl()
403     {
404         delete pBookmk;
405         delete pGrf;
406     }
407 };
408 
409 // -----------------------------------------------------------------------------
410 
411 TransferDataContainer::TransferDataContainer()
412     : pImpl( new TransferDataContainer_Impl )
413 {
414 }
415 
416 // -----------------------------------------------------------------------------
417 
418 TransferDataContainer::~TransferDataContainer()
419 {
420     delete pImpl;
421 }
422 
423 // -----------------------------------------------------------------------------
424 
425 void TransferDataContainer::AddSupportedFormats()
426 {
427 }
428 
429 // -----------------------------------------------------------------------------
430 
431 sal_Bool TransferDataContainer::GetData( const
432             ::com::sun::star::datatransfer::DataFlavor& rFlavor )
433 {
434     TDataCntnrEntryList::iterator   aIter( pImpl->aFmtList.begin() ),
435                                     aEnd( pImpl->aFmtList.end() );
436     sal_Bool bFnd = sal_False;
437     sal_uLong nFmtId = SotExchange::GetFormat( rFlavor );
438 
439     // test first the list
440     for( ; aIter != aEnd; ++aIter )
441     {
442         TDataCntnrEntry_Impl& rEntry = (TDataCntnrEntry_Impl&)*aIter;
443         if( nFmtId == rEntry.nId )
444         {
445             bFnd = SetAny( rEntry.aAny, rFlavor );
446             break;
447         }
448     }
449 
450     // test second the bookmark pointer
451     if( !bFnd )
452         switch( nFmtId )
453         {
454         case SOT_FORMAT_STRING:
455         case SOT_FORMATSTR_ID_SOLK:
456         case SOT_FORMATSTR_ID_NETSCAPE_BOOKMARK:
457         case SOT_FORMATSTR_ID_FILECONTENT:
458         case SOT_FORMATSTR_ID_FILEGRPDESCRIPTOR:
459         case SOT_FORMATSTR_ID_UNIFORMRESOURCELOCATOR:
460             if( pImpl->pBookmk )
461                 bFnd = SetINetBookmark( *pImpl->pBookmk, rFlavor );
462             break;
463 
464         case SOT_FORMATSTR_ID_SVXB:
465         case SOT_FORMAT_BITMAP:
466         case SOT_FORMAT_GDIMETAFILE:
467             if( pImpl->pGrf )
468                 bFnd = SetGraphic( *pImpl->pGrf, rFlavor );
469             break;
470         }
471 
472     return bFnd;
473 }
474 
475 // -----------------------------------------------------------------------------
476 
477 void TransferDataContainer::ClearData()
478 {
479     delete pImpl;
480     pImpl = new TransferDataContainer_Impl;
481     ClearFormats();
482 }
483 
484 // -----------------------------------------------------------------------------
485 
486 void TransferDataContainer::CopyINetBookmark( const INetBookmark& rBkmk )
487 {
488     if( !pImpl->pBookmk )
489         pImpl->pBookmk = new INetBookmark( rBkmk );
490     else
491         *pImpl->pBookmk = rBkmk;
492 
493     AddFormat( SOT_FORMAT_STRING );
494     AddFormat( SOT_FORMATSTR_ID_SOLK );
495     AddFormat( SOT_FORMATSTR_ID_NETSCAPE_BOOKMARK );
496     AddFormat( SOT_FORMATSTR_ID_FILECONTENT );
497     AddFormat( SOT_FORMATSTR_ID_FILEGRPDESCRIPTOR );
498     AddFormat( SOT_FORMATSTR_ID_UNIFORMRESOURCELOCATOR );
499 }
500 
501 // -----------------------------------------------------------------------------
502 
503 void TransferDataContainer::CopyAnyData( sal_uLong nFormatId,
504                                         const sal_Char* pData, sal_uLong nLen )
505 {
506     if( nLen )
507     {
508         TDataCntnrEntry_Impl aEntry;
509         aEntry.nId = nFormatId;
510 
511         Sequence< sal_Int8 > aSeq( nLen  );
512         memcpy( aSeq.getArray(), pData, nLen );
513         aEntry.aAny <<= aSeq;
514         pImpl->aFmtList.push_back( aEntry );
515         AddFormat( nFormatId );
516     }
517 }
518 
519 // -----------------------------------------------------------------------------
520 
521 void TransferDataContainer::CopyByteString( sal_uLong nFormatId,
522                                             const ByteString& rStr )
523 {
524     CopyAnyData( nFormatId, rStr.GetBuffer(), rStr.Len() );
525 }
526 
527 // -----------------------------------------------------------------------------
528 
529 void TransferDataContainer::CopyINetImage( const INetImage& rINtImg )
530 {
531     SvMemoryStream aMemStm( 1024, 1024 );
532     aMemStm.SetVersion( SOFFICE_FILEFORMAT_50 );
533     rINtImg.Write( aMemStm, SOT_FORMATSTR_ID_INET_IMAGE );
534     CopyAnyData( SOT_FORMATSTR_ID_INET_IMAGE, (sal_Char*)aMemStm.GetData(),
535                     aMemStm.Seek( STREAM_SEEK_TO_END ) );
536 }
537 
538 // -----------------------------------------------------------------------------
539 
540 void TransferDataContainer::CopyImageMap( const ImageMap& rImgMap )
541 {
542     SvMemoryStream aMemStm( 8192, 8192 );
543     aMemStm.SetVersion( SOFFICE_FILEFORMAT_50 );
544     rImgMap.Write( aMemStm, String() );
545     CopyAnyData( SOT_FORMATSTR_ID_SVIM, (sal_Char*)aMemStm.GetData(),
546                     aMemStm.Seek( STREAM_SEEK_TO_END ) );
547 }
548 
549 // -----------------------------------------------------------------------------
550 
551 void TransferDataContainer::CopyGraphic( const Graphic& rGrf )
552 {
553     GraphicType nType = rGrf.GetType();
554     if( GRAPHIC_NONE != nType )
555     {
556         if( !pImpl->pGrf )
557             pImpl->pGrf = new Graphic( rGrf );
558         else
559             *pImpl->pGrf = rGrf;
560 
561         AddFormat( SOT_FORMATSTR_ID_SVXB );
562         if( GRAPHIC_BITMAP == nType )
563             AddFormat( SOT_FORMAT_BITMAP );
564         else if( GRAPHIC_GDIMETAFILE == nType )
565             AddFormat( SOT_FORMAT_GDIMETAFILE );
566     }
567 }
568 
569 // -----------------------------------------------------------------------------
570 
571 void TransferDataContainer::CopyString( sal_uInt16 nFmt, const String& rStr )
572 {
573     if( rStr.Len() )
574     {
575         TDataCntnrEntry_Impl aEntry;
576         aEntry.nId = nFmt;
577         rtl::OUString aStr( rStr );
578         aEntry.aAny <<= aStr;
579         pImpl->aFmtList.push_back( aEntry );
580         AddFormat( aEntry.nId );
581     }
582 }
583 
584 // -----------------------------------------------------------------------------
585 
586 void TransferDataContainer::CopyString( const String& rStr )
587 {
588     CopyString( SOT_FORMAT_STRING, rStr );
589 }
590 
591 // -----------------------------------------------------------------------------
592 
593 void TransferDataContainer::CopyAny( sal_uInt16 nFmt,
594                                     const ::com::sun::star::uno::Any& rAny )
595 {
596     TDataCntnrEntry_Impl aEntry;
597     aEntry.nId = nFmt;
598     aEntry.aAny = rAny;
599     pImpl->aFmtList.push_back( aEntry );
600     AddFormat( aEntry.nId );
601 }
602 
603 // -----------------------------------------------------------------------------
604 
605 sal_Bool TransferDataContainer::HasAnyData() const
606 {
607     return pImpl->aFmtList.begin() != pImpl->aFmtList.end() ||
608             0 != pImpl->pBookmk;
609 }
610 
611 // -----------------------------------------------------------------------------
612 
613 void TransferDataContainer::StartDrag(
614         Window* pWindow, sal_Int8 nDragSourceActions,
615         const Link& rLnk, sal_Int32 nDragPointer, sal_Int32 nDragImage )
616 {
617     pImpl->aFinshedLnk = rLnk;
618     TransferableHelper::StartDrag( pWindow, nDragSourceActions,
619                                     nDragPointer, nDragImage );
620 }
621 
622 // -----------------------------------------------------------------------------
623 
624 void TransferDataContainer::DragFinished( sal_Int8 nDropAction )
625 {
626     if( pImpl->aFinshedLnk.IsSet() )
627         pImpl->aFinshedLnk.Call( &nDropAction );
628 }
629