QT QLineEdit失去焦点事件问题与解决

本文介绍如何获得QLineEdit的失去焦点事件和获得焦点的输入框也会触发失去焦点事件的问题!

目录

一、QLineEdit获得失去焦点事件

1.自定义类继承自QLineEdit

2.重写 focusOutEvent

3.使用

二、失去焦点事件问题

1.问题描述

2.问题解决

三、源码分享

lineeditfocus.h

lineeditfocus.cpp

widget.cpp


一、QLineEdit获得失去焦点事件

1.自定义类继承自QLineEdit

class LineEditFocus : public QLineEdit {
    Q_OBJECT
public:
    explicit LineEditFocus(QWidget *parent = nullptr) { }
    ~LineEditFocus() override { }
}

2.重写 focusOutEvent

protected:
    // 焦点离开
    void focusOutEvent(QFocusEvent *event) override { }
    // 获得焦点
    void focusInEvent(QFocusEvent *event) override { }

当获得焦点时,focusInEvent方法会被触发;当失去焦点时,focusOutEvent方法会被触发;

然后就可以在方法内部做一些我们的需求处理,例如可以通过信号与槽通知主程序等;

3.使用

直接使用我们自定义的类,创建LineEditFocus对象即可;

如果使用的是ui布局中的部件,那么可以将部件提升为我们自定义的LineEditFocus即可;

二、失去焦点事件问题

1.问题描述

如果有多个输入框部件,且当前输入框部件失去焦点,且另一个获得焦点的部件不是输入框时,那么是没有问题的!

如果当前输入框失去焦点,且另一个获得焦点的部件也是输入框,那么这样就会出现问题;

会优先触发另一个输入框的失去焦点事件,然后才会触发当前输入框的失去焦点事件,最后再触发另一个输入框的获得焦点事件!!!

案例:

(1).继续在自定义类中添加信号 ;可以根据个人需求传输参数值,例如可以将当前输入框的文本传送;

signals:
    void signalLoseFocus(int index);

(2).然后在失去焦点方法中,触发此信号

void LineEditFocus::focusOutEvent(QFocusEvent *event)
{
    static int index = 1;
    emit signalLoseFocus(index++);
    QLineEdit::focusOutEvent(event);
}

(3).最后在主窗体中使用即可(ui部件,提升即可)

void Widget::init()
{
    LineEditFocus *le1 = new LineEditFocus(this);
    LineEditFocus *le2 = new LineEditFocus(this);
    LineEditFocus *le3 = new LineEditFocus(this);

    QList<LineEditFocus *> list;
    list << le1 << le2 << le3;

    foreach (LineEditFocus *lef, list) {
        lef->setFixedSize(350, 50);
        connect(lef, &LineEditFocus::signalLoseFocus, this, &Widget::onLeaveFocus);
    }


    QVBoxLayout *layout = new QVBoxLayout(this);
    layout->addWidget(le1);
    layout->addWidget(le2);
    layout->addWidget(le3);
    this->setLayout(layout);
}

(4).在槽函数中将接收到的index,通过messagebox提示出来

void Widget::onLeaveFocus(int index)
{
    QMessageBox::information(this, "提示", QString("输入框焦点离开:%1").arg(index));
}

(5).运行效果

可以看出,确实是有问题的!

2.问题解决

具体是什么原因导致出现这样的问题,我也没搞明白;

但是办法总比困难多,我们转换一下思路去解决他;

(1).增加变量用于标志是否获取到了焦点

private:
    // 焦点获得标志
    bool m_focus;

(2).焦点获得,获得焦点时,m_focus赋值true

void LineEditFocus::focusInEvent(QFocusEvent *event)
{
    m_focus = true;    // 标志当前编辑框已经获得焦点
    QLineEdit::focusInEvent(event);
}

(3).失去焦点,通过m_focus变量辅助配合判断

void LineEditFocus::focusOutEvent(QFocusEvent *event)
{
    if (m_focus && event->lostFocus()) {    // event->lostFocus(): type() == FocusOut
        m_focus = false;    // 焦点失去
        emit signalLoseFocus(m_index);
    }

    QLineEdit::focusOutEvent(event);
}

通过上面的测试可知,当从第一个输入框点击第二个输入框时,优先触发第二个输入框的失去焦点事件,此时,第二个输入框是还没有获得焦点的,即m_focus变量值为false,信号就没法触发;

紧接着第一个输入框的失去焦点事件触发,因为先前已经获得了焦点,即m_focus变量值为true,所以第一个输入框的失去焦点事件可以正常发射信号;

最后才会触发第二个输入框的获得焦点事件。

(3).运行测试

问题完美解决!

三、源码分享

lineeditfocus.h

#ifndef LINEEDIT_FOCUS_H
#define LINEEDIT_FOCUS_H

#include <QLineEdit>

