Fixed sorting and commiting support for native virtual wxListCtrl, although leaving it off for now since initial creation of the list items takes a fair amount of time when the item amount is significantly large (as in the listctrl demo).

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@41515 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Kevin Ollivier
2006-09-30 03:58:30 +00:00
parent 13f6c4701e
commit 67650ea0b9
3 changed files with 297 additions and 141 deletions

View File

@@ -986,8 +986,8 @@ public :
virtual wxClientDataType GetClientDataType() const; virtual wxClientDataType GetClientDataType() const;
virtual void SetClientDataType(wxClientDataType clientDataItemsType); virtual void SetClientDataType(wxClientDataType clientDataItemsType);
virtual ListSortOrder GetSortOrder() const; //virtual ListSortOrder GetSortOrder() const;
virtual void SetSortOrder(const ListSortOrder sort); //virtual void SetSortOrder(const ListSortOrder sort);

View File

@@ -139,12 +139,6 @@ public:
virtual int GetColumnImageValue( unsigned int column ); virtual int GetColumnImageValue( unsigned int column );
virtual void SetColumnImageValue( unsigned int column, int imageIndex ); virtual void SetColumnImageValue( unsigned int column, int imageIndex );
virtual OSStatus GetSetData( wxMacDataItemBrowserControl *owner ,
DataBrowserPropertyID property,
DataBrowserItemDataRef itemData,
bool changeValue );
virtual ~wxMacListCtrlItem(); virtual ~wxMacListCtrlItem();
protected: protected:
wxListItemList m_rowItems; wxListItemList m_rowItems;
@@ -167,7 +161,23 @@ public:
virtual void UpdateState(wxMacDataItem* dataItem, wxListItem* item); virtual void UpdateState(wxMacDataItem* dataItem, wxListItem* item);
protected: protected:
// we need to override to provide specialized handling for virtual wxListCtrls
virtual OSStatus GetSetItemData(DataBrowserItemID itemID,
DataBrowserPropertyID property,
DataBrowserItemDataRef itemData,
Boolean changeValue );
virtual void ItemNotification(
DataBrowserItemID itemID,
DataBrowserItemNotification message,
DataBrowserItemDataRef itemData);
virtual Boolean CompareItems(DataBrowserItemID itemOneID,
DataBrowserItemID itemTwoID,
DataBrowserPropertyID sortProperty);
wxClientDataType m_clientDataItemsType; wxClientDataType m_clientDataItemsType;
bool m_isVirtual;
}; };
@@ -260,7 +270,7 @@ bool wxListCtrl::Create(wxWindow *parent,
// Also, use generic list control in VIRTUAL mode. // Also, use generic list control in VIRTUAL mode.
if ( (wxSystemOptions::HasOption( wxMAC_ALWAYS_USE_GENERIC_LISTCTRL ) if ( (wxSystemOptions::HasOption( wxMAC_ALWAYS_USE_GENERIC_LISTCTRL )
&& (wxSystemOptions::GetOptionInt( wxMAC_ALWAYS_USE_GENERIC_LISTCTRL ) == 1)) || && (wxSystemOptions::GetOptionInt( wxMAC_ALWAYS_USE_GENERIC_LISTCTRL ) == 1)) ||
(style & wxLC_ICON) || (style & wxLC_SMALL_ICON) || (style & wxLC_LIST) || (style & wxLC_VIRTUAL)) (style & wxLC_ICON) || (style & wxLC_SMALL_ICON) || (style & wxLC_LIST) || (style & wxLC_VIRTUAL) )
{ {
m_macIsUserPane = true; m_macIsUserPane = true;
@@ -705,6 +715,7 @@ void wxListCtrl::SetItemTextColour( long item, const wxColour &col )
if (m_genericImpl) if (m_genericImpl)
{ {
m_genericImpl->SetItemTextColour(item, col); m_genericImpl->SetItemTextColour(item, col);
return;
} }
wxListItem info; wxListItem info;
@@ -1047,7 +1058,7 @@ bool wxListCtrl::EnsureVisible(long item)
m_dbImpl->RevealItem(dataItem, kDataBrowserRevealWithoutSelecting); m_dbImpl->RevealItem(dataItem, kDataBrowserRevealWithoutSelecting);
} }
return false; return true;
} }
// Find an item whose label matches this string, starting from the item after 'start' // Find an item whose label matches this string, starting from the item after 'start'
@@ -1186,7 +1197,6 @@ long wxListCtrl::InsertColumn(long col, wxListItem& item)
type = kDataBrowserIconAndTextType; type = kDataBrowserIconAndTextType;
} }
fprintf(stderr, "Flush is %d\n", item.GetAlign());
SInt16 just = teFlushDefault; SInt16 just = teFlushDefault;
if (item.GetMask() & wxLIST_MASK_FORMAT) if (item.GetMask() & wxLIST_MASK_FORMAT)
{ {
@@ -1305,6 +1315,22 @@ void wxListCtrl::SetItemCount(long count)
return; return;
} }
if (m_dbImpl)
{
// we need to temporarily disable the new item creation notification
// procedure to speed things up
// FIXME: Even this doesn't seem to help much...
DataBrowserCallbacks callbacks;
DataBrowserItemNotificationUPP itemUPP;
GetDataBrowserCallbacks(m_dbImpl->GetControlRef(), &callbacks);
itemUPP = callbacks.u.v1.itemNotificationCallback;
callbacks.u.v1.itemNotificationCallback = 0;
m_dbImpl->SetCallbacks(&callbacks);
::AddDataBrowserItems(m_dbImpl->GetControlRef(), kDataBrowserNoItem,
count, NULL, kDataBrowserItemNoProperty);
callbacks.u.v1.itemNotificationCallback = itemUPP;
m_dbImpl->SetCallbacks(&callbacks);
}
m_count = count; m_count = count;
} }
@@ -1434,6 +1460,10 @@ wxMacDataBrowserListCtrlControl::wxMacDataBrowserListCtrlControl( wxWindow *peer
{ {
OSStatus err = noErr; OSStatus err = noErr;
m_clientDataItemsType = wxClientData_None; m_clientDataItemsType = wxClientData_None;
m_isVirtual = false;
if ( style & wxLC_VIRTUAL )
m_isVirtual = true;
DataBrowserSelectionFlags options = kDataBrowserDragSelect; DataBrowserSelectionFlags options = kDataBrowserDragSelect;
if ( style & wxLC_SINGLE_SEL ) if ( style & wxLC_SINGLE_SEL )
@@ -1457,24 +1487,20 @@ wxMacDataBrowserListCtrlControl::wxMacDataBrowserListCtrlControl( wxWindow *peer
if ( style & wxLC_LIST || style & wxLC_NO_HEADER ) if ( style & wxLC_LIST || style & wxLC_NO_HEADER )
verify_noerr( SetHeaderButtonHeight( 0 ) ); verify_noerr( SetHeaderButtonHeight( 0 ) );
SetDataBrowserSortProperty( m_controlRef , kMinColumnId ); SetSortProperty( kMinColumnId );
if ( style & wxLC_SORT_ASCENDING ) if ( style & wxLC_SORT_ASCENDING )
{ {
m_sortOrder = SortOrder_Text_Ascending; m_sortOrder = SortOrder_Text_Ascending;
SetDataBrowserSortProperty( m_controlRef , kMinColumnId ); SetSortOrder( kDataBrowserOrderIncreasing );
SetDataBrowserSortOrder( m_controlRef , kDataBrowserOrderIncreasing);
} }
else if ( style & wxLC_SORT_DESCENDING ) else if ( style & wxLC_SORT_DESCENDING )
{ {
m_sortOrder = SortOrder_Text_Descending; m_sortOrder = SortOrder_Text_Descending;
SetDataBrowserSortProperty( m_controlRef , kMinColumnId ); SetSortOrder( kDataBrowserOrderDecreasing );
SetDataBrowserSortOrder( m_controlRef , kDataBrowserOrderDecreasing);
} }
else else
{ {
m_sortOrder = SortOrder_None; m_sortOrder = SortOrder_None;
SetDataBrowserSortProperty( m_controlRef , kMinColumnId);
SetDataBrowserSortOrder( m_controlRef , kDataBrowserOrderIncreasing);
} }
if ( style & wxLC_VRULES ) if ( style & wxLC_VRULES )
@@ -1484,11 +1510,244 @@ wxMacDataBrowserListCtrlControl::wxMacDataBrowserListCtrlControl( wxWindow *peer
#endif #endif
} }
verify_noerr( AutoSizeColumns() );
verify_noerr( SetHiliteStyle(kDataBrowserTableViewFillHilite ) ); verify_noerr( SetHiliteStyle(kDataBrowserTableViewFillHilite ) );
err = SetHasScrollBars( (style & wxHSCROLL) != 0 , true ); err = SetHasScrollBars( (style & wxHSCROLL) != 0 , true );
} }
OSStatus wxMacDataBrowserListCtrlControl::GetSetItemData(DataBrowserItemID itemID,
DataBrowserPropertyID property,
DataBrowserItemDataRef itemData,
Boolean changeValue )
{
wxString text;
int imgIndex = -1;
short listColumn = property - kMinColumnId;
OSStatus err = errDataBrowserPropertyNotSupported;
wxListCtrl* list = wxDynamicCast( GetPeer() , wxListCtrl );
if (listColumn >= 0)
{
if (!m_isVirtual)
{
wxMacListCtrlItem* lcItem = (wxMacListCtrlItem*) itemID;
if (lcItem->HasColumnInfo(listColumn)){
wxListItem* item = lcItem->GetColumnInfo(listColumn);
if (item->GetMask() & wxLIST_MASK_TEXT)
text = item->GetText();
if (item->GetMask() & wxLIST_MASK_IMAGE)
imgIndex = item->GetImage();
}
}
else
{
text = list->OnGetItemText( (long)itemID-1, listColumn );
imgIndex = list->OnGetItemColumnImage( (long)itemID-1, listColumn );
}
}
if ( !changeValue )
{
switch (property)
{
case kDataBrowserItemIsEditableProperty :
if ( list && list->HasFlag( wxLC_EDIT_LABELS ) )
{
verify_noerr(SetDataBrowserItemDataBooleanValue( itemData, true ));
err = noErr ;
}
break ;
default :
if ( property >= kMinColumnId )
{
wxMacCFStringHolder cfStr;
if (text){
cfStr.Assign( text, wxLocale::GetSystemEncoding() );
err = ::SetDataBrowserItemDataText( itemData, cfStr );
err = noErr;
}
if ( imgIndex != -1 )
{
wxImageList* imageList = list->GetImageList(wxIMAGE_LIST_SMALL);
if (imageList && imageList->GetImageCount() > 0){
wxBitmap bmp = imageList->GetBitmap(imgIndex);
IconRef icon = bmp.GetBitmapData()->GetIconRef();
::SetDataBrowserItemDataIcon(itemData, icon);
}
}
}
break ;
}
}
else
{
switch (property)
{
default:
if ( property >= kMinColumnId )
{
short listColumn = property - kMinColumnId;
// TODO probably send the 'end edit' from here, as we
// can then deal with the veto
CFStringRef sr ;
verify_noerr( GetDataBrowserItemDataText( itemData , &sr ) ) ;
wxMacCFStringHolder cfStr(sr) ;;
list->SetItem( (long)itemData , listColumn, cfStr.AsString() ) ;
err = noErr ;
}
break;
}
}
return err;
}
void wxMacDataBrowserListCtrlControl::ItemNotification(DataBrowserItemID itemID,
DataBrowserItemNotification message,
DataBrowserItemDataRef itemData )
{
// we want to depend on as little as possible to make sure tear-down of controls is safe
if ( message == kDataBrowserItemRemoved)
{
// make sure MacDelete does the proper teardown.
return;
}
else if ( message == kDataBrowserItemAdded )
{
// we don't issue events on adding, the item is not really stored in the list yet, so we
// avoid asserts by gettting out now
return ;
}
wxListCtrl *list = wxDynamicCast( GetPeer() , wxListCtrl );
if ( list )
{
bool trigger = false;
wxListEvent event( wxEVT_COMMAND_LIST_ITEM_SELECTED, list->GetId() );
bool isSingle = list->GetWindowStyle() | wxLC_SINGLE_SEL;
event.SetEventObject( list );
if ( !list->IsVirtual() )
{
wxMacDataItem* item = (wxMacDataItem*)itemID;
DataBrowserTableViewRowIndex result = 0;
verify_noerr( GetItemRow( itemID, &result ) ) ;
event.m_itemIndex = result;
if (event.m_itemIndex >= 0)
MacGetColumnInfo(event.m_itemIndex,0,event.m_item);
}
else
{
event.m_itemIndex = (long)itemID;
}
switch (message)
{
case kDataBrowserItemDeselected:
event.SetEventType(wxEVT_COMMAND_LIST_ITEM_DESELECTED);
if ( !isSingle )
trigger = IsSelectionSuppressed();
break;
case kDataBrowserItemSelected:
trigger = IsSelectionSuppressed();
break;
case kDataBrowserItemDoubleClicked:
event.SetEventType( wxEVT_LEFT_DCLICK );
trigger = true;
break;
case kDataBrowserEditStarted :
// TODO : how to veto ?
event.SetEventType( wxEVT_COMMAND_LIST_BEGIN_LABEL_EDIT ) ;
trigger = true ;
break ;
case kDataBrowserEditStopped :
// TODO probably trigger only upon the value store callback, because
// here IIRC we cannot veto
event.SetEventType( wxEVT_COMMAND_LIST_END_LABEL_EDIT ) ;
trigger = true ;
break ;
default:
break;
}
if ( trigger )
{
// direct notification is not always having the listbox GetSelection() having in synch with event
wxPostEvent( list->GetEventHandler(), event );
}
}
}
Boolean wxMacDataBrowserListCtrlControl::CompareItems(DataBrowserItemID itemOneID,
DataBrowserItemID itemTwoID,
DataBrowserPropertyID sortProperty)
{
bool retval = false;
wxString itemText;
wxString otherItemText;
int colId = sortProperty - kMinColumnId;
long itemNum = 0;
long otherItemNum = 0;
wxListCtrl* list = wxDynamicCast( GetPeer() , wxListCtrl );
// means we need to
if (colId >= 0)
{
if (!m_isVirtual)
{
wxMacListCtrlItem* item = (wxMacListCtrlItem*)itemOneID;
wxMacListCtrlItem* otherItem = (wxMacListCtrlItem*)itemTwoID;
itemNum = item->GetOrder();
otherItemNum = otherItem->GetOrder();
if (item->HasColumnInfo(colId))
itemText = item->GetColumnInfo(colId)->GetText();
if (otherItem->HasColumnInfo(colId))
otherItemText = otherItem->GetColumnInfo(colId)->GetText();
}
else
{
itemNum = (long)itemOneID;
otherItemNum = (long)itemTwoID;
itemText = list->OnGetItemText( itemNum-1, colId );
otherItemText = list->OnGetItemText( otherItemNum-1, colId );
}
DataBrowserSortOrder sort;
verify_noerr(GetSortOrder(&sort));
if ( sort == kDataBrowserOrderIncreasing )
{
retval = itemText.CmpNoCase( otherItemText ) > 0;
}
else if ( sort == kDataBrowserOrderDecreasing )
{
retval = itemText.CmpNoCase( otherItemText ) < 0;
}
}
else{
// fallback for undefined cases
retval = itemOneID < itemTwoID;
}
return retval;
}
wxMacDataBrowserListCtrlControl::~wxMacDataBrowserListCtrlControl() wxMacDataBrowserListCtrlControl::~wxMacDataBrowserListCtrlControl()
{ {
} }
@@ -1617,19 +1876,10 @@ void wxMacListCtrlItem::SetColumnInfo( unsigned int column, wxListItem* item )
if ( column >= m_rowItems.GetCount() ) if ( column >= m_rowItems.GetCount() )
{ {
wxListItem* listItem = new wxListItem(*item); wxListItem* listItem = new wxListItem(*item);
//listItem->SetAlign(item->GetAlign());
//listItem->SetBackgroundColour(item->GetBackgroundColour());
//listItem->SetColumn(item->GetColumn());
//listItem->SetData(item->GetData());
//listItem->SetFont(item->GetFont());
//listItem->SetId(GetOrder());
//listItem->SetImage(item->GetImage());
//listItem->SetMask(item->GetMask());
//listItem->SetText(item->GetText());
m_rowItems.Append( listItem ); m_rowItems.Append( listItem );
} }
else{ else
{
wxListItem* listItem = GetColumnInfo( column ); wxListItem* listItem = GetColumnInfo( column );
long mask = item->GetMask(); long mask = item->GetMask();
if (mask & wxLIST_MASK_TEXT) if (mask & wxLIST_MASK_TEXT)
@@ -1647,88 +1897,4 @@ void wxMacListCtrlItem::SetColumnInfo( unsigned int column, wxListItem* item )
} }
} }
OSStatus wxMacListCtrlItem::GetSetData( wxMacDataItemBrowserControl *owner ,
DataBrowserPropertyID property,
DataBrowserItemDataRef itemData,
bool changeValue )
{
OSStatus err = errDataBrowserPropertyNotSupported;
wxListCtrl* list = wxDynamicCast( owner->GetPeer() , wxListCtrl );
if ( !changeValue )
{
switch (property)
{
case kDataBrowserItemIsEditableProperty :
if ( list && list->HasFlag( wxLC_EDIT_LABELS ) )
{
verify_noerr(SetDataBrowserItemDataBooleanValue( itemData, true ));
err = noErr ;
}
break ;
default :
if ( property >= kMinColumnId )
{
short listColumn = property - kMinColumnId;
if (HasColumnInfo(listColumn))
{
wxListItem* item = GetColumnInfo(listColumn);
wxMacCFStringHolder cfStr;
if (item->GetText())
{
cfStr.Assign( item->GetText(), wxLocale::GetSystemEncoding() );
err = ::SetDataBrowserItemDataText( itemData, cfStr );
err = noErr;
}
int imgIndex = item->GetImage();
if ( (item->GetMask() & wxLIST_MASK_IMAGE) )
{
wxImageList* imageList = list->GetImageList(wxIMAGE_LIST_SMALL);
if (imageList && imageList->GetImageCount() > 0)
{
wxBitmap bmp = imageList->GetBitmap(imgIndex);
IconRef icon = bmp.GetBitmapData()->GetIconRef();
::SetDataBrowserItemDataIcon(itemData, icon);
}
}
}
}
break ;
}
}
else
{
switch (property)
{
default:
if ( property >= kMinColumnId )
{
short listColumn = property - kMinColumnId;
if (HasColumnInfo(listColumn))
{
// TODO probably send the 'end edit' from here, as we
// can then deal with the veto
CFStringRef sr ;
verify_noerr( GetDataBrowserItemDataText( itemData , &sr ) ) ;
wxMacCFStringHolder cfStr(sr) ;;
list->SetItem( owner->GetLineFromItem(this) , listColumn, cfStr.AsString() ) ;
err = noErr ;
}
}
break;
}
}
// don't duplicate the numeric order column handling
if (err == errDataBrowserPropertyNotSupported)
err = wxMacDataItem::GetSetData(owner, property, itemData, changeValue);
return err;
}
#endif // wxUSE_LISTCTRL #endif // wxUSE_LISTCTRL

