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 void wxDataFormat::InitFromString( const wxString 
&id 
) 
  67 wxDataFormat::wxDataFormat( NativeFormat format 
) 
  73 void wxDataFormat::SetType( wxDataFormatId type 
) 
  80     if (m_type 
== wxDF_UNICODETEXT
) 
  81         m_format 
= g_textAtom
; 
  82     else if (m_type 
== wxDF_TEXT
) 
  83         m_format 
= g_altTextAtom
; 
  84 #else // !wxUSE_UNICODE 
  85     // notice that we don't map wxDF_UNICODETEXT to g_textAtom here, this 
  86     // would lead the code elsewhere to treat data objects with this format as 
  87     // containing UTF-8 data which is not true 
  88     if (m_type 
== wxDF_TEXT
) 
  89         m_format 
= g_textAtom
; 
  90 #endif // wxUSE_UNICODE/!wxUSE_UNICODE 
  92     if (m_type 
== wxDF_BITMAP
) 
  95     if (m_type 
== wxDF_FILENAME
) 
  96         m_format 
= g_fileAtom
; 
  99        wxFAIL_MSG( wxT("invalid dataformat") ); 
 103 wxDataFormatId 
wxDataFormat::GetType() const 
 108 wxString 
wxDataFormat::GetId() const 
 110     wxGtkString 
atom_name(gdk_atom_name(m_format
)); 
 111     return wxString::FromAscii(atom_name
); 
 114 void wxDataFormat::SetId( NativeFormat format 
) 
 119     if (m_format 
== g_textAtom
) 
 121         m_type 
= wxDF_UNICODETEXT
; 
 126     if (m_format 
== g_altTextAtom
) 
 129     if (m_format 
== g_pngAtom
) 
 130         m_type 
= wxDF_BITMAP
; 
 132     if (m_format 
== g_fileAtom
) 
 133         m_type 
= wxDF_FILENAME
; 
 135         m_type 
= wxDF_PRIVATE
; 
 138 void wxDataFormat::SetId( const wxString
& id 
) 
 141     m_type 
= wxDF_PRIVATE
; 
 142     m_format 
= gdk_atom_intern( id
.ToAscii(), FALSE 
); 
 145 void wxDataFormat::PrepareFormats() 
 147     // VZ: GNOME included in RedHat 6.1 uses the MIME types below and not the 
 148     //     atoms STRING and file:ALL as the old code was, but normal X apps 
 149     //     use STRING for text selection when transferring the data via 
 150     //     clipboard, for example, so do use STRING for now (GNOME apps will 
 151     //     probably support STRING as well for compatibility anyhow), but use 
 152     //     text/uri-list for file dnd because compatibility is not important 
 157         g_textAtom 
= gdk_atom_intern( "UTF8_STRING", FALSE 
); 
 158         g_altTextAtom 
= gdk_atom_intern( "STRING", FALSE 
); 
 160         g_textAtom 
= gdk_atom_intern( "STRING" /* "text/plain" */, FALSE 
); 
 164         g_pngAtom 
= gdk_atom_intern( "image/png", FALSE 
); 
 166         g_fileAtom 
= gdk_atom_intern( "text/uri-list", FALSE 
); 
 169 //------------------------------------------------------------------------- 
 171 //------------------------------------------------------------------------- 
 173 wxDataObject::wxDataObject() 
 177 wxDataObject::~wxDataObject() 
 179     // dtor is empty but needed for Darwin and AIX -- otherwise it doesn't link 
 182 bool wxDataObject::IsSupportedFormat(const wxDataFormat
& format
, Direction dir
) const 
 184     size_t nFormatCount 
= GetFormatCount(dir
); 
 185     if ( nFormatCount 
== 1 ) 
 187         return format 
== GetPreferredFormat(); 
 191         wxDataFormat 
*formats 
= new wxDataFormat
[nFormatCount
]; 
 192         GetAllFormats(formats
,dir
); 
 195         for ( n 
= 0; n 
< nFormatCount
; n
++ ) 
 197             if ( formats
[n
] == format 
) 
 204         return n 
< nFormatCount
; 
 208 // ---------------------------------------------------------------------------- 
 210 // ---------------------------------------------------------------------------- 
 215 wxTextDataObject::GetAllFormats(wxDataFormat 
*formats
, 
 216                                 wxDataObjectBase::Direction 
WXUNUSED(dir
)) const 
 218     *formats
++ = GetPreferredFormat(); 
 219     *formats 
= g_altTextAtom
; 
 222 #endif // wxUSE_UNICODE 
 224 // ---------------------------------------------------------------------------- 
 226 // ---------------------------------------------------------------------------- 
 228 bool wxFileDataObject::GetDataHere(void *buf
) const 
 232     for (size_t i 
= 0; i 
< m_filenames
.GetCount(); i
++) 
 234         filenames 
+= wxT("file:"); 
 235         filenames 
+= m_filenames
[i
]; 
 236         filenames 
