2701

    技术2024-04-15  17

    给你一个思路。 单击的时候取得 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);

    }

     

     

    最新回复(0)