Qt QComboBox之setEditable和currentTextChanged以及其源码分析

前言

最近做了一个QComboBox里有选项,然后选中选项之后就会自动触发条件搜索。然后我发现,在我初始化comboBox时,由于信号连接的原因会触发这个currentTextChanged信号。代码大致如下:

connect(ui->comboBox, &QComboBox::currentTextChanged,
this, &CountryType::slot_pageSearch); void Country::setComboBox()
{
QStringList content;
int maxLen = 0;
QFont font;
font.setFamily("Microsoft YaHei");
font.setPixelSize(16);
QFontMetrics fontMetrics(font); QString command = jointQueryComboBoxTextCommand();
QList<QStringList> texts = m_oracle->runSelectCommand(command);
foreach (QStringList text, texts) {
QString item = text.value(0)+"-"+text.value(1);
content.push_back(item);
// 计算最大宽度
maxLen = maxLen > fontMetrics.boundingRect(item).width() ?
maxLen :
fontMetrics.boundingRect(item).width();
} // comboBox的宽度为 文字的最大宽度 + 下拉箭头的宽度 + 文字两边的间距
ui->comboBox->setMinimumWidth(maxLen + 38 + 8);
ui->comboBox->clear();
// 填充一个空选项作为筛选所有
ui->comboBox->addItem("");
ui->comboBox->addItems(content);
} void Country::search()
{
setComboBox();
}

问题的出现

在我每一次对页面进行切换的时候,我发现这个search都会触发这个slot_pageSearch槽函数,然后执行条件搜索。

但是我今天突发奇想,我是不是应该让使用者能够手动的输入这个条件呢,于是我setEditable(true);,将编辑打开了。

也就是:

ui->comboBox->setEditable(true);

在设置了这个之后,我惊奇的发现,并没有像之前一样会触发slot_pageSearch这个槽函数。

问题分析

因为我只修改了ui->comboBox->setEditable(true);,所以我肯定,问题就是发生在这个地方,于是我在网上搜索与这个问题有关联的答案。

最后,我还是在QT的官方文档中对于currentText这个部分的介绍中,找到了问题的原因。

大概意思就是说,当你将QComboBox设置成可编辑的状态时(setEditable(true)),currentText就是当前的框内显示的文字。当不为可编辑的状态时,currentText就是当前的选项或者是一个空的字符串。

所以我猜想,设置成不可编辑状态时,由于我进行了一个条目的添加,所以就将当前的选项改变了。

currentTextChanged信号触发

于是我在正常的流程下,添加了一些打印语句,用于证实我的猜想。

    void Country::setComboBox() {
...
// 填充一个空选项作为筛选所有 qDebug() << "1";
ui->comboBox->addItem("");
qDebug() << "2";
ui->comboBox->addItems(content);
qDebug() << "3";
...
} void CountryType::slot_pageSearch()
{
...
qDebug() << "111";
...
}

输出的结果为:

    1
111
2
3

这也就表明了,我是在setItem之后,就会触发槽函数。但是具体为啥是这样的,为啥addItems不会触发currentTextChanged呢?

所以我带着问题,决定去源码里找答案

源码分析

// 代码调用结构
1. QComboBox::addItem(int , const QIcon &, const QString &, const QVariant &)
----> QStandardItem::setData(const QVariant &, int )
----> QStandardItemModelPrivate::itemChanged(QStandardItem *, const QVector<int> &)
----> signal: QStandardItemModel::dataChanged(QModelIndex,QModelIndex) slot: QComboBox::_q_dataChanged(QModelIndex,QModelIndex)
----> if (lineEdit) lineEdit->setText(); else emit currentTextChanged(QString); 2. QComboxBox::addItems(QStringList)
----> QComboxBox::insertItems(int, QStringList)
----> QStandardItem::insertRows(int, QList<QStandardItem*>)
----> QStandardItemPrivate::insertRows(int, QList<QStandardItem*>)
----> rowsAboutToBeInserted(QStandardItem *, int , int)
----> QAbstractItemModel::beginInsertRows(const QModelIndex &, int , int )
----> signal: rowsAboutToBeInserted(const QModelIndex &, int , int ) slot:
----> QAbstractItemModelPrivate::rowsAboutToBeInserted(const QModelIndex &, int , int )
----> QStandardItemModelPrivate::rowsInserted(QStandardItem *, int , int )
----> QAbstractItemModel::endInsertRows()
----> void QAbstractItemModelPrivate::rowsInserted(const QModelIndex &, int , int )
----> signal: QAbstractItemModel::rowsInserted(QModelIndex,int,int) slot: QComboBox::_q_rowsInserted(QModelIndex,int,int)
  1. 首先,我从最简单的来入手——addItem



    在上面这张图里可以知道,addItem调用的是insertItem这个函数,这个是用来插入条目的一个函数;

