1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: common/dobjcmn.cpp
3 // Purpose: implementation of data object methods common to all platforms
4 // Author: Vadim Zeitlin, Robert Roebling
8 // Copyright: (c) wxWidgets Team
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
21 #pragma implementation "dataobjbase.h"
24 #include "wx/wxprec.h"
37 #include "wx/dataobj.h"
39 // ----------------------------------------------------------------------------
41 // ----------------------------------------------------------------------------
43 #include "wx/listimpl.cpp"
45 WX_DEFINE_LIST(wxSimpleDataObjectList
);
47 // ----------------------------------------------------------------------------
49 // ----------------------------------------------------------------------------
51 static wxDataFormat dataFormatInvalid
;
52 WXDLLEXPORT
const wxDataFormat
& wxFormatInvalid
= dataFormatInvalid
;
54 // ============================================================================
56 // ============================================================================
58 // ----------------------------------------------------------------------------
60 // ----------------------------------------------------------------------------
62 wxDataObjectBase::~wxDataObjectBase()
66 bool wxDataObjectBase::IsSupported(const wxDataFormat
& format
,
69 size_t nFormatCount
= GetFormatCount(dir
);
70 if ( nFormatCount
== 1 )
72 return format
== GetPreferredFormat(dir
);
76 wxDataFormat
*formats
= new wxDataFormat
[nFormatCount
];
77 GetAllFormats(formats
, dir
);
80 for ( n
= 0; n
< nFormatCount
; n
++ )
82 if ( formats
[n
] == format
)
89 return n
< nFormatCount
;
93 // ----------------------------------------------------------------------------
94 // wxDataObjectComposite
95 // ----------------------------------------------------------------------------
97 wxDataObjectComposite::wxDataObjectComposite()
102 wxDataObjectComposite::~wxDataObjectComposite()
104 WX_CLEAR_LIST(wxSimpleDataObjectList
, m_dataObjects
);
108 wxDataObjectComposite::GetObject(const wxDataFormat
& format
) const
110 wxSimpleDataObjectList::compatibility_iterator node
= m_dataObjects
.GetFirst();
113 wxDataObjectSimple
*dataObj
= node
->GetData();
115 if ( dataObj
->GetFormat() == format
)
120 node
= node
->GetNext();
123 return (wxDataObjectSimple
*)NULL
;
126 void wxDataObjectComposite::Add(wxDataObjectSimple
*dataObject
, bool preferred
)
129 m_preferred
= m_dataObjects
.GetCount();
131 m_dataObjects
.Append( dataObject
);
135 wxDataObjectComposite::GetPreferredFormat(Direction
WXUNUSED(dir
)) const
137 wxSimpleDataObjectList::compatibility_iterator node
= m_dataObjects
.Item( m_preferred
);
139 wxCHECK_MSG( node
, wxFormatInvalid
, wxT("no preferred format") );
141 wxDataObjectSimple
* dataObj
= node
->GetData();
143 return dataObj
->GetFormat();
146 #if defined(__WXMSW__)
148 size_t wxDataObjectComposite::GetBufferOffset( const wxDataFormat
& format
)
150 wxDataObjectSimple
*dataObj
= GetObject(format
);
152 wxCHECK_MSG( dataObj
, 0,
153 wxT("unsupported format in wxDataObjectComposite"));
155 return dataObj
->GetBufferOffset( format
);
159 const void* wxDataObjectComposite::GetSizeFromBuffer( const void* buffer
,
161 const wxDataFormat
& format
)
163 wxDataObjectSimple
*dataObj
= GetObject(format
);
165 wxCHECK_MSG( dataObj
, NULL
,
166 wxT("unsupported format in wxDataObjectComposite"));
168 return dataObj
->GetSizeFromBuffer( buffer
, size
, format
);
172 void* wxDataObjectComposite::SetSizeInBuffer( void* buffer
, size_t size
,
173 const wxDataFormat
& format
)
175 wxDataObjectSimple
*dataObj
= GetObject(format
);
177 wxCHECK_MSG( dataObj
, NULL
,
178 wxT("unsupported format in wxDataObjectComposite"));
180 return dataObj
->SetSizeInBuffer( buffer
, size
, format
);
185 size_t wxDataObjectComposite::GetFormatCount(Direction
WXUNUSED(dir
)) const
187 // TODO what about the Get/Set only formats?
188 return m_dataObjects
.GetCount();
191 void wxDataObjectComposite::GetAllFormats(wxDataFormat
*formats
,
192 Direction
WXUNUSED(dir
)) const
195 wxSimpleDataObjectList::compatibility_iterator node
;
196 for ( node
= m_dataObjects
.GetFirst(); node
; node
= node
->GetNext() )
198 // TODO if ( !outputOnlyToo ) && this one counts ...
199 formats
[n
++] = node
->GetData()->GetFormat();
203 size_t wxDataObjectComposite::GetDataSize(const wxDataFormat
& format
) const
205 wxDataObjectSimple
*dataObj
= GetObject(format
);
207 wxCHECK_MSG( dataObj
, 0,
208 wxT("unsupported format in wxDataObjectComposite"));
210 return dataObj
->GetDataSize();
213 bool wxDataObjectComposite::GetDataHere(const wxDataFormat
& format
,
216 wxDataObjectSimple
*dataObj
= GetObject(format
);
218 wxCHECK_MSG( dataObj
, FALSE
,
219 wxT("unsupported format in wxDataObjectComposite"));
221 return dataObj
->GetDataHere(buf
);
224 bool wxDataObjectComposite::SetData(const wxDataFormat
& format
,
228 wxDataObjectSimple
*dataObj
= GetObject(format
);
230 wxCHECK_MSG( dataObj
, FALSE
,
231 wxT("unsupported format in wxDataObjectComposite"));
233 return dataObj
->SetData(len
, buf
);
236 // ----------------------------------------------------------------------------
238 // ----------------------------------------------------------------------------
240 #if defined(__WXGTK20__) && wxUSE_UNICODE
242 size_t wxTextDataObject::GetDataSize(const wxDataFormat
& format
) const
244 if (format
== wxDF_UNICODETEXT
)
247 wxCharBuffer buffer
= wxConvUTF8
.cWX2MB( GetText().c_str() );
248 return strlen( (const char*) buffer
) + 1;
252 wxCharBuffer buffer
= wxConvLibc
.cWX2MB( GetText().c_str() );
253 return strlen( (const char*) buffer
) + 1;
257 bool wxTextDataObject::GetDataHere(const wxDataFormat
& format
, void *buf
) const
259 if (format
== wxDF_UNICODETEXT
)
262 wxCharBuffer buffer
= wxConvUTF8
.cWX2MB( GetText().c_str() );
263 strcpy( (char*) buf
, (const char*) buffer
);
267 wxCharBuffer buffer
= wxConvLibc
.cWX2MB( GetText().c_str() );
268 strcpy( (char*) buf
, (const char*) buffer
);
274 bool wxTextDataObject::SetData(const wxDataFormat
& format
,
275 size_t WXUNUSED(len
), const void *buf
)
277 if (format
== wxDF_UNICODETEXT
)
278 SetText( wxConvUTF8
.cMB2WX( (const char*) buf
) );
280 SetText( wxConvLibc
.cMB2WX( (const char*) buf
) );
285 #elif wxUSE_UNICODE && defined(__WXMAC__)
287 size_t wxTextDataObject::GetDataSize(const wxDataFormat
& format
) const
289 if (format
== wxDF_UNICODETEXT
)
291 // host native is UTF16
292 wxMBConvUTF16BE converter
;
293 return converter
.WC2MB( NULL
, GetText().c_str() , 0 ) + 2; // add space for trailing unichar 0
297 wxCharBuffer buffer
= wxConvLibc
.cWX2MB( GetText().c_str() );
298 return strlen( (const char*) buffer
) + 1;
302 bool wxTextDataObject::GetDataHere(const wxDataFormat
& format
, void *buf
) const
304 if (format
== wxDF_UNICODETEXT
)
306 // host native is UTF16
307 wxMBConvUTF16BE converter
;
308 size_t len
= converter
.WC2MB( NULL
, GetText().c_str() , 0 ) ;
309 wxCharBuffer buffer
= converter
.cWX2MB( GetText().c_str() );
310 memcpy( (char*) buf
, (const char*) buffer
, len
+ 2); // trailing unichar 0
314 wxCharBuffer buffer
= wxConvLibc
.cWX2MB( GetText().c_str() );
315 strcpy( (char*) buf
, (const char*) buffer
);
321 bool wxTextDataObject::SetData(const wxDataFormat
& format
,
322 size_t WXUNUSED(len
), const void *buf
)
324 if (format
== wxDF_UNICODETEXT
)
326 // host native is UTF16
327 wxMBConvUTF16BE converter
;
328 SetText( converter
.cMB2WX( (const char*) buf
) );
331 SetText( wxConvLibc
.cMB2WX( (const char*) buf
) );
338 size_t wxTextDataObject::GetDataSize() const
340 return GetTextLength() * sizeof(wxChar
);
343 bool wxTextDataObject::GetDataHere(void *buf
) const
345 wxStrcpy((wxChar
*)buf
, GetText().c_str());
350 bool wxTextDataObject::SetData(size_t WXUNUSED(len
), const void *buf
)
352 SetText(wxString((const wxChar
*)buf
));
359 // ----------------------------------------------------------------------------
360 // wxFileDataObjectBase
361 // ----------------------------------------------------------------------------
363 // VZ: I don't need this in MSW finally, so if it is needed in wxGTK, it should
364 // be moved to gtk/dataobj.cpp
367 wxString
wxFileDataObjectBase::GetFilenames() const
370 size_t count
= m_filenames
.GetCount();
371 for ( size_t n
= 0; n
< count
; n
++ )
373 str
<< m_filenames
[n
] << wxT('\0');
379 void wxFileDataObjectBase::SetFilenames(const wxChar
* filenames
)
384 for ( const wxChar
*pc
= filenames
; ; pc
++ )
394 // 2 consecutive NULs - this is the end of the string
398 m_filenames
.Add(current
);
406 // ----------------------------------------------------------------------------
407 // wxCustomDataObject
408 // ----------------------------------------------------------------------------
410 wxCustomDataObject::wxCustomDataObject(const wxDataFormat
& format
)
411 : wxDataObjectSimple(format
)
413 m_data
= (void *)NULL
;
416 wxCustomDataObject::~wxCustomDataObject()
421 void wxCustomDataObject::TakeData(size_t size
, void *data
)
429 void *wxCustomDataObject::Alloc(size_t size
)
431 return (void *)new char[size
];
434 void wxCustomDataObject::Free()
436 delete [] (char *)m_data
;
438 m_data
= (void *)NULL
;
441 size_t wxCustomDataObject::GetDataSize() const
446 bool wxCustomDataObject::GetDataHere(void *buf
) const
448 void *data
= GetData();
452 memcpy(buf
, data
, GetSize());
457 bool wxCustomDataObject::SetData(size_t size
, const void *buf
)
461 m_data
= Alloc(size
);
465 memcpy(m_data
, buf
, m_size
= size
);
470 // ============================================================================
471 // some common dnd related code
472 // ============================================================================
474 #if wxUSE_DRAG_AND_DROP
478 // ----------------------------------------------------------------------------
480 // ----------------------------------------------------------------------------
482 // NB: we can't use "new" in ctor initializer lists because this provokes an
483 // internal compiler error with VC++ 5.0 (hey, even gcc compiles this!),
484 // so use SetDataObject() instead
486 wxTextDropTarget::wxTextDropTarget()
488 SetDataObject(new wxTextDataObject
);
491 wxDragResult
wxTextDropTarget::OnData(wxCoord x
, wxCoord y
, wxDragResult def
)
496 wxTextDataObject
*dobj
= (wxTextDataObject
*)m_dataObject
;
497 return OnDropText(x
, y
, dobj
->GetText()) ? def
: wxDragNone
;
500 // ----------------------------------------------------------------------------
502 // ----------------------------------------------------------------------------
504 wxFileDropTarget::wxFileDropTarget()
506 SetDataObject(new wxFileDataObject
);
509 wxDragResult
wxFileDropTarget::OnData(wxCoord x
, wxCoord y
, wxDragResult def
)
514 wxFileDataObject
*dobj
= (wxFileDataObject
*)m_dataObject
;
515 return OnDropFiles(x
, y
, dobj
->GetFilenames()) ? def
: wxDragNone
;
518 #endif // wxUSE_DRAG_AND_DROP
520 #endif // wxUSE_DATAOBJ