+= wxT("\r\n"); 
 239     memcpy( buf
, filenames
.mbc_str(), filenames
.length() + 1 ); 
 244 size_t wxFileDataObject::GetDataSize() const 
 248     for (size_t i 
= 0; i 
< m_filenames
.GetCount(); i
++) 
 250         // This is junk in UTF-8 
 251         res 
+= m_filenames
[i
].length(); 
 252         res 
+= 5 + 2; // "file:" (5) + "\r\n" (2) 
 258 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". size includes 
 262     // the trailing zero (in theory, not for Nautilus in early GNOME 
 267     const gchar 
*nexttemp 
= (const gchar
*) buf
; 
 271         const gchar 
*temp 
= nexttemp
; 
 278                     // if an app omits '\r''\n' 
 285             if (temp
[len
] == '\r') 
 287                 if (temp
[len
+1] == '\n') 
 288                     nexttemp 
= temp
+len
+2; 
 290                     nexttemp 
= temp
+len
+1; 
 299         // required to give it a trailing zero 
 300         gchar 
*uri 
= g_strndup( temp
, len 
); 
 302         gchar 
*fn 
= g_filename_from_uri( uri
, NULL
, NULL 
); 
 308             AddFile( wxConvFileName
->cMB2WX( fn 
) ); 
 316 void wxFileDataObject::AddFile( const wxString 
&filename 
) 
 318    m_filenames
.Add( filename 
); 
 321 // ---------------------------------------------------------------------------- 
 322 // wxBitmapDataObject 
 323 // ---------------------------------------------------------------------------- 
 325 wxBitmapDataObject::wxBitmapDataObject() 
 330 wxBitmapDataObject::wxBitmapDataObject( const wxBitmap
& bitmap 
) 
 331                   : wxBitmapDataObjectBase(bitmap
) 
 338 wxBitmapDataObject::~wxBitmapDataObject() 
 343 void wxBitmapDataObject::SetBitmap( const wxBitmap 
&bitmap 
) 
 347     wxBitmapDataObjectBase::SetBitmap(bitmap
); 
 352 bool wxBitmapDataObject::GetDataHere(void *buf
) const 
 356         wxFAIL_MSG( wxT("attempt to copy empty bitmap failed") ); 
 361     memcpy(buf
, m_pngData
, m_pngSize
); 
 366 bool wxBitmapDataObject::SetData(size_t size
, const void *buf
) 
 370     wxCHECK_MSG( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
, 
 371                  false, wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") ); 
 374     m_pngData 
= malloc(m_pngSize
); 
 376     memcpy(m_pngData
, buf
, m_pngSize
); 
 378     wxMemoryInputStream 
mstream((char*) m_pngData
, m_pngSize
); 
 380     if ( !image
.LoadFile( mstream
, wxBITMAP_TYPE_PNG 
) ) 
 385     m_bitmap 
= wxBitmap(image
); 
 387     return m_bitmap
.Ok(); 
 390 void wxBitmapDataObject::DoConvertToPng() 
 392     if ( !m_bitmap
.Ok() ) 
 395     wxCHECK_RET( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
, 
 396                  wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") ); 
 398     wxImage image 
= m_bitmap
.ConvertToImage(); 
 400     wxCountingOutputStream count
; 
 401     image
.SaveFile(count
, wxBITMAP_TYPE_PNG
); 
 403     m_pngSize 
= count
.GetSize() + 100; // sometimes the size seems to vary ??? 
 404     m_pngData 
= malloc(m_pngSize
); 
 406     wxMemoryOutputStream 
mstream((char*) m_pngData
, m_pngSize
); 
 407     image
.SaveFile(mstream
, wxBITMAP_TYPE_PNG
); 
 410 // ---------------------------------------------------------------------------- 
 412 // ---------------------------------------------------------------------------- 
 414 wxURLDataObject::wxURLDataObject(const wxString
& url
) : 
 415    wxDataObjectSimple( wxDataFormat( gdk_atom_intern("text/x-moz-url",FALSE
) ) ) 
 420 size_t wxURLDataObject::GetDataSize() const 
 425     return 2*m_url
.Len()+2; 
 428 bool wxURLDataObject::GetDataHere(void *buf
) const 
 433     wxCSConv 
conv( "UCS2" ); 
 434     conv
.FromWChar( (char*) buf
, 2*m_url
.Len()+2, m_url
.wc_str() ); 
 439     // copy data from buffer to our data 
 440 bool wxURLDataObject::SetData(size_t len
, const void *buf
) 
 444         m_url 
= wxEmptyString
; 
 448     wxCSConv 
conv( "UCS2" ); 
 449     wxWCharBuffer res 
= conv
.cMB2WC( (const char*) buf 
); 
 451     int pos 
= m_url
.Find( '\n' ); 
 452     if (pos 
!= wxNOT_FOUND
) 
 453         m_url
.Remove( pos
, m_url
.Len() - pos 
); 
 459 #endif // wxUSE_DATAOBJ