热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MFCListControl控件添加单元格编辑和单元格下拉列表项以适用于数据库相关操作

作为现代的软件,往往是连着数据库的,而连着和用户方便地操作之间,还有着界面这道坎。MFC是Windows上比较好开发用户界面的框架

       作为现代的软件,往往是连着数据库的,而连着和用户方便地操作之间,还有着界面这道坎。MFC是Windows上比较好开发用户界面的框架,然而其自带的控件中没有对于数据库表格支持较好的控件,而使用网上提到的 DataGrid 等控件在本人的win8.1+VS2013平台上老出现找不到控件或者头文件的问题,搞的烦死人。最后想到 List Control 控件只要稍作修改,加上单元格编辑和单元格下拉列表,其实就能和数据库进行良好的对接,一百度,果然有人已经做了这件事,实在是太让人感动了!  下面将介绍一下如何实现上面提到的这两个功能,由于是参考别人的,如果想看原文:请点击这里。至于跟原文最大的不同,我想是我把流程操作尽我所能写的详细,并最大化的简化代码,最后顺带加上一些本人觉得能给数据库操作带来方便的功能的实现方法上去。 OK,下面我们来看看这两个功能如何实现吧。其实实现的原理很简单,就是在点击到列表相应单元格的时候,在相应的位置生成一个跟单元格大小相当的编辑框或者下拉列表框并贴在单元格上面而已。 实现如下: 在对话框类声明中添加如下变量和函数:

int e_Item; //刚编辑的行
int e_SubItem; //刚编辑的列 CEdit m_Edit; //生成单元编辑框对象
bool haveeditcreate;//标志编辑框已经被创建
void createEdit(NM_LISTVIEW *pEditCtrl, CEdit *createdit, int &Item, int &SubItem, bool &havecreat);//创建单元格编辑框函数
void distroyEdit(CListCtrl *list, CEdit* distroyedit, int &Item, int &SubItem);//销毁单元格编辑框对象CComboBox m_comBox;//生产单元格下拉列表对象
bool haveccomboboxcreate;//标志下拉列表框已经被创建
void createCcombobox(NM_LISTVIEW *pEditCtrl, CComboBox *createccomboboxobj, int &Item, int &SubItem, bool &havecreat);//创建单元格下拉列表框函数
void distroyCcombobox(CListCtrl *list, CComboBox* distroyccomboboxobj, int &Item, int &SubItem);//销毁单元格下拉列表框

在对话框的初始化函数OnInitDialog()中添加初始化代码如下:

haveeditcreate = false;//初始化标志位,表示还没有创建编辑框
haveccomboboxcreate = false;//初始化标志位,表示还没有创建下拉列表框
RECT m_rect;
m_list.GetClientRect(&m_rect); //获取list的客户区,方便调节每一列的宽度
m_list.SetExtendedStyle(LVS_EX_GRIDLINES | LVS_EX_FULLROWSELECT); //设置list风格,LVS_EX_GRIDLINES为网格线(只适用与report风格的listctrl)//LVS_EX_FULLROWSELECT为选中某行使整行高亮(只适用与report风格的listctrl)
m_list.InsertColumn(0, _T("学号"), LVCFMT_LEFT, m_rect.right / 4);
m_list.InsertColumn(1, _T("姓名"), LVCFMT_LEFT, m_rect.right / 4);
m_list.InsertColumn(2, _T("性别"), LVCFMT_LEFT, m_rect.right / 4);
m_list.InsertColumn(3, _T("班级"), LVCFMT_LEFT, m_rect.right / 4);m_list.InsertItem(0, _T("09090901"));//添加第一个学生数据
m_list.SetItemText(0, 1, _T("小李"));
m_list.SetItemText(0, 2, _T("男"));
m_list.SetItemText(0, 3, _T("计科0901"));m_list.InsertItem(1, _T("09090902"));//添加第二个学生数据
m_list.SetItemText(1, 1, _T("小王"));
m_list.SetItemText(1, 2, _T("男"));
m_list.SetItemText(1, 3, _T("计科0902"));

为列表框添加单击响应函数:

