给你一个思路。 单击的时候取得 cell 的矩形; new 一个 CComboBox (或者预先建立好,隐藏),移动到矩形中,显示。 单击其它地方的时候把选中的值显示在 cell 中 delete
做过添加 Eidt 框的,是重载 CListCtrl ,在其 OnCreate 事件那里预先创建一个 CEdit ,需要的时候再把 Eidt 框 move 到那里再显示,不需要时就 Hide
CListCtrl *pListCtrl = (CListCtrl*)GetDlgItem(IDC_LIST); CRect rc(0,0,0,0); pListCtrl->GetItemRect(m_nCurSel,&rc,LVIR_BOUNDS); 可以获得 m_nCurSel 表示的行的坐标 pListCtrl->GetColumnWidth() 可以获得每一列的宽度,加上 rc ,就可以计算出要放 COMBOBOX 的那个单元的 rc 。 创建好一个 COMBOBOX ,用 pCombo->SetParent(pListCtrl); 设置 COMBOBOX 的父指针为列表,这样就不用考虑相对坐标。 然 后 m_pCombo->SetWindowPos(NULL,rc.left,rc.top,rc.Width(),rc.Height(),SWP_SHOWWINDOW) 就可以在特定的单元显示下拉列表了。(要显示多个下拉列表,只需要按照同样的步骤重复执行就可以了。) 复选框也是一样的计算。
你需要在对话框中放置一个组合框控件,初始对话框时,不要让组合框显示。当用户点击单元格中的文本时,让组合框显示,并将单元格文本赋给组合框。当用户鼠标离开组合框时,隐藏组合框,并将组合框中的文本赋给单元格。 1. 定义成员变量: CComboBox m_Cmb; 、、将它与组合框控件关联, int m_row,m_col; // 记录用户点击的那个单元格所在的行与列号 2. 添加消息声明 afx_msg void OnCbnKillfocusCombo1(); afx_msg void OnNMClickList2(NMHDR *pNMHDR, LRESULT *pResult); virtual BOOL PreTranslateMessage(MSG* pMsg); 3. 添加消息注册 BEGIN_MESSAGE_MAP(CListDlg1Dlg, CDialog) .... ON_CBN_KILLFOCUS(IDC_COMBO1, &CListDlg1Dlg::OnCbnKillfocusCombo1) ON_NOTIFY(NM_CLICK, IDC_LIST2, &CListDlg1Dlg::OnNMClickList2) END_MESSAGE_MAP() 4. 对话框初始化时,添加如下代码: m_Cmb.SetParent(&m_List); // 确保CComboBox 的坐标是相对于列表控件而言的。 5. 当用户点单元格的文本域时,显示出CComboBox 控件
C/C++ code
void CListDlg1Dlg::OnNMClickList2(NMHDR *pNMHDR, LRESULT *pResult)
{
// TODO: Add your control notification handler code here
LPNMLISTVIEW pNMTreeView = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);
// TODO: 在此添加控件通知处理程序代码
POINT PT;
GetCursorPos(&PT);
m_List.ScreenToClient(&PT);
LVHITTESTINFO hitInfo;
hitInfo.pt=PT;
//hitem=m_Tree.GetSelectedItem();
m_List.SubItemHitTest(&hitInfo);
if (hitInfo.flags & LVHT_ONITEMLABEL) // 判断是否单击在文本上
{
CRect rect;
m_List.GetSubItemRect(hitInfo.iItem,hitInfo.iSubItem,LVIR_BOUNDS,rect);
if (hitInfo.iSubItem== 0 )
{
rect.right=rect.left+m_List.GetColumnWidth( 0 );
}
CString mes=m_List.GetItemText(hitInfo.iItem,hitInfo.iSubItem);
rect.InflateRect( 0 , 0 , 0 , 2 ); // 增大组合框的高度使其可以容纳整行文本。
m_col=hitInfo.iSubItem;
m_row=hitInfo.iItem;
m_Cmb.MoveWindow(&rect,TRUE);
m_Cmb.ShowWindow(SW_NORMAL);
m_Cmb.SetWindowText(mes);
m_Cmb.BringWindowToTop();
m_Cmb.SetFocus(); // 使组合框聚焦
}
*pResult = 1 ;
}
6. 当用户鼠标离开CComboBox 控件时,设置单元格文本,并隐藏CComboBox 控件
C/C++ code
void CListDlg1Dlg::OnCbnKillfocusCombo1()
{
// TODO: 在此添加控件通知处理程序代码
POINT pt;
int i,j;
GetCursorPos(&pt);
m_List.ScreenToClient(&pt);
CRect rect;
m_List.GetSubItemRect(m_row,m_col,LVIR_BOUNDS,rect);
if (!rect.PtInRect(pt)) // 如果单击在一个节点文本区域内
{
CString text;
m_Cmb.GetWindowText(text);
m_List.SetItemText(m_row,m_col,text);
m_Cmb.ShowWindow(SW_HIDE); // 将组合框隐藏
}
}
7. 当用户在CCombobox 中按下[Enter] 键时,设置单元格文本,并隐藏CComboBox 控件
C/C++ code
BOOL CListDlg1Dlg::PreTranslateMessage(MSG *pMsg)
{
if (pMsg->wParam==VK_RETURN)
{
int i,j;
CString text;
m_Cmb.GetWindowText(text);
m_List.SetItemText(m_row,m_col,text);
m_Cmb.ShowWindow(SW_HIDE); // 隐藏组合框
return TRUE; // 阻止了父类对消息的处理,使得按下回车键时,不会关闭对话框
}
return CDialog::PreTranslateMessage(pMsg);
}