1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/dataobj.cpp 
   3 // Purpose:     wxDataObject class 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 /////////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  15 #include "wx/dataobj.h" 
  23 #include "wx/mstream.h" 
  26 #include "wx/gtk/private.h" 
  28 //------------------------------------------------------------------------- 
  30 //------------------------------------------------------------------------- 
  32 GdkAtom  g_textAtom        
= 0; 
  33 GdkAtom  g_altTextAtom     
= 0; 
  34 GdkAtom  g_pngAtom         
= 0; 
  35 GdkAtom  g_fileAtom        
= 0; 
  37 //------------------------------------------------------------------------- 
  39 //------------------------------------------------------------------------- 
  41 wxDataFormat::wxDataFormat() 
  43     // do *not* call PrepareFormats() from here for 2 reasons: 
  45     // 1. we will have time to do it later because some other Set function 
  46     //    must be called before we really need them 
  48     // 2. doing so prevents us from declaring global wxDataFormats because 
  49     //    calling PrepareFormats (and thus gdk_atom_intern) before GDK is 
  50     //    initialised will result in a crash 
  51     m_type 
= wxDF_INVALID
; 
  52     m_format 
= (GdkAtom
) 0; 
  55 wxDataFormat::wxDataFormat( wxDataFormatId type 
) 
  61 wxDataFormat::wxDataFormat( const wxChar 
*id 
) 
  67 wxDataFormat::wxDataFormat( const wxString 
&id 
) 
  73 wxDataFormat::wxDataFormat( NativeFormat format 
) 
  79 void wxDataFormat::SetType( wxDataFormatId type 
) 
  86     if (m_type 
== wxDF_UNICODETEXT
) 
  87         m_format 
= g_textAtom
; 
  88     else if (m_type 
== wxDF_TEXT
) 
  89         m_format 
= g_altTextAtom
; 
  91     if (m_type 
== wxDF_TEXT 
|| m_type 
== wxDF_UNICODETEXT
) 
  92         m_format 
= g_textAtom
; 
  95     if (m_type 
== wxDF_BITMAP
) 
  98     if (m_type 
== wxDF_FILENAME
) 
  99         m_format 
= g_fileAtom
; 
 102        wxFAIL_MSG( wxT("invalid dataformat") ); 
 106 wxDataFormatId 
wxDataFormat::GetType() const 
 111 wxString 
wxDataFormat::GetId() const 
 113     wxGtkString 
atom_name(gdk_atom_name(m_format
)); 
 114     return wxString::FromAscii(atom_name
); 
 117 void wxDataFormat::SetId( NativeFormat format 
) 
 122     if (m_format 
== g_textAtom
) 
 124         m_type 
= wxDF_UNICODETEXT
; 
 129     if (m_format 
== g_altTextAtom
) 
 132     if (m_format 
== g_pngAtom
) 
 133         m_type 
= wxDF_BITMAP
; 
 135     if (m_format 
== g_fileAtom
) 
 136         m_type 
= wxDF_FILENAME
; 
 138         m_type 
= wxDF_PRIVATE
; 
 141 void wxDataFormat::SetId( const wxChar 
*id 
) 
 144     m_type 
= wxDF_PRIVATE
; 
 146     m_format 
= gdk_atom_intern( (const char*) tmp
.ToAscii(), FALSE 
); 
 149 void wxDataFormat::PrepareFormats() 
 151     // VZ: GNOME included in RedHat 6.1 uses the MIME types below and not the 
 152     //     atoms STRING and file:ALL as the old code was, but normal X apps 
 153     //     use STRING for text selection when transfering the data via 
 154     //     clipboard, for example, so do use STRING for now (GNOME apps will 
 155     //     probably support STRING as well for compatibility anyhow), but use 
 156     //     text/uri-list for file dnd because compatibility is not important 
 160         g_textAtom 
= gdk_atom_intern( "UTF8_STRING", FALSE 
); 
 161         g_altTextAtom 
= gdk_atom_intern( "STRING", FALSE 
); 
 163         g_textAtom 
= gdk_atom_intern( "STRING" /* "text/plain" */, FALSE 
); 
 166         g_pngAtom 
= gdk_atom_intern( "image/png", FALSE 
); 
 168         g_fileAtom 
= gdk_atom_intern( "text/uri-list", FALSE 
); 
 171 //------------------------------------------------------------------------- 
 173 //------------------------------------------------------------------------- 
 175 wxDataObject::wxDataObject() 
 179 wxDataObject::~wxDataObject() 
 181     // dtor is empty but needed for Darwin and AIX -- otherwise it doesn't link 
 184 bool wxDataObject::IsSupportedFormat(const wxDataFormat
& format
, Direction dir
) const 
 186     size_t nFormatCount 
= GetFormatCount(dir
); 
 187     if ( nFormatCount 
== 1 ) 
 189         return format 
== GetPreferredFormat(); 
 193         wxDataFormat 
*formats 
= new wxDataFormat
[nFormatCount
]; 
 194         GetAllFormats(formats
,dir
); 
 197         for ( n 
= 0; n 
< nFormatCount
; n
++ ) 
 199             if ( formats
[n
] == format 
) 
 206         return n 
< nFormatCount
; 
 210 // ---------------------------------------------------------------------------- 
 212 // ---------------------------------------------------------------------------- 
 214 #if defined(__WXGTK20__) && wxUSE_UNICODE 
 215 void wxTextDataObject::GetAllFormats(wxDataFormat 
*formats
, wxDataObjectBase::Direction dir
) const 
 217     *formats
