1 ///////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxDataObject class
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 ///////////////////////////////////////////////////////////////////////////////
10 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
11 #pragma implementation "dataobj.h"
14 // For compilers that support precompilation, includes "wx.h".
15 #include "wx/wxprec.h"
17 #include "wx/dataobj.h"
20 #include "wx/mstream.h"
26 //-------------------------------------------------------------------------
28 //-------------------------------------------------------------------------
30 GdkAtom g_textAtom
= 0;
31 GdkAtom g_pngAtom
= 0;
32 GdkAtom g_fileAtom
= 0;
34 //-------------------------------------------------------------------------
36 //-------------------------------------------------------------------------
38 wxDataFormat::wxDataFormat()
40 // do *not* call PrepareFormats() from here for 2 reasons:
42 // 1. we will have time to do it later because some other Set function
43 // must be called before we really need them
45 // 2. doing so prevents us from declaring global wxDataFormats because
46 // calling PrepareFormats (and thus gdk_atom_intern) before GDK is
47 // initialised will result in a crash
48 m_type
= wxDF_INVALID
;
49 m_format
= (GdkAtom
) 0;
52 wxDataFormat::wxDataFormat( wxDataFormatId type
)
58 wxDataFormat::wxDataFormat( const wxChar
*id
)
64 wxDataFormat::wxDataFormat( const wxString
&id
)
70 wxDataFormat::wxDataFormat( NativeFormat format
)
76 void wxDataFormat::SetType( wxDataFormatId type
)
80 if (type
== wxDF_UNICODETEXT
)
85 if (m_type
== wxDF_TEXT
)
86 m_format
= g_textAtom
;
88 if (m_type
== wxDF_BITMAP
)
91 if (m_type
== wxDF_FILENAME
)
92 m_format
= g_fileAtom
;
95 wxFAIL_MSG( wxT("invalid dataformat") );
99 wxDataFormatId
wxDataFormat::GetType() const
104 wxString
wxDataFormat::GetId() const
106 wxString ret
= wxString::FromAscii( gdk_atom_name( m_format
) );
110 void wxDataFormat::SetId( NativeFormat format
)
115 if (m_format
== g_textAtom
)
118 if (m_format
== g_pngAtom
)
119 m_type
= wxDF_BITMAP
;
121 if (m_format
== g_fileAtom
)
122 m_type
= wxDF_FILENAME
;
124 m_type
= wxDF_PRIVATE
;
127 void wxDataFormat::SetId( const wxChar
*id
)
130 m_type
= wxDF_PRIVATE
;
132 m_format
= gdk_atom_intern( (const char*) tmp
.ToAscii(), FALSE
);
135 void wxDataFormat::PrepareFormats()
137 // VZ: GNOME included in RedHat 6.1 uses the MIME types below and not the
138 // atoms STRING and file:ALL as the old code was, but normal X apps
139 // use STRING for text selection when transfering the data via
140 // clipboard, for example, so do use STRING for now (GNOME apps will
141 // probably support STRING as well for compatibility anyhow), but use
142 // text/uri-list for file dnd because compatibility is not important
146 g_textAtom
= gdk_atom_intern( "UTF8_STRING", FALSE
);
148 g_textAtom
= gdk_atom_intern( "STRING" /* "text/plain" */, FALSE
);
151 g_pngAtom
= gdk_atom_intern( "image/png", FALSE
);
153 g_fileAtom
= gdk_atom_intern( "text/uri-list", FALSE
);
156 //-------------------------------------------------------------------------
158 //-------------------------------------------------------------------------
160 wxDataObject::wxDataObject()
164 wxDataObject::~wxDataObject()
166 // dtor is empty but needed for Darwin and AIX -- otherwise it doesn't link
169 bool wxDataObject::IsSupportedFormat(const wxDataFormat
& format
, Direction dir
) const
171 size_t nFormatCount
= GetFormatCount(dir
);
172 if ( nFormatCount
== 1 )
174 return format
== GetPreferredFormat();
178 wxDataFormat
*formats
= new wxDataFormat
[nFormatCount
];
179 GetAllFormats(formats
,dir
);
182 for ( n
= 0; n
< nFormatCount
; n
++ )
184 if ( formats
[n
] == format
)
191 return n
< nFormatCount
;
195 // ----------------------------------------------------------------------------
197 // ----------------------------------------------------------------------------
199 bool wxFileDataObject::GetDataHere(void *buf
) const
203 for (size_t i
= 0; i
< m_filenames
.GetCount(); i
++)
205 filenames
+= wxT("file:");
206 filenames
+= m_filenames
[i
];
207 filenames
+= wxT("\r\n");
210 memcpy( buf
, filenames
.mbc_str(), filenames
.Len() + 1 );
215 size_t wxFileDataObject::GetDataSize() const
219 for (size_t i
= 0; i
< m_filenames
.GetCount(); i
++)
221 // This is junk in UTF-8
222 res
+= m_filenames
[i
].Len();
223 res
+= 5 + 2; // "file:" (5) + "\r\n" (2)
229 bool wxFileDataObject::SetData(size_t WXUNUSED(size
), const void *buf
)
233 // filenames are stores as a string with #0 as deliminators
234 const char *filenames
= (const char*) buf
;
238 if (filenames
[0] == 0)
242 wxString
file( filenames
); // this returns the first file
245 filenames
+= file
.Len()+1;
250 // the text/uri-list format is a sequence of URIs (filenames prefixed by
251 // "file:" as far as I see) delimited by "\r\n" of total length size
252 // (I wonder what happens if the file has '\n' in its filename??)
254 for ( const char *p
= (const char *)buf
; ; p
++ )
256 // some broken programs (testdnd GTK+ sample!) omit the trailing
257 // "\r\n", so check for '\0' explicitly here instead of doing it in
258 // the loop statement to account for it
259 if ( (*p
== '\r' && *(p
+1) == '\n') || !*p
)
261 size_t lenPrefix
= 5; // strlen("file:")
262 if ( filename
.Left(lenPrefix
).MakeLower() == _T("file:") )
264 // sometimes the syntax is "file:filename", sometimes it's
265 // URL-like: "file://filename" - deal with both
266 if ( filename
[lenPrefix
] == _T('/') &&
267 filename
[lenPrefix
+ 1] == _T('/') )
273 AddFile(filename
.c_str() + lenPrefix
);
278 wxLogDebug(_T("Unsupported URI '%s' in wxFileDataObject"),
298 void wxFileDataObject::AddFile( const wxString
&filename
)
300 m_filenames
.Add( filename
);
303 // ----------------------------------------------------------------------------
304 // wxBitmapDataObject
305 // ----------------------------------------------------------------------------
307 wxBitmapDataObject::wxBitmapDataObject()
312 wxBitmapDataObject::wxBitmapDataObject( const wxBitmap
& bitmap
)
313 : wxBitmapDataObjectBase(bitmap
)
320 wxBitmapDataObject::~wxBitmapDataObject()
325 void wxBitmapDataObject::SetBitmap( const wxBitmap
&bitmap
)
329 wxBitmapDataObjectBase::SetBitmap(bitmap
);
334 bool wxBitmapDataObject::GetDataHere(void *buf
) const
338 wxFAIL_MSG( wxT("attempt to copy empty bitmap failed") );
343 memcpy(buf
, m_pngData
, m_pngSize
);
348 bool wxBitmapDataObject::SetData(size_t size
, const void *buf
)
352 wxCHECK_MSG( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
,
353 FALSE
, wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") );
356 m_pngData
= malloc(m_pngSize
);
358 memcpy(m_pngData
, buf
, m_pngSize
);
360 wxMemoryInputStream
mstream((char*) m_pngData
, m_pngSize
);
362 if ( !image
.LoadFile( mstream
, wxBITMAP_TYPE_PNG
) )
367 m_bitmap
= wxBitmap(image
);
369 return m_bitmap
.Ok();
372 void wxBitmapDataObject::DoConvertToPng()
374 if ( !m_bitmap
.Ok() )
377 wxCHECK_RET( wxImage::FindHandler(wxBITMAP_TYPE_PNG
) != NULL
,
378 wxT("You must call wxImage::AddHandler(new wxPNGHandler); to be able to use clipboard with bitmaps!") );
380 wxImage image
= m_bitmap
.ConvertToImage();
382 wxCountingOutputStream count
;
383 image
.SaveFile(count
, wxBITMAP_TYPE_PNG
);
385 m_pngSize
= count
.GetSize() + 100; // sometimes the size seems to vary ???
386 m_pngData
= malloc(m_pngSize
);
388 wxMemoryOutputStream
mstream((char*) m_pngData
, m_pngSize
);
389 image
.SaveFile(mstream
, wxBITMAP_TYPE_PNG
);