class LineEditFocus : public QLineEdit {
    Q_OBJECT
public:
    explicit LineEditFocus(QWidget *parent = nullptr);
    ~LineEditFocus() override;

    void SetIndex(int index);
    int GetIndex() const;


signals:
    void signalLoseFocus(int index);
    void signalInFocus(int index);

protected:
    // 焦点离开
    void focusOutEvent(QFocusEvent *event) override;
    // 获得焦点
    void focusInEvent(QFocusEvent *event) override;

private:
    // 记录标志
    int m_index;

    // 焦点获得标志
    bool m_focus;
};

#endif // LINEEDIT_FOCUS_H

lineeditfocus.cpp

#include "lineeditfocus.h"

#include <QFocusEvent>

LineEditFocus::LineEditFocus(QWidget *parent) : QLineEdit(parent)
{
    m_index = 0;
    m_focus = false;
}

LineEditFocus::~LineEditFocus()
{

}

void LineEditFocus::SetIndex(int index)
{
    m_index = index;
}

void LineEditFocus::focusOutEvent(QFocusEvent *event)
{
    if (m_focus && event->lostFocus()) {    // event->lostFocus() --> return type() == FocusOut;
        m_focus = false;    // 焦点失去
        emit signalLoseFocus(m_index);
    }

    QLineEdit::focusOutEvent(event);
}

void LineEditFocus::focusInEvent(QFocusEvent *event)
{
    m_focus = true; // 焦点获得
    QLineEdit::focusInEvent(event);
}

widget.cpp

#include "widget.h"
#include "ui_widget.h"

#include "lineeditfocus.h"

#include <QMessageBox>

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);

    init();
}

Widget::~Widget()
{
    delete ui;
}

void Widget::init()
{
    LineEditFocus *le1 = new LineEditFocus(this);
//    le1->setFixedSize(350, 50);
//    le1->SetIndex(1);
    LineEditFocus *le2 = new LineEditFocus(this);
//    le2->setFixedSize(350, 50);
//    le2->SetIndex(2);
    LineEditFocus *le3 = new LineEditFocus(this);
//    le3->setFixedSize(350, 50);
//    le3->SetIndex(3);


    QList<LineEditFocus *> list;
    list << le1 << le2 << le3;

    int index = 1;
    foreach (LineEditFocus *lef, list) {
        lef->setFixedSize(350, 50);
        lef->SetIndex(index++);
        connect(lef, &LineEditFocus::signalLoseFocus, this, &Widget::onLeaveFocus);
    }


    QVBoxLayout *layout = new QVBoxLayout(this);
    layout->addWidget(le1);
    layout->addWidget(le2);
    layout->addWidget(le3);
    this->setLayout(layout);


}


void Widget::onLeaveFocus(int index)
{
    QMessageBox::information(this, "提示", QString("输入框焦点离开:%1").arg(index));
}

完!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/916596.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

vscode执行npm install报错

npm install一直提示报错 以管理员身份运行vscode&#xff0c;如果每次觉得很麻烦可以做如下修改&#xff1a;

【算法】树状数组

前言 众所周知&#xff0c;通过前缀和&#xff0c;我们可以很快的在一个很大的数组中求出区间和&#xff0c;但是如果想要去修改数组中的一个数的值&#xff0c;前缀和就无法实现。所以来学习一个新的数据结构&#xff1a;树状数组 &#xff08;文章中关于树状数组的截图来自于…

