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