1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #include "vcl/unohelp.hxx"
25 #include <DataFlavorMapping.hxx>
26 #include "HtmlFmtFlt.hxx"
27 #include "PictToBmpFlt.hxx"
28 #include "com/sun/star/datatransfer/UnsupportedFlavorException.hpp"
29 #include "com/sun/star/datatransfer/XMimeContentType.hpp"
30 #include "com/sun/star/lang/XMultiServiceFactory.hpp"
31 #include "com/sun/star/uno/Sequence.hxx"
32 
33 #include <rtl/ustring.hxx>
34 #include <rtl/memory.h>
35 #include <osl/endian.h>
36 
37 #include <vector>
38 #include <stdio.h>
39 
40 #include <premac.h>
41 #include <Cocoa/Cocoa.h>
42 #include <postmac.h>
43 
44 using namespace ::com::sun::star::datatransfer;
45 using namespace rtl;
46 using namespace ::com::sun::star::uno;
47 using namespace com::sun::star::lang;
48 using namespace cppu;
49 using namespace std;
50 
51 namespace // private
52 {
53   const Type CPPUTYPE_SEQINT8  = getCppuType((Sequence<sal_Int8>*)0);
54   const Type CPPUTYPE_OUSTRING = getCppuType( (OUString*)0 );
55 
56   /* Determine whether or not a DataFlavor is valid.
57    */
58   bool isValidFlavor(const DataFlavor& aFlavor)
59   {
60 	size_t len = aFlavor.MimeType.getLength();
61 	Type dtype = aFlavor.DataType;
62 	return ((len > 0) && ((dtype == CPPUTYPE_SEQINT8) || (dtype == CPPUTYPE_OUSTRING)));
63   }
64 
65   typedef vector<sal_Unicode> UnicodeBuffer;
66 
67   OUString NSStringToOUString(NSString* cfString)
68   {
69 	BOOST_ASSERT(cfString && "Invalid parameter");
70 
71 	const char* utf8Str = [cfString UTF8String];
72 	unsigned int len = rtl_str_getLength(utf8Str);
73 
74 	return OUString(utf8Str, len, RTL_TEXTENCODING_UTF8);
75   }
76 
77   NSString* OUStringToNSString(const OUString& ustring)
78   {
79 	OString utf8Str = OUStringToOString(ustring, RTL_TEXTENCODING_UTF8);
80 	return [NSString stringWithCString: utf8Str.getStr() encoding: NSUTF8StringEncoding];
81   }
82 
83 
84   const NSString* PBTYPE_UT16 = @"CorePasteboardFlavorType 0x75743136";
85   const NSString* PBTYPE_PICT = @"CorePasteboardFlavorType 0x50494354";
86   const NSString* PBTYPE_HTML = @"CorePasteboardFlavorType 0x48544D4C";
87   const NSString* PBTYPE_SODX = @"application/x-openoffice-objectdescriptor-xml;windows_formatname=\"Star Object Descriptor (XML)\"";
88   const NSString* PBTYPE_SESX = @"application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\"";
89   const NSString* PBTYPE_SLSDX = @"application/x-openoffice-linksrcdescriptor-xml;windows_formatname=\"Star Link Source Descriptor (XML)\"";
90   const NSString* PBTYPE_ESX = @"application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\"";
91   const NSString* PBTYPE_LSX = @"application/x-openoffice-link-source-xml;windows_formatname=\"Star Link Source (XML)\"";
92   const NSString* PBTYPE_EOX = @"application/x-openoffice-embedded-obj-xml;windows_formatname=\"Star Embedded Object (XML)\"";
93   const NSString* PBTYPE_SVXB = @"application/x-openoffice-svbx;windows_formatname=\"SVXB (StarView Bitmap/Animation)\"";
94   const NSString* PBTYPE_GDIMF = @"application/x-openoffice-gdimetafile;windows_formatname=\"GDIMetaFile\"";
95   const NSString* PBTYPE_WMF = @"application/x-openoffice-wmf;windows_formatname=\"Image WMF\"";
96   const NSString* PBTYPE_EMF = @"application/x-openoffice-emf;windows_formatname=\"Image EMF\"";
97 
98   const NSString* PBTYPE_DUMMY_INTERNAL = @"application/x-openoffice-internal";
99 
100   const char* FLAVOR_SODX = "application/x-openoffice-objectdescriptor-xml;windows_formatname=\"Star Object Descriptor (XML)\"";
101   const char* FLAVOR_SESX = "application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\"";
102   const char* FLAVOR_SLSDX = "application/x-openoffice-linksrcdescriptor-xml;windows_formatname=\"Star Link Source Descriptor (XML)\"";
103   const char* FLAVOR_ESX = "application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\"";
104   const char* FLAVOR_LSX = "application/x-openoffice-link-source-xml;windows_formatname=\"Star Link Source (XML)\"";
105   const char* FLAVOR_EOX = "application/x-openoffice-embedded-obj-xml;windows_formatname=\"Star Embedded Object (XML)\"";
106   const char* FLAVOR_SVXB = "application/x-openoffice-svbx;windows_formatname=\"SVXB (StarView Bitmap/Animation)\"";
107   const char* FLAVOR_GDIMF = "application/x-openoffice-gdimetafile;windows_formatname=\"GDIMetaFile\"";
108   const char* FLAVOR_WMF = "application/x-openoffice-wmf;windows_formatname=\"Image WMF\"";
109   const char* FLAVOR_EMF = "application/x-openoffice-emf;windows_formatname=\"Image EMF\"";
110 
111   const char* FLAVOR_DUMMY_INTERNAL = "application/x-openoffice-internal";
112 
113 
114   struct FlavorMap
115   {
116 	NSString* SystemFlavor;
117 	const char* OOoFlavor;
118 	const char* HumanPresentableName;
119 	Type DataType;
120   };
121 
122   /* At the moment it appears as if only MS Office pastes "public.html" to the clipboard.
123    */
124   FlavorMap flavorMap[] =
125 	{
126 	  { NSStringPboardType, "text/plain;charset=utf-16", "Unicode Text (UTF-16)", CPPUTYPE_OUSTRING },
127 	  { NSRTFPboardType, "text/richtext", "Rich Text Format", CPPUTYPE_SEQINT8 },
128 	  { NSTIFFPboardType, "image/bmp", "Windows Bitmap", CPPUTYPE_SEQINT8 },
129 	  { NSPICTPboardType, "image/bmp", "Windows Bitmap", CPPUTYPE_SEQINT8 },
130 	  { NSHTMLPboardType, "text/html", "Plain Html", CPPUTYPE_SEQINT8 },
131 	  { NSFilenamesPboardType, "application/x-openoffice-filelist;windows_formatname=\"FileList\"", "FileList", CPPUTYPE_SEQINT8 },
132 	  { PBTYPE_SESX, FLAVOR_SESX, "Star Embed Source (XML)", CPPUTYPE_SEQINT8 },
133 	  { PBTYPE_SLSDX, FLAVOR_SLSDX, "Star Link Source Descriptor (XML)", CPPUTYPE_SEQINT8 },
134 	  { PBTYPE_ESX, FLAVOR_ESX, "Star Embed Source (XML)", CPPUTYPE_SEQINT8 },
135 	  { PBTYPE_LSX, FLAVOR_LSX, "Star Link Source (XML)", CPPUTYPE_SEQINT8 },
136 	  { PBTYPE_EOX, FLAVOR_EOX, "Star Embedded Object (XML)", CPPUTYPE_SEQINT8 },
137 	  { PBTYPE_SVXB, FLAVOR_SVXB, "SVXB (StarView Bitmap/Animation", CPPUTYPE_SEQINT8 },
138 	  { PBTYPE_GDIMF, FLAVOR_GDIMF, "GDIMetaFile", CPPUTYPE_SEQINT8 },
139 	  { PBTYPE_WMF, FLAVOR_WMF, "Windows MetaFile", CPPUTYPE_SEQINT8 },
140 	  { PBTYPE_EMF, FLAVOR_EMF, "Windows Enhanced MetaFile", CPPUTYPE_SEQINT8 },
141 	  { PBTYPE_SODX, FLAVOR_SODX, "Star Object Descriptor (XML)", CPPUTYPE_SEQINT8 },
142       { PBTYPE_DUMMY_INTERNAL, FLAVOR_DUMMY_INTERNAL, "internal data",CPPUTYPE_SEQINT8 }
143 	};
144 
145 
146   #define SIZE_FLAVOR_MAP (sizeof(flavorMap)/sizeof(FlavorMap))
147 
148 
149   inline bool isByteSequenceType(const Type& theType)
150   {
151 	return (theType == CPPUTYPE_SEQINT8);
152   }
153 
154   inline bool isOUStringType(const Type& theType)
155   {
156 	return (theType == CPPUTYPE_OUSTRING);
157   }
158 
159 } // namespace private
160 
161 
162 //###########################
163 
164 /* A base class for other data provider.
165  */
166 class DataProviderBaseImpl : public DataProvider
167 {
168 public:
169   DataProviderBaseImpl(const Any& data);
170   DataProviderBaseImpl(id data);
171   virtual ~DataProviderBaseImpl();
172 
173 protected:
174   Any mData;
175   //NSData* mSystemData;
176   id mSystemData;
177 };
178 
179 DataProviderBaseImpl::DataProviderBaseImpl(const Any& data) :
180   mData(data),
181   mSystemData(nil)
182 {
183 }
184 
185 DataProviderBaseImpl::DataProviderBaseImpl(id data) :
186   mSystemData(data)
187 {
188   [mSystemData retain];
189 }
190 
191 
192 DataProviderBaseImpl::~DataProviderBaseImpl()
193 {
194   if (mSystemData)
195 	{
196 	  [mSystemData release];
197 	}
198 }
199 
200 //#################################
201 
202 class UniDataProvider : public DataProviderBaseImpl
203 {
204 public:
205   UniDataProvider(const Any& data);
206 
207   UniDataProvider(NSData* data);
208 
209   virtual NSData* getSystemData();
210 
211   virtual Any getOOoData();
212 };
213 
214 UniDataProvider::UniDataProvider(const Any& data) :
215   DataProviderBaseImpl(data)
216 {
217 }
218 
219 UniDataProvider::UniDataProvider(NSData* data) :
220   DataProviderBaseImpl(data)
221 {
222 }
223 
224 NSData* UniDataProvider::getSystemData()
225 {
226   OUString ustr;
227   mData >>= ustr;
228 
229   OString strUtf8;
230   ustr.convertToString(&strUtf8, RTL_TEXTENCODING_UTF8, OUSTRING_TO_OSTRING_CVTFLAGS);
231 
232   return [NSData dataWithBytes: strUtf8.getStr() length: strUtf8.getLength()];
233 }
234 
235 Any UniDataProvider::getOOoData()
236 {
237   Any oOOData;
238 
239   if (mSystemData)
240 	{
241 	  oOOData = makeAny(OUString(reinterpret_cast<const sal_Char*>([mSystemData bytes]),
242 								 [mSystemData length],
243 								 RTL_TEXTENCODING_UTF8));
244 	}
245   else
246 	{
247 	  oOOData = mData;
248 	}
249 
250   return oOOData;
251 }
252 
253 //###########################
254 
255 class ByteSequenceDataProvider : public DataProviderBaseImpl
256 {
257 public:
258   ByteSequenceDataProvider(const Any& data);
259 
260   ByteSequenceDataProvider(NSData* data);
261 
262   virtual NSData* getSystemData();
263 
264   virtual Any getOOoData();
265 };
266 
267 ByteSequenceDataProvider::ByteSequenceDataProvider(const Any& data) :
268   DataProviderBaseImpl(data)
269 {
270 }
271 
272 ByteSequenceDataProvider::ByteSequenceDataProvider(NSData* data) :
273   DataProviderBaseImpl(data)
274 {
275 }
276 
277 
278 NSData* ByteSequenceDataProvider::getSystemData()
279 {
280    Sequence<sal_Int8> rawData;
281    mData >>= rawData;
282 
283    return [NSData dataWithBytes: rawData.getArray()	length: rawData.getLength()];
284 }
285 
286 Any ByteSequenceDataProvider::getOOoData()
287 {
288   Any oOOData;
289 
290   if (mSystemData)
291 	{
292 	  unsigned int flavorDataLength = [mSystemData length];
293 	  Sequence<sal_Int8> byteSequence;
294 	  byteSequence.realloc(flavorDataLength);
295 	  memcpy(byteSequence.getArray(), [mSystemData bytes], flavorDataLength);
296 	  oOOData = makeAny(byteSequence);
297 	}
298   else
299 	{
300 	  oOOData =  mData;
301 	}
302 
303   return oOOData;
304 }
305 
306 
307 //###########################
308 
309 class HTMLFormatDataProvider : public DataProviderBaseImpl
310 {
311 public:
312   HTMLFormatDataProvider(const Any& data);
313 
314   HTMLFormatDataProvider(NSData* data);
315 
316   virtual NSData* getSystemData();
317 
318   virtual Any getOOoData();
319 };
320 
321 HTMLFormatDataProvider::HTMLFormatDataProvider(const Any& data) :
322   DataProviderBaseImpl(data)
323 {
324 }
325 
326 HTMLFormatDataProvider::HTMLFormatDataProvider(NSData* data) :
327   DataProviderBaseImpl(data)
328 {
329 }
330 
331 NSData* HTMLFormatDataProvider::getSystemData()
332 {
333   Sequence<sal_Int8> textHtmlData;
334   mData >>= textHtmlData;
335 
336   Sequence<sal_Int8> htmlFormatData = TextHtmlToHTMLFormat(textHtmlData);
337 
338   return [NSData dataWithBytes: htmlFormatData.getArray() length: htmlFormatData.getLength()];
339 }
340 
341 Any HTMLFormatDataProvider::getOOoData()
342 {
343   Any oOOData;
344 
345   if (mSystemData)
346 	{
347 	  unsigned int flavorDataLength = [mSystemData length];
348 	  Sequence<sal_Int8> unkHtmlData;
349 
350 	  unkHtmlData.realloc(flavorDataLength);
351 	  memcpy(unkHtmlData.getArray(), [mSystemData bytes], flavorDataLength);
352 
353 	  Sequence<sal_Int8>* pPlainHtml = &unkHtmlData;
354 	  Sequence<sal_Int8> plainHtml;
355 
356 	  if (isHTMLFormat(unkHtmlData))
357 		{
358 		  plainHtml = HTMLFormatToTextHtml(unkHtmlData);
359 		  pPlainHtml = &plainHtml;
360 		}
361 
362 	  oOOData = makeAny(*pPlainHtml);
363 	}
364   else
365 	{
366 	  oOOData = mData;
367 	}
368 
369   return oOOData;
370 }
371 
372 //###########################
373 
374 class BMPDataProvider : public DataProviderBaseImpl
375 {
376     NSBitmapImageFileType meImageType;
377 public:
378   BMPDataProvider(const Any& data, NSBitmapImageFileType eImageType );
379 
380   BMPDataProvider(NSData* data, NSBitmapImageFileType eImageType);
381 
382   virtual NSData* getSystemData();
383 
384   virtual Any getOOoData();
385 };
386 
387 BMPDataProvider::BMPDataProvider(const Any& data, NSBitmapImageFileType eImageType) :
388   DataProviderBaseImpl(data),
389   meImageType( eImageType )
390 {
391 }
392 
393 BMPDataProvider::BMPDataProvider(NSData* data, NSBitmapImageFileType eImageType) :
394   DataProviderBaseImpl(data),
395   meImageType( eImageType )
396 {
397 }
398 
399 NSData* BMPDataProvider::getSystemData()
400 {
401   Sequence<sal_Int8> bmpData;
402   mData >>= bmpData;
403 
404   Sequence<sal_Int8> pictData;
405   NSData* sysData = NULL;
406 
407   if (BMPToImage(bmpData, pictData, meImageType))
408 	{
409 	  sysData = [NSData dataWithBytes: pictData.getArray() length: pictData.getLength()];
410 	}
411 
412   return sysData;
413 }
414 
415 /* At the moment the OOo 'PCT' filter is not good enough to be used
416    and there is no flavor defined for exchanging 'PCT' with OOo so
417    we will at the moment convert 'PCT' to a Windows BMP and provide
418    this to OOo
419 */
420 Any BMPDataProvider::getOOoData()
421 {
422   Any oOOData;
423 
424   if (mSystemData)
425 	{
426 	  unsigned int flavorDataLength = [mSystemData length];
427 	  Sequence<sal_Int8> pictData(flavorDataLength);
428 
429 	  memcpy(pictData.getArray(), [mSystemData bytes], flavorDataLength);
430 
431 	  Sequence<sal_Int8> bmpData;
432 
433 	  if (ImageToBMP(pictData, bmpData, meImageType))
434 		{
435 		  oOOData = makeAny(bmpData);
436 		}
437 	}
438   else
439 	{
440 	  oOOData = mData;
441 	}
442 
443   return oOOData;
444 }
445 
446 //######################
447 
448 class FileListDataProvider : public DataProviderBaseImpl
449 {
450 public:
451   FileListDataProvider(const Any& data);
452   FileListDataProvider(NSArray* data);
453 
454   virtual NSData* getSystemData();
455   virtual Any getOOoData();
456 };
457 
458 FileListDataProvider::FileListDataProvider(const Any& data) :
459   DataProviderBaseImpl(data)
460 {
461 }
462 
463 FileListDataProvider::FileListDataProvider(NSArray* data) :
464   DataProviderBaseImpl(data)
465 {
466 }
467 
468 NSData* FileListDataProvider::getSystemData()
469 {
470   return [NSData data];
471 }
472 
473 Any FileListDataProvider::getOOoData()
474 {
475   Any oOOData;
476 
477   if (mSystemData)
478 	{
479 	  size_t length = [mSystemData count];
480 	  size_t lenSeqRequired = 0;
481 
482 	  for (size_t i = 0; i < length; i++)
483 		{
484 		  NSString* fname = [mSystemData objectAtIndex: i];
485 		  lenSeqRequired += [fname maximumLengthOfBytesUsingEncoding: NSUnicodeStringEncoding] + sizeof(unichar);
486 		}
487 
488 	  Sequence<sal_Int8> oOOFileList(lenSeqRequired);
489 	  unichar* pBuffer = reinterpret_cast<unichar*>(oOOFileList.getArray());
490 	  rtl_zeroMemory(pBuffer, lenSeqRequired);
491 
492 	  for (size_t i = 0; i < length; i++)
493 		{
494 		  NSString* fname = [mSystemData objectAtIndex: i];
495 		  [fname getCharacters: pBuffer];
496 		  size_t l = [fname length];
497 		  pBuffer += l + 1;
498 		}
499 
500 	  oOOData = makeAny(oOOFileList);
501 	}
502   else
503 	{
504 	  oOOData = mData;
505 	}
506 
507   return oOOData;
508 }
509 
510 //###########################
511 
512 DataFlavorMapper::DataFlavorMapper()
513 {
514     Reference<XMultiServiceFactory> mrServiceManager = vcl::unohelper::GetMultiServiceFactory();
515     mrXMimeCntFactory = Reference<XMimeContentTypeFactory>(mrServiceManager->createInstance(
516 	   OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.datatransfer.MimeContentTypeFactory"))), UNO_QUERY);
517 
518   if (!mrXMimeCntFactory.is())
519 	throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM("AquaClipboard: Cannot create com.sun.star.datatransfer.MimeContentTypeFactory")), NULL);
520 }
521 
522 DataFlavorMapper::~DataFlavorMapper()
523 {
524     // release potential NSStrings
525     for( OfficeOnlyTypes::iterator it = maOfficeOnlyTypes.begin(); it != maOfficeOnlyTypes.end(); ++it )
526     {
527         [it->second release];
528         it->second = nil;
529     }
530 }
531 
532 DataFlavor DataFlavorMapper::systemToOpenOfficeFlavor(NSString* systemDataFlavor) const
533 {
534   DataFlavor oOOFlavor;
535 
536   for (size_t i = 0; i < SIZE_FLAVOR_MAP; i++)
537 	{
538 	  if ([systemDataFlavor caseInsensitiveCompare: flavorMap[i].SystemFlavor] == NSOrderedSame)
539 		{
540 		  oOOFlavor.MimeType = OUString::createFromAscii( flavorMap[i].OOoFlavor);
541 		  oOOFlavor.HumanPresentableName = OUString::createFromAscii( flavorMap[i].HumanPresentableName);
542 		  oOOFlavor.DataType = flavorMap[i].DataType;
543 		  return oOOFlavor;
544 		}
545 	} // for
546 
547 	// look if this might be an internal type; if it comes in here it must have
548 	// been through openOfficeToSystemFlavor before, so it should then be in the map
549 	rtl::OUString aTryFlavor( NSStringToOUString( systemDataFlavor ) );
550 	if( maOfficeOnlyTypes.find( aTryFlavor ) != maOfficeOnlyTypes.end() )
551 	{
552 	    oOOFlavor.MimeType = aTryFlavor;
553 	    oOOFlavor.HumanPresentableName = rtl::OUString();
554 	    oOOFlavor.DataType = CPPUTYPE_SEQINT8;
555 	}
556 
557 	return oOOFlavor;
558 }
559 
560 NSString* DataFlavorMapper::openOfficeToSystemFlavor(const DataFlavor& oOOFlavor) const
561 {
562     NSString* sysFlavor = NULL;
563 
564 	for( size_t i = 0; i < SIZE_FLAVOR_MAP; ++i )
565 	{
566 	    if (oOOFlavor.MimeType.compareToAscii(flavorMap[i].OOoFlavor, strlen(flavorMap[i].OOoFlavor)) == 0)
567 		{
568 		    sysFlavor = flavorMap[i].SystemFlavor;
569 		}
570 	}
571 
572 	if( ! sysFlavor )
573 	{
574 		OfficeOnlyTypes::const_iterator it = maOfficeOnlyTypes.find( oOOFlavor.MimeType );
575 		if( it == maOfficeOnlyTypes.end() )
576 			sysFlavor = maOfficeOnlyTypes[ oOOFlavor.MimeType ] = OUStringToNSString( oOOFlavor.MimeType );
577 		else
578 			sysFlavor = it->second;
579 	}
580 
581 	return sysFlavor;
582 }
583 
584 NSString* DataFlavorMapper::openOfficeImageToSystemFlavor(NSPasteboard* pPasteboard) const
585 {
586     NSArray *supportedTypes = [NSArray arrayWithObjects: NSTIFFPboardType, NSPICTPboardType, nil];
587     NSString *sysFlavor = [pPasteboard availableTypeFromArray:supportedTypes];
588     return sysFlavor;
589 }
590 
591 DataProviderPtr_t DataFlavorMapper::getDataProvider(NSString* systemFlavor, Reference<XTransferable> rTransferable) const
592 {
593   DataProviderPtr_t dp;
594 
595   try
596 	{
597 	  DataFlavor oOOFlavor = systemToOpenOfficeFlavor(systemFlavor);
598 
599 	  Any data = rTransferable->getTransferData(oOOFlavor);
600 
601 	  if (isByteSequenceType(data.getValueType()))
602 		{
603 		  /*
604 		     the HTMLFormatDataProvider prepends segment information to HTML
605 		     this is useful for exchange with MS Word (which brings this stuff from Windows)
606 		     but annoying for other applications. Since this extension is not a standard datatype
607 		     on the Mac, let us not provide but provide normal HTML
608 
609 		  if ([systemFlavor caseInsensitiveCompare: NSHTMLPboardType] == NSOrderedSame)
610 			{
611 			  dp = DataProviderPtr_t(new HTMLFormatDataProvider(data));
612 			}
613 		  else
614 		  */
615 		  if ([systemFlavor caseInsensitiveCompare: NSPICTPboardType] == NSOrderedSame)
616 			{
617 			  dp = DataProviderPtr_t(new BMPDataProvider(data, PICTImageFileType));
618 			}
619 		  else if ([systemFlavor caseInsensitiveCompare: NSTIFFPboardType] == NSOrderedSame)
620 			{
621 			  dp = DataProviderPtr_t(new BMPDataProvider(data, NSTIFFFileType));
622 			}
623 		  else if ([systemFlavor caseInsensitiveCompare: NSFilenamesPboardType] == NSOrderedSame)
624 			{
625 			  dp = DataProviderPtr_t(new FileListDataProvider(data));
626 			}
627 		  else
628 			{
629 			  dp = DataProviderPtr_t(new ByteSequenceDataProvider(data));
630 			}
631 		}
632 	  else // Must be OUString type
633 		{
634 		  BOOST_ASSERT(isOUStringType(data.getValueType()));
635 		  dp = DataProviderPtr_t(new UniDataProvider(data));
636 		}
637 	}
638   catch(UnsupportedFlavorException&)
639 	{
640 	  // Somebody violates the contract of the clipboard
641 	  // interface @see XTransferable
642 	}
643 
644   return dp;
645 }
646 
647 DataProviderPtr_t DataFlavorMapper::getDataProvider(const NSString* /*systemFlavor*/, NSArray* systemData) const
648 {
649   return DataProviderPtr_t(new FileListDataProvider(systemData));
650 }
651 
652 DataProviderPtr_t DataFlavorMapper::getDataProvider(const NSString* systemFlavor, NSData* systemData) const
653 {
654   DataProviderPtr_t dp;
655 
656   if ([systemFlavor caseInsensitiveCompare: NSStringPboardType] == NSOrderedSame)
657 	{
658 	  dp = DataProviderPtr_t(new UniDataProvider(systemData));
659 	}
660   else if ([systemFlavor caseInsensitiveCompare: NSHTMLPboardType] == NSOrderedSame)
661 	{
662 	  dp = DataProviderPtr_t(new HTMLFormatDataProvider(systemData));
663 	}
664   else if ([systemFlavor caseInsensitiveCompare: NSPICTPboardType] == NSOrderedSame)
665 	{
666 	  dp = DataProviderPtr_t(new BMPDataProvider(systemData, PICTImageFileType));
667 	}
668   else if ([systemFlavor caseInsensitiveCompare: NSTIFFPboardType] == NSOrderedSame)
669 	{
670 	  dp = DataProviderPtr_t(new BMPDataProvider(systemData, NSTIFFFileType));
671 	}
672   else if ([systemFlavor caseInsensitiveCompare: NSFilenamesPboardType] == NSOrderedSame)
673 	{
674 	  //dp = DataProviderPtr_t(new FileListDataProvider(systemData));
675 	}
676   else
677 	{
678 	  dp = DataProviderPtr_t(new ByteSequenceDataProvider(systemData));
679 	}
680 
681   return dp;
682 }
683 
684 bool DataFlavorMapper::isValidMimeContentType(const rtl::OUString& contentType) const
685 {
686   bool result = true;
687 
688   try
689 	{
690 	  Reference<XMimeContentType> xCntType(mrXMimeCntFactory->createMimeContentType(contentType));
691 	}
692   catch( IllegalArgumentException& )
693 	{
694 	  result = false;
695 	}
696 
697   return result;
698 }
699 
700 NSArray* DataFlavorMapper::flavorSequenceToTypesArray(const com::sun::star::uno::Sequence<com::sun::star::datatransfer::DataFlavor>& flavors) const
701 {
702   sal_uInt32 nFlavors = flavors.getLength();
703   NSMutableArray* array = [[NSMutableArray alloc] initWithCapacity: 1];
704 
705   for (sal_uInt32 i = 0; i < nFlavors; i++)
706   {
707       if( flavors[i].MimeType.compareToAscii( "image/bmp", 9 ) == 0 )
708       {
709           [array addObject: NSTIFFPboardType];
710           [array addObject: NSPICTPboardType];
711       }
712       else
713       {
714           NSString* str = openOfficeToSystemFlavor(flavors[i]);
715 
716           if (str != NULL)
717           {
718               [str retain];
719               [array addObject: str];
720           }
721       }
722   }
723 
724    // #i89462# #i90747#
725    // in case no system flavor was found to report
726    // report at least one so D&D between OOo targets works
727   if( [array count] == 0 )
728   {
729       [array addObject: PBTYPE_DUMMY_INTERNAL];
730   }
731 
732   return [array autorelease];
733 }
734 
735 com::sun::star::uno::Sequence<com::sun::star::datatransfer::DataFlavor> DataFlavorMapper::typesArrayToFlavorSequence(NSArray* types) const
736 {
737   int nFormats = [types count];
738   Sequence<DataFlavor> flavors;
739 
740   for (int i = 0; i < nFormats; i++)
741 	{
742 	  NSString* sysFormat = [types objectAtIndex: i];
743 	  DataFlavor oOOFlavor = systemToOpenOfficeFlavor(sysFormat);
744 
745 	  if (isValidFlavor(oOOFlavor))
746 		{
747 		  flavors.realloc(flavors.getLength() + 1);
748 		  flavors[flavors.getLength() - 1] = oOOFlavor;
749 		}
750 	}
751 
752   return flavors;
753 }
754 
755 
756 NSArray* DataFlavorMapper::getAllSupportedPboardTypes() const
757 {
758   NSMutableArray* array = [[NSMutableArray alloc] initWithCapacity: SIZE_FLAVOR_MAP];
759 
760   for (sal_uInt32 i = 0; i < SIZE_FLAVOR_MAP; i++)
761 	{
762 	  [array addObject: flavorMap[i].SystemFlavor];
763 	}
764 
765   return [array autorelease];
766 }
767