首先,我要感谢所有花时间查看此主题并尝试提供帮助的人。
我在Internet上搜索过,找不到选择标签文本与字符串变量文本相同的树视图节点的示例。
在MSDN上我找到了消息TVM_GETISEARCHSTRING,但我不知道它是否可以用来解决我的问题。即使可以,我仍然不知道如何使用它。
我有一个字符串变量,用于保存数据库中的文本。 程序加载时,树视图应该有一个选择了相同文本的节点。
请帮忙处理一些说明或代码段,因为我不知道如何开始编码。
我使用纯WIN32 API在Windows XP上使用纯WIN32 API在MS Visual Studio Express 2008中工作。
这就是全部,再次感谢所有试图帮助的人。非常感谢你们!
这两个答案对我都有好处,但我不知道如何标记它们,似乎在这个网站上只能接受一个答案。
我不可能只是忽略了他们所投入的所有工作来帮助我,所以我写这个是为了试图回报Jonathan,至少通过正式声明他的解决方案对我来说也是可以接受的,它是只是蒂姆的解决方案更适合我的编码风格。我也会提出两个答案。
答案 0 :(得分:6)
不幸的是,没有记录的方法来按项目标签搜索树视图。
TVM_GETISEARCHSTRING
消息返回用户在树中键入的搜索字符串(增量搜索模式),但它不会触发搜索或让您提供自己的搜索字符串。
唯一的方法是手动迭代树节点并自己比较标签。下面是一个示例函数,请注意它是递归的,并且每个子级别将使用大约半KB的堆栈。
HTREEITEM TreeView_FindLabel(HWND hWnd, HTREEITEM hItemParent, LPCWSTR pszLabel)
{
TVITEM tvi;
wchar_t wchLabel[256];
for (tvi.hItem = TreeView_GetChild(hWnd, hItemParent); tvi.hItem;
tvi.hItem = TreeView_GetNextSibling(hWnd, tvi.hItem))
{
tvi.mask = TVIF_TEXT | TVIF_CHILDREN;
tvi.pszText = wchLabel;
tvi.cchTextMax = _countof(wchLabel);
if (TreeView_GetItem(hWnd, &tvi))
{
if (_wcsicmp(tvi.pszText, pszLabel) == 0)
return tvi.hItem;
if (tvi.cChildren)
{
HTREEITEM hChildSearch = TreeView_FindLabel(hWnd, tvi.hItem, pszLabel);
if (hChildSearch) return hChildSearch;
}
}
}
return 0;
}
这不是一种特别快速的搜索树的方法。如果您需要进行大量搜索,最好使用std::map
自行跟踪标签和树项。例如,
std::map<std::wstring, HTREEITEM> mapTreeItems;
// whenever you add an item
HTREEITEM hItem = ListView_InsertItem(...);
mapTreeItems[strLabel] = hItem;
然后,您可以使用地图按标签查找树项目。您只需记住在从树中删除项目时更新地图并擦除标签。
答案 1 :(得分:6)
treeview控件不提供用于搜索标签的API。您必须手动遍历项目并将它们与您的字符串进行比较。
如果您的树视图深度超过一级,则必须决定如何遍历项目(首先是深度,要么是宽度)。如果有多个项目具有相同的标签,这些策略可能会返回不同的项目。
实现可能如下所示:
// Helper function to return the label of a treeview item
std::wstring GetItemText( HWND hwndTV, HTREEITEM htItem )
{
static const size_t maxLen = 128;
WCHAR buffer[ maxLen + 1 ];
TVITEMW tvi = { 0 };
tvi.hItem = htItem; // Treeview item to query
tvi.mask = TVIF_TEXT; // Request text only
tvi.cchTextMax = maxLen;
tvi.pszText = &buffer[ 0 ];
if ( TreeView_GetItem( hwndTV, &tvi ) )
{
return std::wstring( tvi.pszText );
}
else
{
return std::wstring();
}
}
这是实际遍历发生的地方。递归调用该函数,直到无法搜索任何项目或找到匹配项。此实现使用区分大小写的比较(wstring::operator==( const wstring& )
)。如果您需要不同的谓词,则必须根据需要修改实现。
HTREEITEM FindItemDepthFirstImpl( HWND hwndTV, HTREEITEM htStart, const std::wstring& itemText )
{
HTREEITEM htItemMatch = NULL;
HTREEITEM htItemCurrent = htStart;
// Iterate over items until there are no more items or we found a match
while ( htItemCurrent != NULL && htItemMatch == NULL )
{
if ( GetItemText( hwndTV, htItemCurrent ) == itemText )
{
htItemMatch = htItemCurrent;
}
else
{
// Traverse into child items
htItemMatch = FindItemDepthFirstImpl( hwndTV, TreeView_GetChild( hwndTV, htItemCurrent ), itemText );
}
htItemCurrent = TreeView_GetNextSibling( hwndTV, htItemCurrent );
}
return htItemMatch;
}
以下函数包装递归并将根元素作为起始点传递。这是您在代码中调用的函数。如果找到,则返回HTREEITEM
,否则返回NULL
。
HTREEITEM FindItem( HWND hwndTV, const std::wstring& itemText )
{
HTREEITEM htiRoot = TreeView_GetRoot( hwndTV );
return FindItemDepthFirstImpl( hwndTV, htiRoot, itemText );
}