fix access to client data of wxComboBox in wxUniv by virtualizing access to wxControlWithItems::m_clientDataItemsType

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@48728 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Vadim Zeitlin
2007-09-16 23:12:27 +00:00
parent c4fd5ede54
commit 131b1fba9d
5 changed files with 67 additions and 29 deletions

View File

@@ -121,7 +121,7 @@ private:
int AppendItems(const wxArrayStringsAdapter& items, void **clientData) int AppendItems(const wxArrayStringsAdapter& items, void **clientData)
{ {
wxASSERT_MSG( m_clientDataItemsType != wxClientData_Object, wxASSERT_MSG( GetClientDataType() != wxClientData_Object,
_T("can't mix different types of client data") ); _T("can't mix different types of client data") );
return AppendItems(items, clientData, wxClientData_Void); return AppendItems(items, clientData, wxClientData_Void);
@@ -130,7 +130,7 @@ private:
int AppendItems(const wxArrayStringsAdapter& items, int AppendItems(const wxArrayStringsAdapter& items,
wxClientData **clientData) wxClientData **clientData)
{ {
wxASSERT_MSG( m_clientDataItemsType != wxClientData_Void, wxASSERT_MSG( GetClientDataType() != wxClientData_Void,
_T("can't mix different types of client data") ); _T("can't mix different types of client data") );
return AppendItems(items, wx_reinterpret_cast(void **, clientData), return AppendItems(items, wx_reinterpret_cast(void **, clientData),
@@ -166,7 +166,7 @@ private:
unsigned int pos, unsigned int pos,
void **clientData) void **clientData)
{ {
wxASSERT_MSG( m_clientDataItemsType != wxClientData_Object, wxASSERT_MSG( GetClientDataType() != wxClientData_Object,
_T("can't mix different types of client data") ); _T("can't mix different types of client data") );
return InsertItems(items, pos, clientData, wxClientData_Void); return InsertItems(items, pos, clientData, wxClientData_Void);
@@ -176,7 +176,7 @@ private:
unsigned int pos, unsigned int pos,
wxClientData **clientData) wxClientData **clientData)
{ {
wxASSERT_MSG( m_clientDataItemsType != wxClientData_Void, wxASSERT_MSG( GetClientDataType() != wxClientData_Void,
_T("can't mix different types of client data") ); _T("can't mix different types of client data") );
return InsertItems(items, pos, return InsertItems(items, pos,
@@ -280,6 +280,16 @@ public:
void Delete(unsigned int pos); void Delete(unsigned int pos);
// various accessors
// -----------------
// The control may maintain its items in a sorted order in which case
// items are automatically inserted at the right position when they are
// inserted or appended. Derived classes have to override this method if
// they implement sorting, typically by returning HasFlag(wxXX_SORT)
virtual bool IsSorted() const { return false; }
// client data stuff // client data stuff
// ----------------- // -----------------
@@ -289,19 +299,23 @@ public:
void SetClientObject(unsigned int n, wxClientData* clientData); void SetClientObject(unsigned int n, wxClientData* clientData);
wxClientData* GetClientObject(unsigned int n) const; wxClientData* GetClientObject(unsigned int n) const;
// return the type of client data stored in this control: usually it just
// returns m_clientDataItemsType but must be overridden in the controls
// which delegate their client data storage to another one (e.g. wxChoice
// in wxUniv which stores data in wxListBox which it uses anyhow); don't
// forget to override SetClientDataType() if you override this one
//
// NB: for this to work no code should ever access m_clientDataItemsType
// directly but only via this function!
virtual wxClientDataType GetClientDataType() const
{ return m_clientDataItemsType; }
bool HasClientData() const bool HasClientData() const
{ return m_clientDataItemsType != wxClientData_None; } { return GetClientDataType() != wxClientData_None; }
bool HasClientObjectData() const bool HasClientObjectData() const
{ return m_clientDataItemsType == wxClientData_Object; } { return GetClientDataType() == wxClientData_Object; }
bool HasClientUntypedData() const bool HasClientUntypedData() const
{ return m_clientDataItemsType == wxClientData_Void; } { return GetClientDataType() == wxClientData_Void; }
// The control may maintain its items in a sorted order in which case
// items are automatically inserted at the right position when they are
// inserted or appended. Derived classes have to override this method if
// they implement sorting, typically by returning HasFlag(wxXX_SORT)
virtual bool IsSorted() const { return false; }
protected: protected:
// there is usually no need to override this method but you can do it if it // there is usually no need to override this method but you can do it if it
@@ -364,7 +378,14 @@ protected:
// set it to NULL (must only be called if HasClientObjectData()) // set it to NULL (must only be called if HasClientObjectData())
void ResetItemClientObject(unsigned int n); void ResetItemClientObject(unsigned int n);
// set the type of the client data stored in this control: override this if
// you override GetClientDataType()
virtual void SetClientDataType(wxClientDataType clientDataItemsType)
{
m_clientDataItemsType = clientDataItemsType;
}
private:
// the type of the client data for the items // the type of the client data for the items
wxClientDataType m_clientDataItemsType; wxClientDataType m_clientDataItemsType;
}; };

View File

@@ -142,6 +142,15 @@ public:
return GetStdInputHandler(handlerDef); return GetStdInputHandler(handlerDef);
} }
// we delegate our client data handling to wxListBox which we use for the
// items, so override this and other methods dealing with the client data
virtual wxClientDataType GetClientDataType() const
{
return GetLBox()->GetClientDataType();
}
virtual void SetClientDataType(wxClientDataType clientDataItemsType);
protected: protected:
virtual int DoInsertItems(const wxArrayStringsAdapter& items, virtual int DoInsertItems(const wxArrayStringsAdapter& items,
unsigned int pos, unsigned int pos,
@@ -150,6 +159,7 @@ protected:
virtual void DoSetItemClientData(unsigned int n, void* clientData); virtual void DoSetItemClientData(unsigned int n, void* clientData);
virtual void* DoGetItemClientData(unsigned int n) const; virtual void* DoGetItemClientData(unsigned int n) const;
// common part of all ctors // common part of all ctors
void Init(); void Init();

View File

@@ -102,7 +102,7 @@ void wxItemContainer::Clear()
ResetItemClientObject(i); ResetItemClientObject(i);
} }
m_clientDataItemsType = wxClientData_None; SetClientDataType(wxClientData_None);
DoClear(); DoClear();
} }
@@ -118,7 +118,7 @@ void wxItemContainer::Delete(unsigned int pos)
if ( IsEmpty() ) if ( IsEmpty() )
{ {
m_clientDataItemsType = wxClientData_None; SetClientDataType(wxClientData_None);
} }
} }
@@ -162,23 +162,22 @@ wxItemContainer::DoInsertOneItem(const wxString& WXUNUSED(item),
void wxItemContainer::SetClientObject(unsigned int n, wxClientData *data) void wxItemContainer::SetClientObject(unsigned int n, wxClientData *data)
{ {
wxASSERT_MSG( m_clientDataItemsType == wxClientData_Object || wxASSERT_MSG( !HasClientUntypedData(),
m_clientDataItemsType == wxClientData_None,
wxT("can't have both object and void client data") ); wxT("can't have both object and void client data") );
if ( m_clientDataItemsType == wxClientData_Object ) if ( HasClientObjectData() )
{ {
wxClientData * clientDataOld wxClientData * clientDataOld
= wx_static_cast(wxClientData *, DoGetItemClientData(n)); = wx_static_cast(wxClientData *, DoGetItemClientData(n));
if ( clientDataOld ) if ( clientDataOld )
delete clientDataOld; delete clientDataOld;
} }
else // m_clientDataItemsType == wxClientData_None else // didn't have any client data so far
{ {
// now we have object client data // now we have object client data
DoInitItemClientData(); DoInitItemClientData();
m_clientDataItemsType = wxClientData_Object; SetClientDataType(wxClientData_Object);
} }
DoSetItemClientData(n, data); DoSetItemClientData(n, data);
@@ -186,7 +185,7 @@ void wxItemContainer::SetClientObject(unsigned int n, wxClientData *data)
wxClientData *wxItemContainer::GetClientObject(unsigned int n) const wxClientData *wxItemContainer::GetClientObject(unsigned int n) const
{ {
wxCHECK_MSG( m_clientDataItemsType == wxClientData_Object, NULL, wxCHECK_MSG( HasClientObjectData(), NULL,
wxT("this window doesn't have object client data") ); wxT("this window doesn't have object client data") );
return wx_static_cast(wxClientData *, DoGetItemClientData(n)); return wx_static_cast(wxClientData *, DoGetItemClientData(n));
@@ -194,13 +193,13 @@ wxClientData *wxItemContainer::GetClientObject(unsigned int n) const
void wxItemContainer::SetClientData(unsigned int n, void *data) void wxItemContainer::SetClientData(unsigned int n, void *data)
{ {
if ( m_clientDataItemsType == wxClientData_None ) if ( !HasClientData() )
{ {
DoInitItemClientData(); DoInitItemClientData();
m_clientDataItemsType = wxClientData_Void; SetClientDataType(wxClientData_Void);
} }
wxASSERT_MSG( m_clientDataItemsType == wxClientData_Void, wxASSERT_MSG( HasClientUntypedData(),
wxT("can't have both object and void client data") ); wxT("can't have both object and void client data") );
DoSetItemClientData(n, data); DoSetItemClientData(n, data);
@@ -208,7 +207,7 @@ void wxItemContainer::SetClientData(unsigned int n, void *data)
void *wxItemContainer::GetClientData(unsigned int n) const void *wxItemContainer::GetClientData(unsigned int n) const
{ {
wxCHECK_MSG( m_clientDataItemsType == wxClientData_Void, NULL, wxCHECK_MSG( HasClientUntypedData(), NULL,
wxT("this window doesn't have void client data") ); wxT("this window doesn't have void client data") );
return DoGetItemClientData(n); return DoGetItemClientData(n);

View File

@@ -1028,7 +1028,8 @@ void wxOwnerDrawnComboBox::DoSetItemClientData(unsigned int n, void* clientData)
{ {
EnsurePopupControl(); EnsurePopupControl();
GetVListBoxComboPopup()->SetItemClientData(n,clientData,m_clientDataItemsType); GetVListBoxComboPopup()->SetItemClientData(n, clientData,
GetClientDataType());
} }
void* wxOwnerDrawnComboBox::DoGetItemClientData(unsigned int n) const void* wxOwnerDrawnComboBox::DoGetItemClientData(unsigned int n) const

View File

@@ -92,6 +92,8 @@ protected:
void OnLeftUp(wxMouseEvent& event); void OnLeftUp(wxMouseEvent& event);
private: private:
friend class wxComboBox; // it accesses our DoGetItemClientData()
DECLARE_EVENT_TABLE() DECLARE_EVENT_TABLE()
}; };
@@ -410,6 +412,11 @@ int wxComboBox::GetSelection() const
#endif #endif
} }
void wxComboBox::SetClientDataType(wxClientDataType clientDataItemsType)
{
GetLBox()->SetClientDataType(clientDataItemsType);
}
int wxComboBox::DoInsertItems(const wxArrayStringsAdapter & items, int wxComboBox::DoInsertItems(const wxArrayStringsAdapter & items,
unsigned int pos, unsigned int pos,
void **clientData, wxClientDataType type) void **clientData, wxClientDataType type)
@@ -419,12 +426,12 @@ int wxComboBox::DoInsertItems(const wxArrayStringsAdapter & items,
void wxComboBox::DoSetItemClientData(unsigned int n, void* clientData) void wxComboBox::DoSetItemClientData(unsigned int n, void* clientData)
{ {
GetLBox()->SetClientData(n, clientData); GetLBox()->DoSetItemClientData(n, clientData);
} }
void *wxComboBox::DoGetItemClientData(unsigned int n) const void *wxComboBox::DoGetItemClientData(unsigned int n) const
{ {
return GetLBox()->GetClientData(n); return GetLBox()->DoGetItemClientData(n);
} }
bool wxComboBox::IsEditable() const bool wxComboBox::IsEditable() const