然后就是insertItem这个函数,我们可以看到,这个函数会根据你的是不是原始的QStandardItemModel,是的话,就会去设置数据;



这里有两个分支,

  • setData





    随着函数的调用过程,信号dataChanged被发射了,同时,在qcombobox.cpp中有对这个信号的连接,



    我们进到这个_q_dataChanged()函数里面,



    这里有一段代码:
 if (currentIndex.row() >= topLeft.row() && currentIndex.row() <= bottomRight.row()) {
const QString text = q->itemText(currentIndex.row());
if (lineEdit) {
lineEdit->setText(text);
updateLineEditGeometry();
} else {
emit q->currentTextChanged(text);
}
q->update();
#ifndef QT_NO_ACCESSIBILITY
QAccessibleValueChangeEvent event(q, text);
QAccessible::updateAccessibility(&event);
#endif
}

在这里,我们就找到了我们的目标currentTextChanged这个信号。但是发射这个信号的前提条件是:

  • currentIndex.row() >= topLeft.row() && currentIndex.row() <= bottomRight.row()也就是说,当前的下标的值需要在范围内
  • 当前的状态必须是不可编辑状态才会发射信号

所以这里就是设置成可编辑状态后,不会触发信号的原因;

  • insertRow





    现在关键的函数要来了,这个函数bool QStandardItemPrivate::insertRows,在待会addItems这个函数分析时也会用到。





    在这里发射了这个rowInserted()信号,这个信号,又在QComboBox中进行了槽函数的连接





    所以在这个函数里面,如果是插入的第一个条目,就会把当前的下表设置成0,这时候就会触发另外一个信号currentIndexChanged



    至此,我们就能明白,为什么addItem会触发currentTextChanged的信号。同时,如果设置成可编辑状态,又是为何不会触发currentTextChanged
  1. 其次,我们从第二个函数,也就是addItems





    到这里,就能发现,这个部分调用的还是这个bool QStandardItemPrivate::insertRows,同样根据条件判断,currentIndex = 0而其他两个分别为1和添加条目的数量,很显然不符合要求。

    所以这也就是为什么addItems不会触发currentIndexChange的原因。

    至此,根据源码的分析,所有发生的事情,都能够正常的解释通了。