Java项目实战II基于微信小程序的私家车位共享系统(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 在城市化进…

ZeroSSL HTTPS SSL证书ACMESSL申请3个月证书

目录 一、引言 二、准备工作 三、申请 SSL 证书 四、证书选型 五、ssl重要性 一、引言 目前免费 Lets Encrypt、ZeroSSL、BuyPass、Google Public CA SSL 证书&#xff0c;一般免费3-6个月。从申请难易程度分析&#xff0c;zerossl申请相对快速和简单&#xff0c;亲测速度非…

pipx安装提示找不到包

执行&#xff1a; pipx install --include-deps --force "ansible6.*"WARNING: Retrying (Retry(total4, connectNone, readNone, redirectNone, statusNone)) after connection broken by NewConnectionError(<pip._vendor.urllib3.connection.HTTPSConnection …

react + ts定义接口类型写法

接口&#xff08;未进行ts定义&#xff09; export async function UserList(params: {// keyword?: string;current?: number;pageSize?: number;},// options?: { [key: string]: any }, ) {return request<API1.UserList>(http://geek.itheima.net/v1_0/mp/artic…

Golang超详细入门教程

Golang超详细入门教程 部分图片可能加载不出来&#xff0c;所以这里我上传到了自己的个人网站上也可以查看&#xff1a;http://dahua.bloggo.chat/testimonials/490.html 一、数据类型转换 C语言中数据可以隐式转换或显示转换, 但是Go语言中数据只能显示转换格式: 数据类型(…

Cannot resolve org.apache.tomcat.embed:tomcat-embed-core:9.0.60标红解决办法

解决方法是&#xff1a; MyBatis 会扫描这个包下的所有接口&#xff0c;并将这些接口注册为 MyBatis 的 Mapper。 把这个加上后&#xff0c;问题解决&#xff01;

游戏引擎学习第九天

视频参考:https://www.bilibili.com/video/BV1ouUPYAErK/ 修改之前的方波数据&#xff0c;改播放正弦波 下面主要讲关于浮点数 1. char&#xff08;字符类型&#xff09; 大小&#xff1a;1 字节&#xff08;8 位&#xff09;表示方式&#xff1a;char 存储的是一个字符的 A…

C#设计模式(12)——享元模式(Flyweight Pattern)

前言 享元模式通过共享对象来减少内存使用和提高性能。 代码 public abstract class Flyweight {public abstract void Control(); }public class ComputerFlyweight : Flyweight {private string _operator;public ComputerFlyweight(string name){_operator name;}public o…

upload-labs通关练习

目录 环境搭建 第一关 第二关 第三关 第四关 第五关 第六关 第七关 第八关 第九关 第十关 第十一关 第十二关 第十三关 第十四关 第十五关 第十六关 第十七关 第十八关 第十九关 第二十关 总结 环境搭建 upload-labs是一个使用php语言编写的&#xff0c…

【云原生系列--Longhorn的部署】

Longhorn部署手册 1.部署longhorn longhorn架构图&#xff1a; 1.1部署环境要求 kubernetes版本要大于v1.21 每个节点都必须装open-iscsi &#xff0c;Longhorn依赖于 iscsiadm主机为 Kubernetes 提供持久卷。 apt-get install -y open-iscsiRWX 支持要求每个节点都安装 N…

Elastic Agent:可灵活地在任何地方发送和处理任何数据

作者&#xff1a;来自 Elastic Nima Rezainia Elastic Agent 是一款功能强大且用途广泛的工具&#xff0c;可用于从各种数据源&#xff08;包括自定义用户应用程序&#xff09;收集日志和指标。现在&#xff0c;Elastic Agent 提供了无与伦比的灵活性&#xff0c;可以将数据准确…

一、HTML

一、基础概念 1、浏览器相关知识 这五个浏览器市场份额都非常大&#xff0c;且都有自己的内核。 什么是内核&#xff1a; 内核是浏览器的核心&#xff0c;用于处理浏览器所得到的各种资源。 例如&#xff0c;服务器发送图片、视频、音频的资源&#xff0c;浏览…

IDEA旗舰版编辑器器快速⼊门(笔记)

简介&#xff1a;javaweb开发必备软件之IDEA期间版介绍 DEA编辑器器版本介绍 官⽹网&#xff1a;https://www.jetbrains.com/地址&#xff1a;https://www.jetbrains.com/idea/download/#sectionmac DEA 分社区版(Community) 和 旗舰版(Ultimate)&#xff0c;我们做JavaWeb开…

【鸿蒙开发】第十五章 H5与端侧交互、Cookies以及Web调试

目录 1. H5与端侧交互 1.1 应用侧调用前端页面函数 1.2 前端页面调用应用侧函数 1.2.1 复杂类型使用方法 1.3 建立应用侧与前端页面数据通道 2 管理页面跳转及浏览记录导航 2.1 历史记录导航 2.2 页面跳转 2.3 跨应用跳转 3 管理Cookie及数据存储 3.1 Cookie管理 3…

【优选算法】双指针

目录 一、[移动零](https://leetcode.cn/problems/move-zeroes/description/)二、[复写零](https://leetcode.cn/problems/duplicate-zeros/description/)三、[快乐数](https://leetcode.cn/problems/happy-number/)四、 [盛最多水的容器](https://leetcode.cn/problems/contai…

C++常用的特性-->day05

友元的拓展语法 声明一个类为另外一个类的友元时&#xff0c;不再需要使用class关键字&#xff0c;并且还可以使用类的别名&#xff08;使用 typedef 或者 using 定义&#xff09;。 #include <iostream> using namespace std;// 类声明 class Tom; // 定义别名 using …

Vue3 + Vite 构建组件库的整体流程

Vue3 Vite 构建组件库的流程 本文教你如何用 Vue Vite&#xff0c;一步一步构建一个组件库并发布到 npm 的整体流程 1. 通过 vite 命令创建一个基本的项目结构&#xff08;这里选用 vue ts 的项目&#xff09; npm create vitelatest2. 在项目中创建一个 lib 目录&#xf…

Ubuntu22.04.2 k8s部署

k8s介绍 简单介绍 通俗易懂的解释&#xff1a; Kubernetes&#xff08;也被称为 K8s&#xff09;就像是一个大管家&#xff0c;帮你管理你的云计算服务。想象一下&#xff0c;你有很多个小程序&#xff08;我们称之为“容器”&#xff09;&#xff0c;每个都在做不同的事情&…