void CNephoVisionDatabaseDlg::OnNMClickListStudentdata(NMHDR *pNMHDR, LRESULT *pResult)
{LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast(pNMHDR);NM_LISTVIEW *pEditCtrl = (NM_LISTVIEW *)pNMHDR;printf("行:%d,列:%d\n", pEditCtrl->iItem, pEditCtrl->iSubItem);if (pEditCtrl->iItem==-1)//点击到非工作区{if (haveeditcreate == true)//如果之前创建了编辑框就销毁掉{distroyEdit(&m_list, &m_Edit, e_Item, e_SubItem);//销毁单元格编辑框对象haveeditcreate = false;}if (haveccomboboxcreate == true)//如果之前创建了下拉列表框就销毁掉{distroyCcombobox(&m_list, &m_comBox, e_Item, e_SubItem);haveccomboboxcreate = false;}}else if (pEditCtrl->iSubItem != 2)//如果不是性别选项{if (haveccomboboxcreate == true)//如果之前创建了编辑框就销毁掉{distroyCcombobox(&m_list, &m_comBox, e_Item, e_SubItem);haveccomboboxcreate = false;}if (haveeditcreate == true){if (!(e_Item == pEditCtrl->iItem && e_SubItem == pEditCtrl->iSubItem))//如果点中的单元格不是之前创建好的{distroyEdit(&m_list, &m_Edit, e_Item, e_SubItem);haveeditcreate = false;createEdit(pEditCtrl, &m_Edit, e_Item, e_SubItem, haveeditcreate);//创建编辑框}else//点中的单元格是之前创建好的{m_Edit.SetFocus();//设置为焦点 }}else{e_Item = pEditCtrl->iItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理e_SubItem = pEditCtrl->iSubItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理createEdit(pEditCtrl, &m_Edit, e_Item, e_SubItem, haveeditcreate);//创建编辑框}}else//如果是性别选项,在单元格处生成下拉列表项{if (haveeditcreate == true)//如果之前创建了编辑框就销毁掉{distroyEdit(&m_list, &m_Edit, e_Item, e_SubItem);haveeditcreate = false;}if (haveccomboboxcreate == true){if (!(e_Item == pEditCtrl->iItem && e_SubItem == pEditCtrl->iSubItem))//如果点中的单元格不是之前创建好的{distroyCcombobox(&m_list, &m_comBox, e_Item, e_SubItem);haveccomboboxcreate = false;createCcombobox(pEditCtrl, &m_comBox, e_Item, e_SubItem, haveccomboboxcreate);//创建编辑框m_comBox.AddString(L"男");m_comBox.AddString(L"女");m_comBox.ShowDropDown();//自动下拉}else//点中的单元格是之前创建好的{m_comBox.SetFocus();//设置为焦点 }}else{e_Item = pEditCtrl->iItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理e_SubItem = pEditCtrl->iSubItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理createCcombobox(pEditCtrl, &m_comBox, e_Item, e_SubItem, haveccomboboxcreate);//创建编辑框m_comBox.AddString(L"男");m_comBox.AddString(L"女");m_comBox.ShowDropDown();//自动下拉}}*pResult = 0;
}

写创建与销毁单元格编辑框和下拉列表框函数的实现:

void CNephoVisionDatabaseDlg::createEdit(NM_LISTVIEW *pEditCtrl, CEdit *createdit, int &Item, int &SubItem, bool &havecreat)//创建单元格编辑框函数//pEditCtrl为列表对象指针,createdit为编辑框指针对象,//Item为创建单元格在列表中的行,SubItem则为列,havecreat为对象创建标准
{Item = pEditCtrl->iItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理SubItem = pEditCtrl->iSubItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理createdit->Create(ES_AUTOHSCROLL | WS_CHILD | ES_LEFT | ES_WANTRETURN,CRect(0, 0, 0, 0), this, IDC_EDIT_CREATEID);//创建编辑框对象,IDC_EDIT_CREATEID为控件ID号3000,在文章开头定义havecreat = true;createdit->SetFont(this->GetFont(), FALSE);//设置字体,不设置这里的话上面的字会很突兀的感觉createdit->SetParent(&m_list);//将list control设置为父窗口,生成的Edit才能正确定位,这个也很重要CRect EditRect;m_list.GetSubItemRect(e_Item, e_SubItem, LVIR_LABEL, EditRect);//获取单元格的空间位置信息EditRect.SetRect(EditRect.left+1, EditRect.top+1, EditRect.left + m_list.GetColumnWidth(e_SubItem)-1, EditRect.bottom-1);//+1和-1可以让编辑框不至于挡住列表框中的网格线CString strItem = m_list.GetItemText(e_Item, e_SubItem);//获得相应单元格字符createdit->SetWindowText(strItem);//将单元格字符显示在编辑框上createdit->MoveWindow(&EditRect);//将编辑框位置放在相应单元格上createdit->ShowWindow(SW_SHOW);//显示编辑框在单元格上面createdit->SetFocus();//设置为焦点 createdit->SetSel(-1);//设置光标在文本框文字的最后
}void CNephoVisionDatabaseDlg::distroyEdit(CListCtrl *list,CEdit* distroyedit, int &Item, int &SubItem)
{CString meditdata;distroyedit->GetWindowTextW(meditdata);list->SetItemText(Item, SubItem, meditdata);//获得相应单元格字符distroyedit->DestroyWindow();//销毁对象,有创建就要有销毁,不然会报错
}void CNephoVisionDatabaseDlg::createCcombobox(NM_LISTVIEW *pEditCtrl, CComboBox *createccomboboxobj, int &Item, int &SubItem, bool &havecreat)//创建单元格下拉列表框函数
//pEditCtrl为列表对象指针,createccombobox为下拉列表框指针对象,
//Item为创建单元格在列表中的行,SubItem则为列,havecreat为对象创建标准
{Item = pEditCtrl->iItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理SubItem = pEditCtrl->iSubItem;//将点中的单元格的行赋值给“刚编辑过的行”以便后期处理havecreat = true;createccomboboxobj->Create(WS_CHILD | WS_VISIBLE | CBS_DROPDOWN | CBS_OEMCONVERT, CRect(0, 0, 0, 0), this, IDC_COMBOX_CREATEID);createccomboboxobj->SetFont(this->GetFont(), FALSE);//设置字体,不设置这里的话上面的字会很突兀的感觉createccomboboxobj->SetParent(&m_list);//将list control设置为父窗口,生成的Ccombobox才能正确定位,这个也很重要CRect EditRect;m_list.GetSubItemRect(e_Item, e_SubItem, LVIR_LABEL, EditRect);//获取单元格的空间位置信息EditRect.SetRect(EditRect.left + 1, EditRect.top + 1, EditRect.left + m_list.GetColumnWidth(e_SubItem) - 1, EditRect.bottom - 1);//+1和-1可以让编辑框不至于挡住列表框中的网格线CString strItem = m_list.GetItemText(e_Item, e_SubItem);//获得相应单元格字符createccomboboxobj->SetWindowText(strItem);//将单元格字符显示在编辑框上createccomboboxobj->MoveWindow(&EditRect);//将编辑框位置放在相应单元格上createccomboboxobj->ShowWindow(SW_SHOW);//显示编辑框在单元格上面
}void CNephoVisionDatabaseDlg::distroyCcombobox(CListCtrl *list, CComboBox* distroyccomboboxobj, int &Item, int &SubItem)
{CString meditdata;distroyccomboboxobj->GetWindowTextW(meditdata);list->SetItemText(Item, SubItem, meditdata);//更新相应单元格字符distroyccomboboxobj->DestroyWindow();//销毁对象,有创建就要有销毁,不然会报错
}

OK,到此我们已经实现了添加单元格编辑框和单元格下拉列表框的功能了,点击除性别列外的列表单元格看到以下效果:

点击性别列单元格则看到下拉列表效果,如下:

在单元格编辑框中写东西的时候往往喜欢按回车,这个时候如果不加处理的话会发现一个很崩溃的现象,程序直接退出了。这个可以重载对话框类的OnOK()函数来避免。就是在对话框类声明中添加

afx_msg void OnOK();

然后将其定义为什么都不做:

void CNephoVisionDatabaseDlg::OnOK()
{
}

这样就不会有按回车直接退出程序的事情发生了。

此外,如果希望在编辑框或者下拉列表框失去焦点的时候就将数据列表中的相关信息更新,可以响应动态创建的这两个控件的失去焦点消息,方法如下: 在对话框类声明中添加

afx_msg void OnKillfocusEdit();//动态生成编辑框失去焦点响应函数
afx_msg void OnKillfocusCcomboBox();//动态生成下拉列表框失去焦点响应函数

在对话框类实现的消息映射描述中添加

ON_EN_KILLFOCUS(IDC_EDIT_CREATEID, &CNephoVisionDatabaseDlg::OnKillfocusEdit)//添加动态生成编辑框的失去焦点响应函数
ON_CBN_KILLFOCUS(IDC_COMBOX_CREATEID, &CNephoVisionDatabaseDlg::OnKillfocusCcomboBox)

这样这两个控件在失去焦点的时候就会响应分别定义好的函数,我们只需要在相应的函数添加处理代码就可以了:

