在VC中MSFlexGrid内嵌EDIT、COMBOBOX的实现方法

    技术2022-05-11  110

    VCMSFlexGrid内嵌EDITCOMBOBOX的实现方法

    Key Words: MSFlexGrid 内嵌控件

    网络中经常会看到有人问起各种GRID控件内嵌EDITCOMBOBOX的实现方法,本人在前阶段的开发中也遇到这方面的困难,在网络上找了又找,大多是针对ListViewDBGrid的,而对于MSFlexGrid的实现,则少之又少。在广大网友的支持下,终于本人找到了实现MSFlexGrid内嵌EDITCOMBOBOX的一种方法,我想本文对于采用MSFlexGrid进行应用开发的朋友一定有相当大的帮助。

    总结一些网友以及本人在最初实现MSFlexGrid内嵌控件失败的原因,大多是由两方面造成的:

    1、          坐标系转换问题,MSFlexGrid采用的坐标系和一般的控件不同,所以在操作时,需要进行转换。

    2、          控件在创建上的问题,如果你把控件直接创建在主窗口中,那么往往会存在,程序运行时,鼠标一点网格,控件就HIDE掉,所以在创建控件EDITCOMBOBOX时,要以FlexGrid为父窗口。

    下面,我用一个示例程序来简单的说明一下,同时我们的示例程序还实现了在FlexGrid中按TAB键跳至下一网格[下面提到的网格均指MSFlexGrid中的小单元格]的功能。

    程序运行界面:

    首先,在对话框的初始化中调用我们的初始化函数:

    void CProg5Dlg::InitControls()

    {

    //创建各个内嵌控件

    m_edit.Create(WS_CHILD,CRect(0,0,0,0),&m_FlexGrid,IDC_EDIT);

    m_cmb.Create(WS_CHILD|CBS_DROPDOWNLIST,CRect(0,0,0,0),&m_FlexGrid,IDC_CMB);

    //设置为和主窗口相同字体

    m_edit.SetFont(GetFont());

    m_cmb.SetFont(GetFont());

    //用数据填充Grid ComboBox

    long lRow ;

    long lRowCount = m_FlexGrid.GetRows();

    long lCol ;

    long lColCount = m_FlexGrid.GetCols();

     

    for (lRow = 1; lRow < lRowCount; lRow++)

    {

            m_FlexGrid.SetRow(lRow);

     

            for(lCol = 1; lCol < lColCount; lCol++)

            {

                   m_FlexGrid.SetCol(lCol);

     

                   CString strText;

                   strText.Format("%ld-%ld",lRow,lCol);

                   //用数据填充Grid

                   m_FlexGrid.SetText(strText);

                   //用数据填充ComboBox

                   m_cmb.AddString(strText);

            }

    }

    }

     

    其中m_editm_cmb是我们声明的类数据成员:

    private:

    CEdit m_edit;

    CComboBox m_cmb;

    它们的创建一定要用.Create的方法并以MSFlexGrid为父窗口,要不然,程序运行时,你一点MsflexGrid,你的EditComboBox就不见了 [这是因为,MSFlexGrid和你的EditComboBox同以Dialog为父窗口,你点了MsflexGrid,在Z坐标上,它就盖住了你的内嵌控件] ,因为在创建之后,它们采用的字体可能和你的主窗口风格不一致,所以还要设置一下字体。

    接下来就是程序中最重要的一个函数了:

    void CProg5Dlg::GridEdit(WORD nKeyAsciiCode, CWnd *p_wnd)

    {

    if(p_wnd == NULL)

    {//得到当前编辑的网格的内嵌控件是m_edit or m_cmb

            p_wnd = GetThisCellMaskControl();

    }

     

    ASSERT(p_wnd != NULL);

    //支持坐标变换

    CDC* pDC = m_FlexGrid.GetDC();

    int nLogX = pDC->GetDeviceCaps(LOGPIXELSX);

    int nLogY = pDC->GetDeviceCaps(LOGPIXELSY);

    ReleaseDC(pDC);

    CString sz;

     

    //当有文字输入时,如果当前控件是Edit,那么光标到末尾

    if (nKeyAsciiCode >= 0 && nKeyAsciiCode < ' ')

    {

            if(p_wnd->IsKindOf(RUNTIME_CLASS(CEdit)))

            {

                   ((CEdit *)p_wnd)->SetSel(-1, -1);

            }

    }

    else

    {

            CString Input = "  ";

            p_wnd->GetWindowText(sz);

     

           

            if (nKeyAsciiCode > 0x100)

            {//用来支持汉字输入

                   Input.SetAt(0, nKeyAsciiCode >> 8);

                   Input.SetAt(1, nKeyAsciiCode & 0xff);

            }

            else

            {//非汉字

                   Input = (char)nKeyAsciiCode;

            }

     

            sz += Input;

            p_wnd->SetWindowText(sz);

    }

     

    if(p_wnd->IsKindOf(RUNTIME_CLASS(CComboBox)))

    {

            p_wnd->MoveWindow(

            (m_FlexGrid.GetCellLeft() * nLogX)/1440 - 3,

            (m_FlexGrid.GetCellTop() * nLogY)/1440  - 3,

            (m_FlexGrid.GetCellWidth()* nLogX)/1440 ,

            (m_FlexGrid.GetCellHeight()* nLogY)/1440 + 100,FALSE);

    }

    else if(p_wnd->IsKindOf(RUNTIME_CLASS(CEdit)))

    {

    p_wnd->MoveWindow(

            (m_FlexGrid.GetCellLeft() * nLogX)/1440 - 3,

            (m_FlexGrid.GetCellTop() * nLogY)/1440  - 3,

            (m_FlexGrid.GetCellWidth()* nLogX)/1440,

            (m_FlexGrid.GetCellHeight()* nLogY)/1440,FALSE);

    }

    else

    {

            ASSERT(0);

    }

    //显示我们的控件

    p_wnd->ShowWindow(SW_SHOW);

    p_wnd->SetFocus();

    p_wnd->GetWindowText(sz);

     

    if(p_wnd->IsKindOf(RUNTIME_CLASS(CEdit)))

    {

           ((CEdit *)p_wnd)->SetSel(sz.GetLength(), sz.GetLength(), FALSE);

    }

     

    m_FlexGrid.RedrawWindow();

    }

     

    [说明:这个函数部分代码并非原创]

     

    这个函数的作用,主要是支持MSFlexGrid的编辑,并把你的内嵌控件显示出来,当然,如果它是一个EDIT,那么我们有责任把EDIT内的光标置于EDIT中字串的最后。这个函数有两个参数:

    WORD nKeyAsciiCode:如果激活编辑FlexGrid的事件是一次按键,那么这个参数当然就和按键的信息有关了,另外在函数中,通过它可以实现支持中文,比如:您把焦点放到一个内嵌为EDIT的网格中[只是让FlexGrid的网格得到焦点,而不让内嵌的EDIT显示出来],直接输入中文,然后就会发现,网格自动进入编辑状态,并且你输入的中文汉字位于字串的最后:

    第二个参数:

     CWnd *p_wnd可以指定你想使用的内嵌控件,传入时可以使用&m_edit&m_cmb

    当然,你可以不指定它而用我们当初设置好的规则[这规则是指MSFlexGrid哪列固定采用哪个内嵌控件],这是用什么实现的呢?

    看到函数GridEdit中的

    if(p_wnd == NULL)

    {

            p_wnd = GetThisCellMaskControl();

    }

    了吧?

    CWnd * CProg5Dlg::GetThisCellMaskControl()

    {

    switch(m_FlexGrid.GetCol())

    {

    //第一列,第三列用ComboBox做为内嵌控件

    case 1:

    case 3:

            return &m_cmb;

            break;

    //其它的用Edit

    default:

            return &m_edit;

    }

    }

     

    我们可通过这个函数来设定一些基本规则。

    那么GridEdit函数是由谁来调用的呢?答案当然是由想实现编辑网格的事件触发的,在这里我设定为鼠标双击和网格有焦点时的按键事件:

    //鼠标双击激发

    void CProg5Dlg::OnDblClickMsflexgrid()

    {

    //第一行和第一列是固定的,我不想编辑它们

    if(m_FlexGrid.GetCol() == 0 || m_FlexGrid.GetRow() == 0)

            return;

    GridEdit(0,NULL);

    }

     

    //按键事件激发

    void CProg5Dlg::OnKeyPressMsflexgrid(short FAR* KeyAscii)

    {

    // TODO: Add your control notification handler code here

    if(m_FlexGrid.GetCol() == 0 || m_FlexGrid.GetRow() == 0)

            return;

    GridEdit(*KeyAscii,NULL);

    }

     

    引发这个函数之前,我们最好把ComboBox的当前选择内容和Edit的内容设为要编辑的那个格子的内容:

    void CProg5Dlg::OnEnterCellMsflexgrid()

    {

    // TODO: Add your control notification handler code here

    int nthisRow = m_FlexGrid.GetRow();

    if(nthisRow == 0)

    {

            return;

    }

     

    CString sz;

    sz = m_FlexGrid.GetText();

    CWnd *pWnd = GetThisCellMaskControl();

    if(pWnd->IsKindOf(RUNTIME_CLASS(CComboBox)))

    {

            ((CComboBox *)pWnd)->SelectString(-1,sz);

    }

    else if(pWnd->IsKindOf(RUNTIME_CLASS(CEdit)))

    {

            pWnd->SetWindowText(sz);

    }

    else

    {

            ASSERT(0);

    }

    }

     

    你觉得这样就行了吗?当然不行!我们在把焦点移到其它网格时[只是把焦点移走],我们有必要把那个已经显示出来的内嵌控件HIDE掉,并把它的内容传给网格,这也是我们编辑的目的。

    void CProg5Dlg::OnLeaveCellMsflexgrid()

    {

    // TODO: Add your control notification handler code here

    int nthisRow = m_FlexGrid.GetRow();

    if(nthisRow == 0)

    {

            return;

    }

    CString sz;

    CWnd * p_ThisWnd = GetThisCellMaskControl();

    ASSERT(p_ThisWnd != NULL);

     

    if (p_ThisWnd->IsWindowVisible())

    {

            p_ThisWnd->GetWindowText(sz);

            m_FlexGrid.SetText(sz); 

            p_ThisWnd->ShowWindow(SW_HIDE);

    }

     

    }

     

    基本上差不多了。

     

    下面简单介绍一下用Tab实现在MSFlexGrid的网格中跳转的问题。

    我一看到这种应用,马上想到采用PreTranslateMessage函数,这个函数可是真好用,一般实现什么窗口内焦点的跳转我都用它。

    在实现它之前,我们先定义一个跳到一下格子的函数:

    void CProg5Dlg::GoToNextCell()

    {

    if(m_FlexGrid.GetCol() == m_FlexGrid.GetCols() - 1)

    {

            if(m_FlexGrid.GetRow() != m_FlexGrid.GetRows() - 1)

            {

                   m_FlexGrid.SetRow(m_FlexGrid.GetRow() + 1);

                   m_FlexGrid.SetCol(1);

            }

            else

            {

                   return;

            }

    }

    else

    {

            m_FlexGrid.SetCol(m_FlexGrid.GetCol() + 1);

    }

    }

     

    在我们的PreTranslateMessage会调用它实现跳到下一网格中:

    BOOL CProg5Dlg::PreTranslateMessage(MSG* pMsg)

    {

    // TODO: Add your specialized code here and/or call the base class

    CWnd *pWnd = CWnd::FromHandle(pMsg->hwnd);

    CWnd *pCon = GetThisCellMaskControl();

     

    if(pMsg->message!=WM_KEYDOWN)

                   return CDialog::PreTranslateMessage(pMsg);

     

    switch(pMsg->wParam)

    {

    case VK_TAB:

            if(pCon->GetSafeHwnd() == pMsg->hwnd)

            {//如果按TAB时,处于EDIT状态,也会跳到下一格子

                   GoToNextCell();

                   return TRUE;

            }

            switch(pWnd->GetDlgCtrlID())

            {

           

            case IDC_MSFLEXGRID:

                   GoToNextCell();

                   return TRUE;

           

            }break;

    }

     

     

    return CDialog::PreTranslateMessage(pMsg);

    }

     

    好了,整个应用就讲完了,我想对于采用MSFlexGrid实现应用的朋友们,这个小东东一定能起到抛砖引玉的作用。

     

    如果您想得到它的源程序,请给我发邮件:

    mahongxi@hotmail.com

     


    最新回复(0)