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

http://www.moon-soft.com/doc/10226.htm

VCMSFlexGrid内嵌EDITCOMBOBOX的实现方法

Key Words: MSFlexGrid 内嵌控件

CSDN网络中经常会看到有人问起各种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实现应用的朋友们,这个小东东一定能起到抛砖引玉的作用。

 

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

[email protected]

你可能感兴趣的:(在VC中MSFlexGrid内嵌EDIT、COMBOBOX的实现方法)