void CNephoVisionDatabaseDlg::OnKillfocusEdit()
{if (haveeditcreate == true)//如果之前创建了编辑框就销毁掉{distroyEdit(&m_list, &m_Edit, e_Item, e_SubItem);//销毁单元格编辑框对象haveeditcreate = false;}if (haveccomboboxcreate == true)//如果之前创建了下拉列表框就销毁掉{distroyCcombobox(&m_list, &m_comBox, e_Item, e_SubItem);haveccomboboxcreate = false;}
}void CNephoVisionDatabaseDlg::OnKillfocusCcomboBox()
{if (haveeditcreate == true)//如果之前创建了编辑框就销毁掉{distroyEdit(&m_list, &m_Edit, e_Item, e_SubItem);//销毁单元格编辑框对象haveeditcreate = false;}if (haveccomboboxcreate == true)//如果之前创建了下拉列表框就销毁掉{distroyCcombobox(&m_list, &m_comBox, e_Item, e_SubItem);haveccomboboxcreate = false;}
}

文章转载于:

MFCListControl控件添加单元格编辑和单元格下拉列表项以适用于数据库相关操作 - 数据库其他综合 - 红黑联盟
https://www.2cto.com/database/201411/355997.html
 


推荐阅读
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • Java学习笔记之使用反射+泛型构建通用DAO
    本文介绍了使用反射和泛型构建通用DAO的方法,通过减少代码冗余度来提高开发效率。通过示例说明了如何使用反射和泛型来实现对不同表的相同操作,从而避免重复编写相似的代码。该方法可以在Java学习中起到较大的帮助作用。 ... [详细]
  • MySQL语句大全:创建、授权、查询、修改等【MySQL】的使用方法详解
    本文详细介绍了MySQL语句的使用方法,包括创建用户、授权、查询、修改等操作。通过连接MySQL数据库,可以使用命令创建用户,并指定该用户在哪个主机上可以登录。同时,还可以设置用户的登录密码。通过本文,您可以全面了解MySQL语句的使用方法。 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • http:my.oschina.netleejun2005blog136820刚看到群里又有同学在说HTTP协议下的Get请求参数长度是有大小限制的,最大不能超过XX ... [详细]
  • 本文详细介绍了在ASP.NET中获取插入记录的ID的几种方法,包括使用SCOPE_IDENTITY()和IDENT_CURRENT()函数,以及通过ExecuteReader方法执行SQL语句获取ID的步骤。同时,还提供了使用这些方法的示例代码和注意事项。对于需要获取表中最后一个插入操作所产生的ID或马上使用刚插入的新记录ID的开发者来说,本文提供了一些有用的技巧和建议。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 本文讨论了在数据库打开和关闭状态下,重新命名或移动数据文件和日志文件的情况。针对性能和维护原因,需要将数据库文件移动到不同的磁盘上或重新分配到新的磁盘上的情况,以及在操作系统级别移动或重命名数据文件但未在数据库层进行重命名导致报错的情况。通过三个方面进行讨论。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • 本文介绍了在Linux下安装和配置Kafka的方法,包括安装JDK、下载和解压Kafka、配置Kafka的参数,以及配置Kafka的日志目录、服务器IP和日志存放路径等。同时还提供了单机配置部署的方法和zookeeper地址和端口的配置。通过实操成功的案例,帮助读者快速完成Kafka的安装和配置。 ... [详细]
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • 移动端常用单位——rem的使用方法和注意事项
    本文介绍了移动端常用的单位rem的使用方法和注意事项,包括px、%、em、vw、vh等其他常用单位的比较。同时还介绍了如何通过JS获取视口宽度并动态调整rem的值,以适应不同设备的屏幕大小。此外,还提到了rem目前在移动端的主流地位。 ... [详细]
  • 如何提高PHP编程技能及推荐高级教程
    本文介绍了如何提高PHP编程技能的方法,推荐了一些高级教程。学习任何一种编程语言都需要长期的坚持和不懈的努力,本文提醒读者要有足够的耐心和时间投入。通过实践操作学习,可以更好地理解和掌握PHP语言的特异性,特别是单引号和双引号的用法。同时,本文也指出了只走马观花看整体而不深入学习的学习方式无法真正掌握这门语言,建议读者要从整体来考虑局部,培养大局观。最后,本文提醒读者完成一个像模像样的网站需要付出更多的努力和实践。 ... [详细]
author-avatar
手机用户2602925875
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有