1*9f62ea84SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3*9f62ea84SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4*9f62ea84SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5*9f62ea84SAndrew Rist  * distributed with this work for additional information
6*9f62ea84SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7*9f62ea84SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8*9f62ea84SAndrew Rist  * "License"); you may not use this file except in compliance
9*9f62ea84SAndrew Rist  * with the License.  You may obtain a copy of the License at
10*9f62ea84SAndrew Rist  *
11*9f62ea84SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12*9f62ea84SAndrew Rist  *
13*9f62ea84SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14*9f62ea84SAndrew Rist  * software distributed under the License is distributed on an
15*9f62ea84SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*9f62ea84SAndrew Rist  * KIND, either express or implied.  See the License for the
17*9f62ea84SAndrew Rist  * specific language governing permissions and limitations
18*9f62ea84SAndrew Rist  * under the License.
19*9f62ea84SAndrew Rist  *
20*9f62ea84SAndrew Rist  *************************************************************/
21*9f62ea84SAndrew Rist 
22*9f62ea84SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_vcl.hxx"
26cdf0e10cSrcweir #include "aqua_clipboard.hxx"
27cdf0e10cSrcweir 
28cdf0e10cSrcweir #include "DataFlavorMapping.hxx"
29cdf0e10cSrcweir #include "OSXTransferable.hxx"
30cdf0e10cSrcweir 
31cdf0e10cSrcweir #include "vcl/unohelp.hxx"
32cdf0e10cSrcweir 
33cdf0e10cSrcweir #include "comphelper/makesequence.hxx"
34cdf0e10cSrcweir 
35cdf0e10cSrcweir #include <boost/assert.hpp>
36cdf0e10cSrcweir 
37cdf0e10cSrcweir using namespace com::sun::star::datatransfer;
38cdf0e10cSrcweir using namespace com::sun::star::datatransfer::clipboard;
39cdf0e10cSrcweir using namespace com::sun::star::lang;
40cdf0e10cSrcweir using namespace com::sun::star::uno;
41cdf0e10cSrcweir using namespace cppu;
42cdf0e10cSrcweir using namespace osl;
43cdf0e10cSrcweir using namespace rtl;
44cdf0e10cSrcweir using namespace std;
45cdf0e10cSrcweir using namespace comphelper;
46cdf0e10cSrcweir 
47cdf0e10cSrcweir 
48cdf0e10cSrcweir @implementation EventListener;
49cdf0e10cSrcweir 
50cdf0e10cSrcweir -(EventListener*)initWithAquaClipboard: (AquaClipboard*) pcb
51cdf0e10cSrcweir {
52cdf0e10cSrcweir     self = [super init];
53cdf0e10cSrcweir 
54cdf0e10cSrcweir     if (self)
55cdf0e10cSrcweir         pAquaClipboard = pcb;
56cdf0e10cSrcweir 
57cdf0e10cSrcweir     return self;
58cdf0e10cSrcweir }
59cdf0e10cSrcweir 
60cdf0e10cSrcweir -(void)pasteboard:(NSPasteboard*)sender provideDataForType:(NSString*)type
61cdf0e10cSrcweir {
62cdf0e10cSrcweir     if( pAquaClipboard )
63cdf0e10cSrcweir         pAquaClipboard->provideDataForType(sender, type);
64cdf0e10cSrcweir }
65cdf0e10cSrcweir 
66cdf0e10cSrcweir -(void)applicationDidBecomeActive:(NSNotification*)aNotification
67cdf0e10cSrcweir {
68cdf0e10cSrcweir     if( pAquaClipboard )
69cdf0e10cSrcweir         pAquaClipboard->applicationDidBecomeActive(aNotification);
70cdf0e10cSrcweir }
71cdf0e10cSrcweir 
72cdf0e10cSrcweir -(void)disposing
73cdf0e10cSrcweir {
74cdf0e10cSrcweir     pAquaClipboard = NULL;
75cdf0e10cSrcweir }
76cdf0e10cSrcweir 
77cdf0e10cSrcweir @end
78cdf0e10cSrcweir 
79cdf0e10cSrcweir 
80cdf0e10cSrcweir OUString clipboard_getImplementationName()
81cdf0e10cSrcweir {
82cdf0e10cSrcweir   return OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.datatransfer.clipboard.AquaClipboard"));
83cdf0e10cSrcweir }
84cdf0e10cSrcweir 
85cdf0e10cSrcweir Sequence<OUString> clipboard_getSupportedServiceNames()
86cdf0e10cSrcweir {
87cdf0e10cSrcweir   return makeSequence(OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.datatransfer.clipboard.SystemClipboard")));
88cdf0e10cSrcweir }
89cdf0e10cSrcweir 
90cdf0e10cSrcweir 
91cdf0e10cSrcweir AquaClipboard::AquaClipboard(NSPasteboard* pasteboard, bool bUseSystemPasteboard) :
92cdf0e10cSrcweir   WeakComponentImplHelper4<XClipboardEx, XClipboardNotifier, XFlushableClipboard, XServiceInfo>(m_aMutex),
93cdf0e10cSrcweir   mIsSystemPasteboard(bUseSystemPasteboard)
94cdf0e10cSrcweir {
95cdf0e10cSrcweir     Reference<XMultiServiceFactory> mrServiceMgr = vcl::unohelper::GetMultiServiceFactory();
96cdf0e10cSrcweir 
97cdf0e10cSrcweir     mrXMimeCntFactory = Reference<XMimeContentTypeFactory>(mrServiceMgr->createInstance(
98cdf0e10cSrcweir 	 OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.datatransfer.MimeContentTypeFactory"))), UNO_QUERY);
99cdf0e10cSrcweir 
100cdf0e10cSrcweir   if (!mrXMimeCntFactory.is())
101cdf0e10cSrcweir 	{
102cdf0e10cSrcweir 	  throw RuntimeException(OUString(
103cdf0e10cSrcweir 			RTL_CONSTASCII_USTRINGPARAM("AquaClipboard: Cannot create com.sun.star.datatransfer.MimeContentTypeFactory")),
104cdf0e10cSrcweir 			static_cast<XClipboardEx*>(this));
105cdf0e10cSrcweir 	}
106cdf0e10cSrcweir 
107cdf0e10cSrcweir   mpDataFlavorMapper = DataFlavorMapperPtr_t(new DataFlavorMapper());
108cdf0e10cSrcweir 
109cdf0e10cSrcweir   if (pasteboard != NULL)
110cdf0e10cSrcweir 	{
111cdf0e10cSrcweir 	  mPasteboard = pasteboard;
112cdf0e10cSrcweir 	  mIsSystemPasteboard = false;
113cdf0e10cSrcweir 	}
114cdf0e10cSrcweir   else
115cdf0e10cSrcweir 	{
116cdf0e10cSrcweir 	  mPasteboard = bUseSystemPasteboard ? [NSPasteboard generalPasteboard] :
117cdf0e10cSrcweir 		[NSPasteboard pasteboardWithName: NSDragPboard];
118cdf0e10cSrcweir 
119cdf0e10cSrcweir 	  if (mPasteboard == nil)
120cdf0e10cSrcweir 		{
121cdf0e10cSrcweir 		  throw RuntimeException(OUString(
122cdf0e10cSrcweir 				RTL_CONSTASCII_USTRINGPARAM("AquaClipboard: Cannot create Cocoa pasteboard")),
123cdf0e10cSrcweir 				static_cast<XClipboardEx*>(this));
124cdf0e10cSrcweir 		}
125cdf0e10cSrcweir 	}
126cdf0e10cSrcweir 
127cdf0e10cSrcweir   [mPasteboard retain];
128cdf0e10cSrcweir 
129cdf0e10cSrcweir   mEventListener = [[EventListener alloc] initWithAquaClipboard: this];
130cdf0e10cSrcweir 
131cdf0e10cSrcweir   if (mEventListener == nil)
132cdf0e10cSrcweir 	{
133cdf0e10cSrcweir 	  [mPasteboard release];
134cdf0e10cSrcweir 
135cdf0e10cSrcweir 	  throw RuntimeException(
136cdf0e10cSrcweir 			OUString(RTL_CONSTASCII_USTRINGPARAM("AquaClipboard: Cannot create pasteboard change listener")),
137cdf0e10cSrcweir 			static_cast<XClipboardEx*>(this));
138cdf0e10cSrcweir 	}
139cdf0e10cSrcweir 
140cdf0e10cSrcweir   if (mIsSystemPasteboard)
141cdf0e10cSrcweir 	{
142cdf0e10cSrcweir 	  NSNotificationCenter* notificationCenter = [NSNotificationCenter defaultCenter];
143cdf0e10cSrcweir 
144cdf0e10cSrcweir 	  [notificationCenter addObserver: mEventListener
145cdf0e10cSrcweir 	   selector: @selector(applicationDidBecomeActive:)
146cdf0e10cSrcweir 	   name: @"NSApplicationDidBecomeActiveNotification"
147cdf0e10cSrcweir 	   object: [NSApplication sharedApplication]];
148cdf0e10cSrcweir 	}
149cdf0e10cSrcweir 
150cdf0e10cSrcweir   mPasteboardChangeCount = [mPasteboard changeCount];
151cdf0e10cSrcweir }
152cdf0e10cSrcweir 
153cdf0e10cSrcweir 
154cdf0e10cSrcweir AquaClipboard::~AquaClipboard()
155cdf0e10cSrcweir {
156cdf0e10cSrcweir   if (mIsSystemPasteboard)
157cdf0e10cSrcweir 	{
158cdf0e10cSrcweir 	  [[NSNotificationCenter defaultCenter] removeObserver: mEventListener];
159cdf0e10cSrcweir 	}
160cdf0e10cSrcweir 
161cdf0e10cSrcweir   [mEventListener disposing];
162cdf0e10cSrcweir   [mEventListener release];
163cdf0e10cSrcweir   [mPasteboard release];
164cdf0e10cSrcweir }
165cdf0e10cSrcweir 
166cdf0e10cSrcweir 
167cdf0e10cSrcweir Reference<XTransferable> SAL_CALL AquaClipboard::getContents() throw(RuntimeException)
168cdf0e10cSrcweir {
169cdf0e10cSrcweir   MutexGuard aGuard(m_aMutex);
170cdf0e10cSrcweir 
171cdf0e10cSrcweir   // Shortcut: If we are clipboard owner already we don't need
172cdf0e10cSrcweir   // to drag the data through the system clipboard
173cdf0e10cSrcweir   if (mXClipboardContent.is())
174cdf0e10cSrcweir 	{
175cdf0e10cSrcweir 	  return mXClipboardContent;
176cdf0e10cSrcweir 	}
177cdf0e10cSrcweir 
178cdf0e10cSrcweir   return Reference<XTransferable>(new OSXTransferable(mrXMimeCntFactory,
179cdf0e10cSrcweir 													  mpDataFlavorMapper,
180cdf0e10cSrcweir 													  mPasteboard));
181cdf0e10cSrcweir }
182cdf0e10cSrcweir 
183cdf0e10cSrcweir 
184cdf0e10cSrcweir void SAL_CALL AquaClipboard::setContents(const Reference<XTransferable>& xTransferable,
185cdf0e10cSrcweir     const Reference<XClipboardOwner>& xClipboardOwner)
186cdf0e10cSrcweir         throw( RuntimeException )
187cdf0e10cSrcweir {
188cdf0e10cSrcweir     NSArray* types = xTransferable.is() ?
189cdf0e10cSrcweir         mpDataFlavorMapper->flavorSequenceToTypesArray(xTransferable->getTransferDataFlavors()) :
190cdf0e10cSrcweir         [NSArray array];
191cdf0e10cSrcweir 
192cdf0e10cSrcweir     ClearableMutexGuard aGuard(m_aMutex);
193cdf0e10cSrcweir 
194cdf0e10cSrcweir     Reference<XClipboardOwner> oldOwner(mXClipboardOwner);
195cdf0e10cSrcweir     mXClipboardOwner = xClipboardOwner;
196cdf0e10cSrcweir 
197cdf0e10cSrcweir     Reference<XTransferable> oldContent(mXClipboardContent);
198cdf0e10cSrcweir     mXClipboardContent = xTransferable;
199cdf0e10cSrcweir 
200cdf0e10cSrcweir     mPasteboardChangeCount = [mPasteboard declareTypes: types owner: mEventListener];
201cdf0e10cSrcweir 
202cdf0e10cSrcweir     aGuard.clear();
203cdf0e10cSrcweir 
204cdf0e10cSrcweir     // if we are already the owner of the clipboard
205cdf0e10cSrcweir     // then fire lost ownership event
206cdf0e10cSrcweir     if (oldOwner.is())
207cdf0e10cSrcweir 	{
208cdf0e10cSrcweir         fireLostClipboardOwnershipEvent(oldOwner, oldContent);
209cdf0e10cSrcweir 	}
210cdf0e10cSrcweir 
211cdf0e10cSrcweir     fireClipboardChangedEvent();
212cdf0e10cSrcweir }
213cdf0e10cSrcweir 
214cdf0e10cSrcweir 
215cdf0e10cSrcweir OUString SAL_CALL AquaClipboard::getName() throw( RuntimeException )
216cdf0e10cSrcweir {
217cdf0e10cSrcweir   return OUString();
218cdf0e10cSrcweir }
219cdf0e10cSrcweir 
220cdf0e10cSrcweir 
221cdf0e10cSrcweir sal_Int8 SAL_CALL AquaClipboard::getRenderingCapabilities() throw( RuntimeException )
222cdf0e10cSrcweir {
223cdf0e10cSrcweir 	return 0;
224cdf0e10cSrcweir }
225cdf0e10cSrcweir 
226cdf0e10cSrcweir 
227cdf0e10cSrcweir void SAL_CALL AquaClipboard::addClipboardListener(const Reference< XClipboardListener >& listener)
228cdf0e10cSrcweir   throw( RuntimeException )
229cdf0e10cSrcweir {
230cdf0e10cSrcweir   MutexGuard aGuard(m_aMutex);
231cdf0e10cSrcweir 
232cdf0e10cSrcweir   if (!listener.is())
233cdf0e10cSrcweir  	throw IllegalArgumentException(OUString(RTL_CONSTASCII_USTRINGPARAM("empty reference")),
234cdf0e10cSrcweir 								   static_cast<XClipboardEx*>(this), 1);
235cdf0e10cSrcweir 
236cdf0e10cSrcweir   mClipboardListeners.push_back(listener);
237cdf0e10cSrcweir }
238cdf0e10cSrcweir 
239cdf0e10cSrcweir 
240cdf0e10cSrcweir void SAL_CALL AquaClipboard::removeClipboardListener(const Reference< XClipboardListener >& listener)
241cdf0e10cSrcweir   throw( RuntimeException )
242cdf0e10cSrcweir {
243cdf0e10cSrcweir   MutexGuard aGuard(m_aMutex);
244cdf0e10cSrcweir 
245cdf0e10cSrcweir   if (!listener.is())
246cdf0e10cSrcweir  	throw IllegalArgumentException(OUString(RTL_CONSTASCII_USTRINGPARAM("empty reference")),
247cdf0e10cSrcweir 								   static_cast<XClipboardEx*>(this), 1);
248cdf0e10cSrcweir 
249cdf0e10cSrcweir   mClipboardListeners.remove(listener);
250cdf0e10cSrcweir }
251cdf0e10cSrcweir 
252cdf0e10cSrcweir 
253cdf0e10cSrcweir void AquaClipboard::applicationDidBecomeActive(NSNotification*)
254cdf0e10cSrcweir {
255cdf0e10cSrcweir   ClearableMutexGuard aGuard(m_aMutex);
256cdf0e10cSrcweir 
257cdf0e10cSrcweir   int currentPboardChgCount = [mPasteboard changeCount];
258cdf0e10cSrcweir 
259cdf0e10cSrcweir   if (currentPboardChgCount != mPasteboardChangeCount)
260cdf0e10cSrcweir 	{
261cdf0e10cSrcweir 	  mPasteboardChangeCount = currentPboardChgCount;
262cdf0e10cSrcweir 
263cdf0e10cSrcweir 	  // Clear clipboard content and owner and send lostOwnership
264cdf0e10cSrcweir 	  // notification to the old clipboard owner as well as
265cdf0e10cSrcweir 	  // ClipboardChanged notification to any clipboard listener
266cdf0e10cSrcweir 	  Reference<XClipboardOwner> oldOwner(mXClipboardOwner);
267cdf0e10cSrcweir 	  mXClipboardOwner = Reference<XClipboardOwner>();
268cdf0e10cSrcweir 
269cdf0e10cSrcweir 	  Reference<XTransferable> oldContent(mXClipboardContent);
270cdf0e10cSrcweir 	  mXClipboardContent = Reference<XTransferable>();
271cdf0e10cSrcweir 
272cdf0e10cSrcweir       aGuard.clear();
273cdf0e10cSrcweir 
274cdf0e10cSrcweir 	  if (oldOwner.is())
275cdf0e10cSrcweir 		{
276cdf0e10cSrcweir 		  fireLostClipboardOwnershipEvent(oldOwner, oldContent);
277cdf0e10cSrcweir 		}
278cdf0e10cSrcweir 
279cdf0e10cSrcweir 	  fireClipboardChangedEvent();
280cdf0e10cSrcweir 	}
281cdf0e10cSrcweir }
282cdf0e10cSrcweir 
283cdf0e10cSrcweir 
284cdf0e10cSrcweir void AquaClipboard::fireClipboardChangedEvent()
285cdf0e10cSrcweir {
286cdf0e10cSrcweir     ClearableMutexGuard aGuard(m_aMutex);
287cdf0e10cSrcweir 
288cdf0e10cSrcweir     list<Reference< XClipboardListener > > listeners(mClipboardListeners);
289cdf0e10cSrcweir 	ClipboardEvent aEvent;
290cdf0e10cSrcweir 
291cdf0e10cSrcweir 	if (listeners.begin() != listeners.end())
292cdf0e10cSrcweir 	  {
293cdf0e10cSrcweir 		aEvent = ClipboardEvent(static_cast<OWeakObject*>(this), getContents());
294cdf0e10cSrcweir 	  }
295cdf0e10cSrcweir 
296cdf0e10cSrcweir     aGuard.clear();
297cdf0e10cSrcweir 
298cdf0e10cSrcweir     while (listeners.begin() != listeners.end())
299cdf0e10cSrcweir     {
300cdf0e10cSrcweir         if (listeners.front().is())
301cdf0e10cSrcweir 		  {
302cdf0e10cSrcweir 			try { listeners.front()->changedContents(aEvent); }
303cdf0e10cSrcweir 			catch (RuntimeException&) { }
304cdf0e10cSrcweir 		  }
305cdf0e10cSrcweir         listeners.pop_front();
306cdf0e10cSrcweir     }
307cdf0e10cSrcweir }
308cdf0e10cSrcweir 
309cdf0e10cSrcweir 
310cdf0e10cSrcweir void AquaClipboard::fireLostClipboardOwnershipEvent(Reference<XClipboardOwner> oldOwner, Reference<XTransferable> oldContent)
311cdf0e10cSrcweir {
312cdf0e10cSrcweir   BOOST_ASSERT(oldOwner.is());
313cdf0e10cSrcweir 
314cdf0e10cSrcweir   try { oldOwner->lostOwnership(static_cast<XClipboardEx*>(this), oldContent); }
315cdf0e10cSrcweir   catch(RuntimeException&) { }
316cdf0e10cSrcweir }
317cdf0e10cSrcweir 
318cdf0e10cSrcweir 
319cdf0e10cSrcweir void AquaClipboard::provideDataForType(NSPasteboard* sender, NSString* type)
320cdf0e10cSrcweir {
321cdf0e10cSrcweir     if( mXClipboardContent.is() )
322cdf0e10cSrcweir     {
323cdf0e10cSrcweir         DataProviderPtr_t dp = mpDataFlavorMapper->getDataProvider(type, mXClipboardContent);
324cdf0e10cSrcweir         NSData* pBoardData = NULL;
325cdf0e10cSrcweir 
326cdf0e10cSrcweir         if (dp.get() != NULL)
327cdf0e10cSrcweir         {
328cdf0e10cSrcweir             pBoardData = (NSData*)dp->getSystemData();
329cdf0e10cSrcweir             [sender setData: pBoardData forType: type];
330cdf0e10cSrcweir         }
331cdf0e10cSrcweir     }
332cdf0e10cSrcweir }
333cdf0e10cSrcweir 
334cdf0e10cSrcweir 
335cdf0e10cSrcweir //------------------------------------------------
336cdf0e10cSrcweir // XFlushableClipboard
337cdf0e10cSrcweir //------------------------------------------------
338cdf0e10cSrcweir 
339cdf0e10cSrcweir void SAL_CALL AquaClipboard::flushClipboard()
340cdf0e10cSrcweir   throw(RuntimeException)
341cdf0e10cSrcweir {
342cdf0e10cSrcweir     if (mXClipboardContent.is())
343cdf0e10cSrcweir 	{
344cdf0e10cSrcweir 	  	Sequence<DataFlavor> flavorList = mXClipboardContent->getTransferDataFlavors();
345cdf0e10cSrcweir 		sal_uInt32 nFlavors = flavorList.getLength();
346cdf0e10cSrcweir 
347cdf0e10cSrcweir 		for (sal_uInt32 i = 0; i < nFlavors; i++)
348cdf0e10cSrcweir 		{
349cdf0e10cSrcweir 			NSString* sysType = mpDataFlavorMapper->openOfficeToSystemFlavor(flavorList[i]);
350cdf0e10cSrcweir 
351cdf0e10cSrcweir 			if (sysType != NULL)
352cdf0e10cSrcweir 			{
353cdf0e10cSrcweir 				provideDataForType(mPasteboard, sysType);
354cdf0e10cSrcweir 			}
355cdf0e10cSrcweir 		}
356cdf0e10cSrcweir 		mXClipboardContent.clear();
357cdf0e10cSrcweir 	}
358cdf0e10cSrcweir }
359cdf0e10cSrcweir 
360cdf0e10cSrcweir 
361cdf0e10cSrcweir NSPasteboard* AquaClipboard::getPasteboard() const
362cdf0e10cSrcweir {
363cdf0e10cSrcweir   return mPasteboard;
364cdf0e10cSrcweir }
365cdf0e10cSrcweir 
366cdf0e10cSrcweir 
367cdf0e10cSrcweir //-------------------------------------------------
368cdf0e10cSrcweir // XServiceInfo
369cdf0e10cSrcweir //-------------------------------------------------
370cdf0e10cSrcweir 
371cdf0e10cSrcweir OUString SAL_CALL AquaClipboard::getImplementationName() throw( RuntimeException )
372cdf0e10cSrcweir {
373cdf0e10cSrcweir   return clipboard_getImplementationName();
374cdf0e10cSrcweir }
375cdf0e10cSrcweir 
376cdf0e10cSrcweir 
377cdf0e10cSrcweir sal_Bool SAL_CALL AquaClipboard::supportsService( const OUString& /*ServiceName*/ ) throw( RuntimeException )
378cdf0e10cSrcweir {
379cdf0e10cSrcweir 	return sal_False;
380cdf0e10cSrcweir }
381cdf0e10cSrcweir 
382cdf0e10cSrcweir 
383cdf0e10cSrcweir Sequence< OUString > SAL_CALL AquaClipboard::getSupportedServiceNames() throw( RuntimeException )
384cdf0e10cSrcweir {
385cdf0e10cSrcweir   return clipboard_getSupportedServiceNames();
386cdf0e10cSrcweir }
387cdf0e10cSrcweir 
388