
Introduction
I had posted an article for float control several months ago. Some body gave me a suggestion that I should add a checkbox in my control. I spade for an hour, and I found it is easy to do that. Here is my result. I got up today, "__Stephane Rodriguez__" (fakedwar@hotmail.com) send me a mail. He suggested to change the title of this article to make it more accurate. Thank you. Anthony_Yio send me a mail suggesting full selection of children and full un-checking of the children in the tree control as Office 2000 features. I have added the function to my tree control too.
Bug fixed
Stephane Rodriguez has mentioned a bug in the click and expand of the tree. It is fixed in this version.
Implementation of custom tree control
In the property page of tree control, you will there is a style named CheckBox. You shall implement your tree control from this point.
Firstly, you shall check CheckBox style in your resource where you want to use checkboxed tree control. Add a new bitmap resource in your project named IDB_STATE which includes 3 images: blank image, unchecked and checked box.
And then, we derive a new class from CTreeControl. Because this sample starts from my previous article, I used the same class as in the previous article. You shall first load the image for state using the following statement:
m_imageState.Create( IDB_STATE, 16, 1, RGB(255,255,255) );
SetImageList( &m_imageState, TVSIL_STATE );
The image will be used together with images for meanings. We shall respond to the Windows' message WM_LBUTTONDOWN to deal with check or uncheck of an item, as follows:
void CXMLTree::OnLButtonDown(UINT nFlags, CPoint point)
{
UINT uFlags=0;
HTREEITEM hti = HitTest(point,&uFlags);
if( uFlags & TVHT_ONITEMSTATEICON )
{
ToggleItemState(hti);
return;
}
CTreeCtrl::OnLButtonDown(nFlags, point);
}
In the standard Windows implementation of a control, you shall use space key to toggle the state of check box. So we need to respond to WM_KEYDOWN for space key.
if( nChar == VK_SPACE )
{
HTREEITEM hti = GetSelectedItem();
ToggleItemState(hti);
return;
}
CTreeCtrl::OnKeyDown(nChar, nRepCnt, nFlags);
We add a new function to support full selection of child items. The pros functions are changed to support it. Following is the function added:
void CXMLTree::ToggleItemState(HTREEITEM hti)
{
int iImage = GetItemState( hti, TVIS_STATEIMAGEMASK )>>12;
SetItemState( hti, INDEXTOSTATEIMAGEMASK(iImage == 1 ? 2 : 1),
TVIS_STATEIMAGEMASK );
if ( ItemHasChildren(hti))
{
HTREEITEM htiChild = this->GetChildItem (hti);
if (htiChild)
ToggleItemState(htiChild);
else
return ;
HTREEITEM htiSibling = GetNextSiblingItem (htiChild);
while (htiSibling )
{
ToggleItemState(htiSibling);
htiSibling = GetNextSiblingItem(htiSibling);
}
}
}
We will implement several helper functions to get or set the state of checkbox in the tree control attached to the tree items. If it is check box, it means multiple selection, we shall iterate all items checked. We also provide the function for iteration. But these functions have not been thoroughly tested.
BOOL CXMLTree::IsItemChecked(HTREEITEM hItem)
{
return GetItemState( hItem, TVIS_STATEIMAGEMASK )>>12 == 2;
}
HTREEITEM CXMLTree::GetFirstCheckedItem()
{
for ( HTREEITEM hItem = GetRootItem();
hItem!=NULL; hItem = GetNextItem( hItem ) )
if ( IsItemChecked(hItem) )
return hItem;
return NULL;
}
HTREEITEM CXMLTree::GetNextCheckedItem(HTREEITEM hItem)
{
for ( hItem = GetNextItem( hItem );
hItem!=NULL; hItem = GetNextItem( hItem ) )
if ( IsItemChecked(hItem) )
return hItem;
return NULL;
}
HTREEITEM CXMLTree::GetPrevCheckedItem(HTREEITEM hItem)
{
for ( hItem = GetPrevItem( hItem );
hItem!=NULL; hItem = GetPrevItem( hItem ) )
if ( IsItemChecked(hItem) )
return hItem;
return NULL;
}
void CXMLTree::SetChecked(HTREEITEM hItem)
{
SetItemState( hItem, INDEXTOSTATEIMAGEMASK(1),
TVIS_STATEIMAGEMASK );
}
HTREEITEM CXMLTree::GetNextItem(HTREEITEM hItem)
{
HTREEITEM hti;
if( ItemHasChildren( hItem ) )
return GetChildItem( hItem ); else{
while( (hti = GetNextSiblingItem( hItem )) == NULL ){
if( (hItem = GetParentItem( hItem ) ) == NULL )
return NULL;
}
}
return hti;
}
HTREEITEM CXMLTree::GetPrevItem(HTREEITEM hItem)
{
HTREEITEM hti;
if( hti = GetPrevSiblingItem(hItem) )
return hti;
else
return GetParentItem(hItem);
}
A new function is added to tree control. If you select the item, all of its children are selected. So it is done for uncheck actions. See: insertItem function.
Using this new control is very simple. Add a tree control in your dialog box and set the checkbox style. The other operations are same as other controls. Enjoy it.
Acknowledgement
Thanks for Anthony_Yio and Stephane Rodriguez. They gave me good suggestions and bug reports. Thanks for Prettybabe Prettybabe for raising the idea to write such a control.
History
- Oct 25th, 2002: version 1.0.
- Fixed bugs in
SetChecked. The state is error.
- Fixed bugs for mouse down to populate nodes.
- Added new functions for full selection and deselection.
- Added new function:
SetUnChecked
- Oct 24th, 2002: version 0.9. initial
I'm write program from 1990. My research field is CAG,CAD and Image processing. I select C/C++, ASP, Java, XML as my usaully developing tools. Occasional , write code in Delphi and VB. I'm using Visual C++ from 1996. If you have anything unclear, e-mail to :zhou_cn123@sina.com Software Engineering and CAD is my mainly research program.
You also can reach me on msn: zhoujohnson@hotmail.com