我绝对需要使用 CDC :: DrawText()函数来动态地(重新)写入单个节点的文本, WITHOUT < / em> 多次调用SetItemText()函数(强制必需)。
然后,我编写了自己的 OnPaint()方法的实现。 我还实现了一个 DrawItems()方法,它在CTreeCtrl中绘制节点。
由于我不想修改其他任何而不是单个节点旁边的单个标签,所以我需要重新实现大部分原始 CTreeCtrl :: OnPaint()代码。
我只有两个疑问:
简单地说,当前选定的节点应显示突出显示。
下面是一些简化的,自动解释的代码:
void MyDlg::OnPaint()
{
CPaintDC dc(this);
CDC dc_ff;
CBitmap bm_ff;
CBitmap *bm_old;
CFont *font;
CFont *old_font;
CFont fontDC;
int old_mode;
GetClientRect(&m_Rect);
dc_ff.CreateCompatibleDC( &dc );
bm_ff.CreateCompatibleBitmap( &dc, m_Rect.Width(), m_Rect.Height() );
dc_ff.SelectObject( &bm_ff );
font = GetFont();
old_font = dc_ff.SelectObject( font );
// Could / Should be called here?
CWnd::DefWindowProc(WM_PAINT, (WPARAM)dc.m_hDC, 0);
old_mode = dc_ff.SetBkMode(TRANSPARENT);
dc_ff.FillSolidRect(m_Rect, dc_ff.GetBkColor());
DrawItems( &dc_ff ); // DrawItems() member function draws single nodes of CTreeCtrl
dc.BitBlt( m_Rect.left, m_Rect.top, m_Rect.Width(), m_Rect.Height(), &dc_ff, 0, 0, SRCCOPY);
dc_ff.SelectObject( old_font );
dc_ff.SetBkMode( old_mode );
dc_ff.SelectObject( bm_old );
}
void MyDlg::DrawItems( CDC *pDC )
{
// draw items
HTREEITEM show_item, parent;
CRect rc_item;
CString name;
DWORD tree_style;
int count = 0;
int state;
bool selected;
bool has_children;
show_item = GetFirstVisibleItem();
if ( show_item == NULL )
return;
color = pDC->GetTextColor();
tree_style = ::GetWindowLong( m_hWnd, GWL_STYLE );
do
{
state = GetItemState( show_item, TVIF_STATE );
parent = GetParentItem( show_item );
has_children = ItemHasChildren( show_item ) || parent == NULL;
selected = (state & TVIS_SELECTED) && ((this == GetFocus()) ||
(tree_style & TVS_SHOWSELALWAYS));
if ( GetItemRect( show_item, rc_item, TRUE ) )
{
if ( has_children || selected )
{
if ( selected )
{
// HERE i need to
}
else
// do some stuff...
if ( has_children )
{
HICON icon;
// HERE I need to load CTreeCtrl nodes _DEFAULT_icon
icon = LoadIcon(NULL, IDI_ASTERISK);
if ( icon != NULL )
DrawIconEx( pDC->m_hDC, rc_item.left - 18, rc_item.top, icon, 16, 16,0,0, DI_NORMAL );
}
}
if ( !has_children )
{
HICON icon;
*// HERE I need to load CTreeCtrl nodes _DEFAULT_icon*
icon = LoadIcon(NULL, IDI_ASTERISK);
if ( icon != NULL )
DrawIconEx( pDC->m_hDC, rc_item.left - 18, rc_item.top, icon, 16, 16,0,0, DI_NORMAL );
}
name = GetItemText( show_item );
// ...
if ( selected )
{
pDC->DrawText( "Temp", rc_item, DT_LEFT );
}
else
{
pDC->DrawText( "Temp", rc_item, DT_LEFT );
}
//if ( state & TVIS_BOLD )
// pDC->SelectObject( font );
}
} while ( (show_item = GetNextVisibleItem( show_item )) != NULL );
}
我需要的只是几乎标准的 CTreeCtrl :: OnPaint()实现的源代码。 任何建议/帮助表示赞赏。 : - )
由于
IT。
答案 0 :(得分:0)
您不需要重载onPaint。如果将树项目文本设置为LPSTR_TEXTCALLBACK,则每次显示该项目时,CtreeCtrl都将触发消息TVN_GETDISPINFO以检索新文本。如果它在父窗口中,则使用ON_NOTIFY重新引用消息处理程序;如果是子类CTreeCtrl,则使用ON_NOTIFY_REFLECT重新引用消息处理程序。此消息处理程序可以分配您想要的文本,但允许treeCtrl继续正常绘制。
如果你去了父Cwnd路线,你需要
所以你需要在cpp文件中分配消息处理程序:
BEGIN_MESSAGE_MAP(MyCWnd,CWnd)
ON_NOTIFY(TVN_GETDISPINFO,tree_ctl_id,CustomTreeControl :: OnGetdispinfo) END_MESSAGE_MAP()
标题中的函数原型
afx_msg void OnGetdispinfo(NMHDR * pNMHDR,LRESULT * pResult);
这是在课程定义的最后
DECLARE_MESSAGE_MAP()
以及处理请求的实际功能
void ColumnTreeControl :: OnGetdispinfo(NMHDR * pNMHDR,LRESULT * pResult) { NMTVDISPINFO * pDispInfo =(NMTVDISPINFO )pNMHDR; TVITEM item =&amp; pDispInfo-&gt; item;
if(item->mask & TVIF_TEXT )
{
item->pszText " YOUR CODE HERE";
}
*pResult = 0;
}