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 static inline wxMBConv
& GetConv(const wxDataFormat
& format
)
244 // use UTF8 for wxDF_UNICODETEXT and UCS4 for wxDF_TEXT
245 return format
== wxDF_UNICODETEXT
? wxConvUTF8
: wxConvLibc
;
248 size_t wxTextDataObject::GetDataSize(const wxDataFormat
& format
) const
250 wxCharBuffer buffer
= GetConv(format
).cWX2MB( GetText().c_str() );
251 return buffer
? strlen(buffer
) + 1 : 0;
254 bool wxTextDataObject::GetDataHere(const wxDataFormat
& format
, void *buf
) const
256 wxCharBuffer buffer
= GetConv(format
).cWX2MB( GetText().c_str() );
260 strcpy( (char*) buf
, buffer
);
265 bool wxTextDataObject::SetData(const wxDataFormat
& format
,
266 size_t WXUNUSED(len
), const void *buf
)
268 wxWCharBuffer buffer
= GetConv(format
).cMB2WX((const char *)buf
);
277 #elif wxUSE_UNICODE && defined(__WXMAC__)
279 static wxMBConvUTF16 sUTF16Converter
;
281 static inline wxMBConv
& GetConv(const wxDataFormat
& format
)
283 return format
== wxDF_UNICODETEXT
? sUTF16Converter
: (wxMBConv
&) wxConvLocal
;
286 size_t wxTextDataObject::GetDataSize(const wxDataFormat
& format
) const
288 size_t len
= GetConv(format
).WC2MB( NULL
, GetText().c_str() , 0 )
289 + ( format
== wxDF_UNICODETEXT
? 2 : 1 ) ;
293 bool wxTextDataObject::GetDataHere(const wxDataFormat
& format
, void *buf
) const
295 wxCharBuffer buffer
= GetConv(format
).cWX2MB( GetText().c_str() );
299 size_t len
= GetConv(format
).WC2MB( NULL
, GetText().c_str() , 0 )
300 + ( format
== wxDF_UNICODETEXT
? 2 : 1 ) ;
302 memcpy( (char*) buf
, (const char*) buffer
, len
); // trailing (uni)char 0
307 bool wxTextDataObject::SetData(const wxDataFormat
& format
,
308 size_t WXUNUSED(len
), const void *buf
)
310 wxWCharBuffer buffer
= GetConv(format
).cMB2WX((const char *)buf
);
321 size_t wxTextDataObject::GetDataSize() const
323 return GetTextLength() * sizeof(wxChar
);
326 bool wxTextDataObject::GetDataHere(void *buf
) const
328 wxStrcpy((wxChar
*)buf
, GetText().c_str());
333 bool wxTextDataObject::SetData(size_t WXUNUSED(len
), const void *buf
)
335 SetText(wxString((const wxChar
*)buf
));
342 // ----------------------------------------------------------------------------
343 // wxFileDataObjectBase
344 // ----------------------------------------------------------------------------
346 // VZ: I don't need this in MSW finally, so if it is needed in wxGTK, it should
347 // be moved to gtk/dataobj.cpp
350 wxString
wxFileDataObjectBase::GetFilenames() const
353 size_t count
= m_filenames
.GetCount();
354 for ( size_t n
= 0; n
< count
; n
++ )
356 str
<< m_filenames
[n
] << wxT('\0');
362 void wxFileDataObjectBase::SetFilenames(const wxChar
* filenames
)
367 for ( const wxChar
*pc
= filenames
; ; pc
++ )
377 // 2 consecutive NULs - this is the end of the string
381 m_filenames
.Add(current
);
389 // ----------------------------------------------------------------------------
390 // wxCustomDataObject
391 // ----------------------------------------------------------------------------
393 wxCustomDataObject::wxCustomDataObject(const wxDataFormat
& format
)
394 : wxDataObjectSimple(format
)
396 m_data
= (void *)NULL
;
399 wxCustomDataObject::~wxCustomDataObject()
404 void wxCustomDataObject::TakeData(size_t size
, void *data
)
412 void *wxCustomDataObject::Alloc(size_t size
)
414 return (void *)new char[size
];
417 void wxCustomDataObject::Free()
419 delete [] (char *)m_data
;
421 m_data
= (void *)NULL
;
424 size_t wxCustomDataObject::GetDataSize() const
429 bool wxCustomDataObject::GetDataHere(void *buf
) const
431 void *data
= GetData();
435 memcpy(buf
, data
, GetSize());
440 bool wxCustomDataObject::SetData(size_t size
, const void *buf
)
444 m_data
= Alloc(size
);
448 memcpy(m_data
, buf
, m_size
= size
);
453 // ============================================================================
454 // some common dnd related code
455 // ============================================================================
457 #if wxUSE_DRAG_AND_DROP
461 // ----------------------------------------------------------------------------
463 // ----------------------------------------------------------------------------
465 // NB: we can't use "new" in ctor initializer lists because this provokes an
466 // internal compiler error with VC++ 5.0 (hey, even gcc compiles this!),
467 // so use SetDataObject() instead
469 wxTextDropTarget::wxTextDropTarget()
471 SetDataObject(new wxTextDataObject
);
474 wxDragResult
wxTextDropTarget::OnData(wxCoord x
, wxCoord y
, wxDragResult def
)
479 wxTextDataObject
*dobj
= (wxTextDataObject
*)m_dataObject
;
480 return OnDropText(x
, y
, dobj
->GetText()) ? def
: wxDragNone
;
483 // ----------------------------------------------------------------------------
485 // ----------------------------------------------------------------------------
487 wxFileDropTarget::wxFileDropTarget()
489 SetDataObject(new wxFileDataObject
);
492 wxDragResult
wxFileDropTarget::OnData(wxCoord x
, wxCoord y
, wxDragResult def
)
497 wxFileDataObject
*dobj
= (wxFileDataObject
*)m_dataObject
;
498 return OnDropFiles(x
, y
, dobj
->GetFilenames()) ? def
: wxDragNone
;
501 #endif // wxUSE_DRAG_AND_DROP
503 #endif // wxUSE_DATAOBJ