++ = GetPreferredFormat(); 
 218     *formats 
= g_altTextAtom
; 
 222 // ---------------------------------------------------------------------------- 
 224 // ---------------------------------------------------------------------------- 
 226 bool wxFileDataObject::GetDataHere(void *buf
) const 
 230     for (size_t i 
= 0; i 
< m_filenames
.GetCount(); i
++) 
 232         filenames 
+= wxT("file:"); 
 233         filenames 
+= m_filenames
[i
]; 
 234         filenames 
+= wxT("\r\n"); 
 237     memcpy( buf
, filenames
.mbc_str(), filenames
.length() + 1 ); 
 242 size_t wxFileDataObject::GetDataSize() const 
 246     for (size_t i 
= 0; i 
< m_filenames
.GetCount(); i
++) 
 248         // This is junk in UTF-8 
 249         res 
+= m_filenames
[i
].length(); 
 250         res 
+= 5 + 2; // "file:" (5) + "\r\n" (2) 
 256 bool wxFileDataObject::SetData(size_t WXUNUSED(size
), const void *buf
) 
 260     // we get data in the text/uri-list format, i.e. as a sequence of URIs 
 261     // (filenames prefixed by "file:") delimited by "\r\n" 
 263     for ( const char *p 
= (const char *)buf
; ; p
++ ) 
 265         // some broken programs (testdnd GTK+ sample!) omit the trailing 
 266         // "\r\n", so check for '\0' explicitly here instead of doing it in 
 267         // the loop statement to account for it 
 268         if ( (*p 
== '\r' && *(p
+1) == '\n') || !*p 
) 
 270             size_t lenPrefix 
= 5; // strlen("file:") 
 271             if ( filename
.Left(lenPrefix
).MakeLower() == _T("file:") ) 
 273                 // sometimes the syntax is "file:filename", sometimes it's 
 274                 // URL-like: "file://filename" - deal with both 
 275                 if ( filename
[lenPrefix
] == _T('/') && 
 276                      filename
[lenPrefix 
+ 1] == _T('/') ) 
 282                 // It would probably be nicer to use a GTK or Glib 
 283                 // function to unescape the 8-bit strings pointed to 
 284                 // by buf, but this does the same in wx code. 
 285                 wxString filename_unicode 
= wxURI::Unescape(filename
.c_str() + lenPrefix
); 
 286                 wxCharBuffer filename_8bit 
= wxConvISO8859_1
.cWX2MB( filename_unicode 
); 
 287                 filename_unicode 
= wxConvFileName
->cMB2WX( filename_8bit 
); 
 288                 AddFile( filename_unicode 
); 
 291             else if ( !filename
.empty() ) 
 293                 wxLogDebug(_T("Unsupported URI \"%s\" in wxFileDataObject"), 
 305             // The string is in ISO-8859-1 according to XDND spec 
 313 void wxFileDataObject::AddFile( const wxString 
&filename 
) 
 315    m_filenames
.Add( filename 
); 
 318 // ---------------------------------------------------------------------------- 
 319 // wxBitmapDataObject 
 320 // ---------------------------------------------------------------------------- 
 322 wxBitmapDataObject::wxBitmapDataObject() 
 327 wxBitmapDataObject::wxBitmapDataObject( const wxBitmap
& bitmap 
) 
 328                   : wxBitmapDataObjectBase(bitmap
) 
 335 wxBitmapDataObject::~wxBitmapDataObject() 
 340 void wxBitmapDataObject::SetBitmap( const wxBitmap 
&bitmap 
) 
 344     wxBitmapDataObjectBase::SetBitmap(bitmap
); 
 349 bool wxBitmapDataObject::GetDataHere(void *buf
) const 
 353         wxFAIL_MSG( wxT("attempt to copy empty bitmap failed") ); 
 358     memcpy(buf
, m_pngData
, m_pngSize
); 
 363 bool wxBitmapDataObject::SetData(size_t size
, const void *buf
) 
 367     wxCHECK_MSG( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
, 
 368                  false, wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") ); 
 371     m_pngData 
= malloc(m_pngSize
); 
 373     memcpy(m_pngData
, buf
, m_pngSize
); 
 375     wxMemoryInputStream 
mstream((char*) m_pngData
, m_pngSize
); 
 377     if ( !image
.LoadFile( mstream
, wxBITMAP_TYPE_PNG 
) ) 
 382     m_bitmap 
= wxBitmap(image
); 
 384     return m_bitmap
.Ok(); 
 387 void wxBitmapDataObject::DoConvertToPng() 
 389     if ( !m_bitmap
.Ok() ) 
 392     wxCHECK_RET( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
, 
 393                  wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") ); 
 395     wxImage image 
= m_bitmap
.ConvertToImage(); 
 397     wxCountingOutputStream count
; 
 398     image
.SaveFile(count
, wxBITMAP_TYPE_PNG
); 
 400     m_pngSize 
= count
.GetSize() + 100; // sometimes the size seems to vary ??? 
 401     m_pngData 
= malloc(m_pngSize
); 
 403     wxMemoryOutputStream 
mstream((char*) m_pngData
, m_pngSize
); 
 404     image
.SaveFile(mstream
, wxBITMAP_TYPE_PNG
); 
 407 #endif // wxUSE_DATAOBJ