View File

@@ -1646,11 +1646,11 @@ bool wxMacDataItem::IsLessThan(wxMacDataItemBrowserControl *owner ,
const wxMacDataItem* otherItem = dynamic_cast<const wxMacDataItem*>(rhs); const wxMacDataItem* otherItem = dynamic_cast<const wxMacDataItem*>(rhs);
bool retval = false; bool retval = false;
if ( sortProperty == m_colId && owner->GetSortOrder() != SortOrder_None){ if ( sortProperty == m_colId ){
retval = m_label.CmpNoCase( otherItem->m_label) < 0; retval = m_label.CmpNoCase( otherItem->m_label) < 0;
} }
else if ( owner->GetSortOrder() == SortOrder_None || sortProperty == kNumericOrderColumnId ) else if ( sortProperty == kNumericOrderColumnId )
retval = m_order < otherItem->m_order; retval = m_order < otherItem->m_order;
return retval; return retval;
@@ -1703,16 +1703,6 @@ wxMacDataItemBrowserControl::wxMacDataItemBrowserControl( wxWindow* peer , const
m_clientDataItemsType = wxClientData_None; m_clientDataItemsType = wxClientData_None;
} }
ListSortOrder wxMacDataItemBrowserControl::GetSortOrder() const
{
return m_sortOrder;
}
void wxMacDataItemBrowserControl::SetSortOrder(const ListSortOrder sort)
{
m_sortOrder = sort;
}
wxMacDataItem* wxMacDataItemBrowserControl::CreateItem() wxMacDataItem* wxMacDataItemBrowserControl::CreateItem()
{ {
return new wxMacDataItem(); return new wxMacDataItem();
@@ -1910,9 +1900,9 @@ void wxMacDataItemBrowserControl::InsertColumn(int colId, DataBrowserPropertyTyp
columnDesc.propertyDesc.propertyID = (kMinColumnId + colId); columnDesc.propertyDesc.propertyID = (kMinColumnId + colId);
columnDesc.propertyDesc.propertyType = colType; columnDesc.propertyDesc.propertyType = colType;
columnDesc.propertyDesc.propertyFlags = kDataBrowserListViewDefaultColumnFlags | kDataBrowserListViewTypeSelectColumn;; columnDesc.propertyDesc.propertyFlags = kDataBrowserListViewDefaultColumnFlags | kDataBrowserListViewTypeSelectColumn;
#if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4 #if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4
columnDesc.propertyDesc.propertyFlags = kDataBrowserListViewNoGapForIconInHeaderButton; columnDesc.propertyDesc.propertyFlags |= kDataBrowserListViewNoGapForIconInHeaderButton;
#endif #endif
verify_noerr( AddColumn( &columnDesc, kDataBrowserListViewAppendColumn ) ); verify_noerr( AddColumn( &columnDesc, kDataBrowserListViewAppendColumn ) );