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
.IsOk();
390 void wxBitmapDataObject::DoConvertToPng()
392 if ( !m_bitmap
.IsOk() )
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