給你一個思路。
單擊的時候取得
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);
}