2701

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

}

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章