Qt QComboBox之setEditable和currentTextChanged及其源码分析的更多相关文章

  1. hadoop之hdfs------------------FileSystem及其源码分析

    FileSystem及其源码分析 FileSystem这个抽象类提供了丰富的方法用于对文件系统的操作,包括上传.下载.删除.创建等.这里多说的文件系统通常指的是HDFS(DistributedFile ...

  2. [Qt] 《开发指南》samp4.1 源码分析

    界面: 功能: 输入单价和数量,计算总价:进制转换 控件: Qlabel QLineEdit QPushButton 文件依赖关系图(depend on): main.cpp:程序入口 widget. ...

  3. LinkedHashMap及其源码分析

    以下内容基于jdk1.7.0_79源码: 什么是LinkedHashMap 继承自HashMap,一个有序的Map接口实现,这里的有序指的是元素可以按插入顺序或访问顺序排列: LinkedHashMa ...

  4. CoordinatorLayout自定义Bahavior特效及其源码分析

    @[CoordinatorLayout, Bahavior] CoordinatorLayout是android support design包中可以算是最重要的一个东西,运用它可以做出一些不错的特效 ...

  5. String,StringBuffer,StringBuilder的区别及其源码分析

    String,StringBuffer,StringBuilder的区别这个问题几乎是面试必问的题,这里做了一些总结: 1.先来分析一下这三个类之间的关系 乍一看它们都是用于处理字符串的java类,而 ...

  6. 经典的HTML5游戏及其源码分析

    HTML5已经相当强大,在HTML5平台上,我们可以完成很多非常复杂的动画效果,包括游戏在内.早期我们只能利用flash来实现网络游戏,现在我们又多了一种选择,即用HTML5制作游戏.相比flash, ...

  7. [Java] I/O底层原理之一:字符流、字节流及其源码分析

    关于 I/O 的类可以分为四种: 关于字节的操作:InputStream 和 OutPutStream: 关于字符的操作:Writer 和 Reader: 关于磁盘的操作:File: 关于网络的操作: ...

  8. pdfmake.js使用及其源码分析

    公司项目在需要将页面的文本导出成DPF,和支持打印时,一直没有做过这样的功能,花了一点时间将其做了出来,并且本着开源的思想和技术分享的目的,将自己的编码经验分享给大家,希望对大家有用. 现在是有一个文 ...

  9. 8.深入k8s:资源控制Qos和eviction及其源码分析

    转载请声明出处哦~,本篇文章发布于luozhiyun的博客:https://www.luozhiyun.com,源码版本是1.19 又是一个周末,可以愉快的坐下来静静的品味一段源码,这一篇涉及到资源的 ...

随机推荐

  1. keepalived yum安装后启动报错解决

    [root@centos8 ~]yum install keepalived -y [root@centos8 ~]systemctl start keepalived.services [root@ ...

  2. CentOS 8 关闭 Firewalld 及 SELinux

    检查 SELinux 是否开启 执行 sestatus 指令可以检视目前 SELinux 的状态, 其中一项是是否有开启, 执行以下指令: # sestatus | grep status 如果看到 ...

  3. SpringBoot学习入门之Hello项目的构建、单元测试和热部署等(配图文,配置信息详解,附案例源码)

    前言: 本文章主要是个人在学习SpringBoot框架时做的一些准备,参考老师讲解进行完善对SpringBoot构建简单项目的学习汇集成本篇文章,作为自己对SpringBoot框架的总结与笔记. 你将 ...

  4. 面试官:Redis中的缓冲区了解吗

    hello 大家好,我是七淅(xī). Redis 大家肯定不陌生,但在使用层面看不到的地方,就容易被忽略.今天想和大家分享的内容是 Redis 各个缓冲区的作用.溢出的后果及优化方向. 在开始正文前 ...

  5. Linux企业常用命令详解

    cat :查看 cat [-AbeEnstTuv] [--help] [--version] fileName 常用参数: -n :由 1 开始对所有输出的行数编号 -b :和 -n 相似,对于空白行 ...

  6. AQS 详解之共享锁模式

    概括 AQS框架数据结构是一个先进先出的双向队列,当多个线程进行竞争资源时,那些竞争失败的线程会加入到队列中.他向上层提供了很多接口,其中一个是acquireShared获取共享模式的接口.本文将会根 ...

  7. kafka 第一次小整理(草稿篇)————分发的基本思路[三]

    前言 简单整理一下分发的基本思路. 正文 kafka 原本是做日志管理系统,主要是分发这块的. 那么如何做分发呢? 分发的是什么呢? 分发的其实是日志,日志是事件状态,kafka 内部就叫做Recor ...

  8. ADT环境搭建手册

    前言 笔者在搭建ADT环境之前一脸懵逼,甚至不知道ADT是什么,更别说与之相关的SDK.eclipse等,相信很多小伙伴跟我一样也是一脸茫然,所以在搭建环境之前有必要先了解一下它们是什么,有什么样的关 ...

  9. 解释一下numa

    NUMA : 非一致性存储 当多个处理器访问同一个存储器时,会有性能损失,NUMA通过提供分离的存储器给各个处理器. NUMA系统的结点通常是由一组CPU和本地内存组成,有的结点可能还有I/O子系统. ...

  10. 为什么要从Web form过渡到MVC中

    可以说,在未来几年中,Web form的使用会逐渐减少,而取而代之的就是MVC.可能你不会同意我的观点,那么我就试着阐述一下我的观点,如果你还是不能接受,那么请你反驳我. 学习一个新语言或者是新架构是 ...