Win32 SDK 下的TreeCtrl控件的封装
自己照着MFC封装,半月前封装了一半,碰到问题就放下了,今天终于封装好了.
[Ctree.h]文件
#pragma once
#include <windows.h>
#include <commctrl.h> struct MTVINSERTSTRUCT
{
TVINSERTSTRUCT tvInsert;
TCHAR strMid[2500];
}; class CTree
{
public:
CTree();
~CTree(); HWND m_hWnd;
public:
//获得树控件的句柄
BOOL GetHwnd(HWND _pHwnd, UINT _idd); //插入树节点
HTREEITEM InsertItem(LPTVINSERTSTRUCT lpInsertStruct);
HTREEITEM InsertItem(LPCTSTR lpszItem, int nImage, int nSelectedImage,
HTREEITEM hParent, HTREEITEM hInsertAfter); //获得选中节点的句柄
HTREEITEM GetSelectedItem()const;
//删除选中节点
BOOL DeleteItem(HTREEITEM hItem);
//删除所有的节点项
BOOL DeleteAllItem();
//设置节点附加的32位数据
BOOL SetItemData(HTREEITEM hItem, DWORD dwData);
//设置节点项数据
BOOL SetItem(TVITEM *pItem);
BOOL SetItem(HTREEITEM hItem, UINT nMask, LPCTSTR lpszItem, int nImage, int nSelectedImage,
UINT nState, UINT nStateMask, LPARAM lParam);
//获得选中节点项附加的32位数据
DWORD GetItemData(HTREEITEM hItem)const;
//获得选中节点项的显示的名称
TCHAR* GetItemText(HTREEITEM hItem)const;
//获得选中节点项的图标索引值
BOOL GetItemImage(HTREEITEM hItem, int &nImage, int &nSelectedImage)const;
//获得选中节点项的展开状态
BOOL GetItemState(HTREEITEM hItem, UINT nStateMask) const;
};
[Ctree.cpp]文件
#include "CTree.h"
#include <tchar.h>
#include <atlchecked.h> #pragma region MY_CtreeCtrl_Class CTree::CTree()
{
m_hWnd = NULL; } CTree::~CTree()
{ } BOOL CTree::GetHwnd(HWND _pHwnd, UINT _idd)
{
m_hWnd = GetDlgItem(_pHwnd, _idd);
if (m_hWnd)
{
return TRUE;
}
return FALSE;
} HTREEITEM CTree::InsertItem(LPTVINSERTSTRUCT lpInsertStruct)
{
IsWindow(m_hWnd);
return (HTREEITEM)::SendMessage(m_hWnd, TVM_INSERTITEM, 0, (LPARAM)lpInsertStruct);
} HTREEITEM CTree::InsertItem(LPCTSTR lpszItem, int nImage, int nSelectedImage, HTREEITEM hParent, HTREEITEM hInsertAfter)
{
IsWindow(m_hWnd);
TVINSERTSTRUCT lpInsert;
lpInsert.item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE;
lpInsert.item.iImage = nImage;
lpInsert.item.iSelectedImage = nSelectedImage;
lpInsert.item.pszText = (LPWSTR)lpszItem; if (NULL == hParent)
hParent = TVI_ROOT;
lpInsert.hParent = hParent; lpInsert.hInsertAfter = hInsertAfter; return InsertItem(&lpInsert);
} HTREEITEM CTree::GetSelectedItem() const
{
IsWindow(m_hWnd);
return (HTREEITEM)::SendMessage(m_hWnd, TVM_GETNEXTITEM, TVGN_CARET, 0);
} BOOL CTree::DeleteItem(HTREEITEM hItem)
{
IsWindow(m_hWnd);
return (BOOL)::SendMessage(m_hWnd, TVM_DELETEITEM, 0, (LPARAM)hItem);
} BOOL CTree::DeleteAllItem()
{
IsWindow(m_hWnd);
return (BOOL)::SendMessage(m_hWnd, TVM_DELETEITEM, 0, (LPARAM)TVI_ROOT);
}
//32位系统可用
BOOL CTree::SetItemData(HTREEITEM hItem, DWORD dwData)
{
IsWindow(m_hWnd); return SetItem(hItem, TVIF_PARAM, NULL, 0, 0, 0, 0, (LPARAM)dwData);
}
//64位系统上要用LPARAM 用DWORD要被截断
//***********************************************************************************
BOOL AfxCTreeCtrl::SetItemData(HTREEITEM hItem, LPARAM dwData)
{
ASSERT(IsWindow(m_hWnd));
return SetItem(hItem, TVIF_PARAM, NULL, 0, 0, 0, 0, (LPARAM)dwData);
}
//***********************************************************************************
BOOL CTree::SetItem(TVITEM * pItem)
{
IsWindow(m_hWnd);
return (BOOL)SendMessage(m_hWnd, TVM_SETITEM, 0, (LPARAM)pItem);
} BOOL CTree::SetItem(HTREEITEM hItem, UINT nMask, LPCTSTR lpszItem, int nImage, int nSelectedImage, UINT nState, UINT nStateMask, LPARAM lParam)
{
IsWindow(m_hWnd);
TVITEM item;
item.hItem = hItem;
item.mask = nMask;
item.pszText = (LPTSTR)lpszItem;
item.iImage = nImage;
item.iSelectedImage = nSelectedImage;
item.state = nState;
item.stateMask = nStateMask;
item.lParam = lParam;
return (BOOL)::SendMessage(m_hWnd, TVM_SETITEM, 0, (LPARAM)&item);
}
//32位系统可用
DWORD CTree::GetItemData(HTREEITEM hItem) const
{
IsWindow(m_hWnd);
if (NULL == hItem)
return 0;
TVITEM item;
item.hItem = hItem;
item.mask = TVIF_PARAM;
::SendMessage(m_hWnd, TVM_GETITEM, 0, (LPARAM)&item);
return item.lParam; }
//64位系统用
LPARAM AfxCTreeCtrl::GetItemData(HTREEITEM hItem) const
{
ASSERT(IsWindow(m_hWnd));
TVITEM item;
item.hItem = hItem;
item.mask = TVIF_PARAM;
::SendMessage(m_hWnd, TVM_GETITEM, 0, (LPARAM)&item);
return (LPARAM)item.lParam;
}
TCHAR* CTree::GetItemText(HTREEITEM hItem) const
{
IsWindow(m_hWnd);
TCHAR* str = new TCHAR[128];
TVITEM item;
item.hItem = hItem;
item.mask = TVIF_TEXT;
item.pszText = str;
item.cchTextMax = 128; ::SendMessage(m_hWnd, TVM_GETITEM, 0, (LPARAM)&item); return str;
} BOOL CTree::GetItemImage(HTREEITEM hItem, int &nImage, int &nSelectedImage) const
{
IsWindow(m_hWnd);
TVITEM item = { 0 };
item.hItem = hItem;
item.mask = TVIF_IMAGE | TVIF_SELECTEDIMAGE;
BOOL bRes = (BOOL)::SendMessage(m_hWnd, TVM_GETITEM, 0, (LPARAM)&item);
if (bRes)
{
nImage = item.iImage;
nSelectedImage = item.iSelectedImage;
}
return bRes;
} BOOL CTree::GetItemState(HTREEITEM hItem, UINT nStateMask) const
{
IsWindow(m_hWnd);
TVITEM item = { 0 };
item.hItem = hItem;
item.mask = TVIF_STATE;
item.stateMask = nStateMask;
item.state = 0;
::SendMessage(m_hWnd, TVM_GETITEM, 0, (LPARAM)&item);
return item.state;
} #pragma endregion MY_CtreeCtrl_Class
[test.cpp]中测试
HTREEITEM hItem = m_tree.GetSelectedItem();
//TVITEM tvi;
//TCHAR szName[20] = { 0 };
////LPARAM lp = 0;
//tvi.mask = TVIF_TEXT | TVIF_PARAM;
//tvi.hItem = hItem;
//tvi.pszText = szName;
//tvi.cchTextMax = sizeof(szName);
////tvi.lParam = lp;
//TreeView_GetItem(m_tree.m_hWnd, &tvi);
//Stu *pstu = (Stu*)tvi.lParam; //获得选中节点的附加(由SetItemData设置的)数据
/*Stu* pstu = (Stu*)m_tree.GetItemData(hItem);
TCHAR *pch = (TCHAR*)m_tree.GetItemData(hItem);*/
//上次在封装这附加数据的时候碰到了问题.
在设置的时候:
TCHAR *pch = new TCHAR;
pch=mid;//mid为要设置的附加数据,类型为TCHAR
m_tree.SetItemData(nItem, (DWORD)pch);
就一直取不出来要的数据,就放下了.今天来看的时候才发现
TCHAR *pch = new TCHAR[128];
_tcscpy_s(pch, 128, mid);
m_tree.SetItemData(nItem, (DWORD)pch);
我用结构体指针为附加数据时,又能正确取出数据.就有点晕菜了,最后才发现.
TCHAR new出来时没给定大小.
正确代码为:
TCHAR *pch = new TCHAR[128];
_tcscpy_s(pch, 128, mid);
m_tree.SetItemData(nItem, (DWORD)pch);
//获得选中节点的Text
/*TCHAR *pch = m_tree.GetItemText(hItem);
MessageBox(NULL, pch, TEXT("123"), MB_OK);*/ //获得选中节点的图标的索引
/*int nImate, nSelectedImage;
m_tree.GetItemImage(hItem, nImate, nSelectedImage);*/ //获得选中节点是否展开,返回0为未展开,非0为展开状态 //UINT n = m_tree.GetItemState(hItem, TVIS_EXPANDED); BOOL m = m_tree.GetItemState(hItem, TVIS_EXPANDED) & TVIS_EXPANDED;
释放节点附加数据内存
//5.附加内存的释放
//在子控件通知消息中捕获TVN_DELETEITEM消息
NMHDR *pNMHDR = (NMHDR*)lParam;
if (pNMHDR->idFrom == IDC_TREE1 && pNMHDR->code==TVN_DELETEITEM)
{
NMTREEVIEW *pnmTV = (NMTREEVIEW*)lParam;
HTREEITEM hSelItem = pnmTV-> itemNew.hItem; //从lParam参数中获得选中项的句柄
//释放节点附加数据内存
TVITEM item = pnmTV->itemOld;
if (item.lParam != 0)
{
delete (TCHAR*)item.lParam;//删除的数据类型要和前面添加时你附加的数据类型一样,
} }
Win32 SDK 下的TreeCtrl控件的封装的更多相关文章
- Win32 SDK程序创建一些控件(简单调用InitCommonControlsEx,并指定ICC_LISTVIEW_CLASSES控件就可以了)
在Win32 SDK中创建一些控件的时候需要注意一下(具体是哪些控件请参看MSDN文档中列出来的) /* MSDN:Carries information used to load common co ...
- 9.2.2 .net framework下的MVC 控件的封装(下)
控件封装的部分说明 可能有人觉得应该前后端分离,我也承认这是应该的方向,我们也在考虑使用ng2等简化前端.但是,我们封装控件还是因为如下原因综合考虑的: 我们这是个框架,上面支撑了许多个应用,包含几百 ...
- 9.2.1 .net framework下的MVC 控件的封装(上)
在写.net core下mvc控件的编写之前,我先说一下.net framework下我们MVC控件的做法. MVC下控件的写法,主要有如下三种,最后一种是泛型的写法,mvc提供的控件都是基本控件. ...
- win32 TreeCtrl控件通知消息, LVN_SELCHANGED和LVN_ITEMCHANGED用法
今天出了个奇怪的问题,当我在主窗口上创建一个用模板对话框的子窗口时, 在子窗口上放的TreeCtrl控件不响应LVN_SELCHANGED消息,也是晕死了, 我以为是消息捕获的问题,我在主窗口上也捕获 ...
- Google SwipeRefreshLayout(Goolge官方下拉刷新控件)尝鲜
前天Google官方终于出了Android刷新控件——SwipeRefreshLayout. 使用前先需要将android.support.v4.jar升级到19.1.升级后,可能会出现SDK版本与A ...
- Android——谷歌官方下拉刷新控件SwipeRefreshLayout(转)
转自:http://blog.csdn.net/zouzhigang96/article/details/50476402 版权声明:本文为博主原创文章,未经博主允许不得转载. 前言: 如今谷歌推出了 ...
- android官方下拉刷新控件SwipeRefreshLayout的使用
可能开发安卓的人大多数都用过很多下拉刷新的开源组件,但是今天用了官方v4支持包的SwipeRefreshLayout觉得效果也蛮不错的,特拿出来分享. 简介:SwipeRefreshLayout组件只 ...
- 分享一个 C# Winfrom 下的 OutlookBar 控件的使用
最近在上网的时候,发现了这个C# 下的 OutlookBar 控件,看了一下感觉还真不错,特此记录一下. using System; using System.Drawing; using Syste ...
- [Android]下拉刷新控件RefreshableView的实现
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/4172483.html 需求:自定义一个ViewGroup,实现 ...
- android SwipeRefreshLayout google官方下拉刷新控件
下拉刷新功能之前一直使用的是XlistView很方便我前面的博客有介绍 SwipeRefreshLayout是google官方推出的下拉刷新控件使用方法也比较简单 今天就来使用下SwipeRefres ...
随机推荐
- 14-vertical-aligin
01 行盒的理解 作用: 将当前行里的所有内容包裹起来 <!DOCTYPE html> <html lang="en"> <head> < ...
- Android 中的 perfboot工具
背景 开机首先加载bootloader,由bootloader启动kernel,然后运行init程序,有init启动Zygote,Zygote进程启动SystemServ进程,在SystemServe ...
- Markdown 文章 跳转
背景 在查阅一些文档的时候,一些比较优秀博客在文章中是带有目录的,点击就会跳转到指定的锚点. 在本人的某些文章中,也想尝试这样的效果. 做法 实现这样的效果有2种做法(不同之处在于 超链接的写法不同) ...
- 【动画进阶】类 ChatGpt 多行文本打字效果
今天我们来学习一个有意思的多行文本输入打字效果,像是这样: 这个效果其实本身并非特别困难,实现的方式也很多,在本文中,我们更多的会聚焦于整个多行打字效果最后的动态光标的实现. 也就是如何在文本不断变长 ...
- mysql求同比环比
-- 参考:SQL计算月环比.月同比_路易吃泡面的博客-CSDN博客 -- mysql同比环比 drop table if EXISTS ordertable; create table ordert ...
- Hadoop集群管理之fsimage和edits工作机制
客户端对hdfs进行写文件时会首先被记录在edits文件中. edits修改时元数据也会更新. 每次hdfs更新时edits先更新后客户端才会看到最新信息. fsimage:是namenode中关于元 ...
- 当一名有着 10w+ 听众的播客主播开始做 App
名字: AAAny 开发者 / 团队: AAAny Team 平台: iOS 请简要介绍下这款产品 AAAny 是一个专为 AMA(我们称之为 AAA)设计的 App.多解释一下的话,其实 AAA 是 ...
- 解决方案 | tk.entry数字验证(输入框如何保证只能输入数字)
from tkinter import * root = Tk() # 创建文本框 entry = Entry(root) entry.pack() # 设置文本框只能输入数字 entry.confi ...
- Ubuntu16.04升级openssh-9.8p1
7月1日OpenSSH官方发布安全更新,忙着处理的同时记录一下升级过程. 系统环境 root@NServer:~# cat /proc/version Linux version 3.4.113-su ...
- ElementUI Dialog 结合Vue实现对话框body“二分”布局
Dialog 结合Vue实现对话框body"二分"布局 需求描述 如下图, 把对话框body内容部分,分成上下两部分,其中上部分高度根据窗口大小动态调整,如果内容过多,则出现滚动条 ...