QLineEdit设置数据的输入范围QIntValidator和QDoubleValidator

在日常开发过程中QLineEdit作为输入框,有时要限制输入的内容,比哪,考试分数为1-100,这个时候就使用QIntValidator作为限制范围,而如何输入的是带小数的呢,那么使用QDoubleValidator可以吗,下面请看具体的示例。

1.限制输入的最初使用的是正则表达式

QRegExp regIntExp("^(([1-9][0-9]|[1-9])|100)$"); 
QValidator *regIntValidator = new QRegExpValidator(regIntExp, ui->lineEdit);
ui->lineEdit->setValidator(regIntValidator);

 这个确实比较精确,但这有一个问题,就是范围变化,每次都要重写正则表达式,用起来比较不友好。那么有没有一个比较好的一次设置都通用呢,下面介绍QIntValidator就能达到这种效果,

例如:

//1-100整数
QIntValidator *intValidator = new QIntValidator(1, 100, this);
intValidator->setRange(1, 100);
ui->lineEdit->setValidator(intValidator);

这也有两个问题,一是输入多个0都接受了,二是最大范围是3位数可以输到999,如果设置1-99那最大值就是99这没问题,但如果设置为100.输入范围就到999了,那么如果才能消除这两个顶端限制的问题了,这个时候就只能自定义定QIntValidator了。

QIntValidator有两个函数

 virtual void fixup(QString &input) const override
 virtual QValidator::State validate(QString &input, int &pos) const override

重写这两个函数,

//validate()是验证的过程,返回结果是State。每当输入有变化时调用此方法
enum State {
    Invalid,        //验证通不过
   Intermediate,   //输入未完成,不确定是否能通过验证

    Acceptable      //验证通过
}

fixup是修复函数,当小于最小值时修复为最小值,例如输入0,自动修复为1

#ifndef CUSTOMINTVALIDATOR_H
#define CUSTOMINTVALIDATOR_H

#include <QIntValidator>

class CustomIntValidator : public QIntValidator
{
    Q_OBJECT
public:
    explicit CustomIntValidator(QObject *parent = nullptr);

    CustomIntValidator(int bottom, int top, QObject *parent);
    void setRange(int bottom, int top) override;
    virtual State validate(QString &s, int &n) const override;
    virtual void fixup(QString &input) const override;

};

#endif // CUSTOMINTVALIDATOR_H

#include "customintvalidator.h"
#include <QDebug>

CustomIntValidator::CustomIntValidator(QObject *parent)
    : QIntValidator{parent}
{

}

CustomIntValidator::CustomIntValidator(int bottom, int top, QObject *parent)
    :QIntValidator(bottom, top, parent)
{

}

void CustomIntValidator::setRange(int bottom, int top)
{
    QIntValidator::setRange(bottom, top);
}
//validate()是验证的过程,返回结果是State。每当输入有变化时调用此方法
//enum State {
//    Invalid,        //验证通不过
//    Intermediate,   //输入未完成,不确定是否能通过验证
//    Acceptable      //验证通过
//}
QValidator::State CustomIntValidator::validate(QString &s, int &n) const
{
    if (s.isEmpty())
    {
        return QValidator::Intermediate;
    }
    //这里判断超过位数了,变成无效的参数
    int intValue = s.toInt();
    if(intValue > top() || intValue < bottom())
        return QIntValidator::Invalid;

    return QIntValidator::validate(s, n);
}
//当验证通不过时,通过调用 QValidator::fixedup()是这个函数修复错误。
void CustomIntValidator::fixup(QString &input) const
{
    qDebug() << "fixup============input==============" << input;
    //这里要做个判断,如果超过最大值取最大值 ,低于最小值取最小值,例如1转换为20, 333转换为100
    int intValue = input.toInt();
    if(intValue < bottom())
        input = QString("%1").arg(bottom());
}

运行结果:

这种自定义的数据范围调整都不会影响效果,比如你调成20~300,那么他的范围就是在20到300之内,比正则表达式友好些。

下面看浮点数的使用

1.用正则表达式来实现

QRegExp regExp("^(([1-9][0-9]|[1-9])(\\.\\d{1,2})?|0\\.\\d{1,2}|100)$");
QValidator *validator = new QRegExpValidator(regExp, ui->lineEdit_3);
ui->lineEdit_3->setValidator(validator);

这个比较精准,就是每次范围变化,正则表达式都要重新按规则写。

下面使用QDoubleValidator来限制输入范围。

// 0.02-100.0  大于0的浮点数
QDoubleValidator *doubleValidator = new QDoubleValidator(this);
doubleValidator->setRange(0.02, 100.0, 2);
ui->lineEdit_5->setValidator(doubleValidator);

运行结果:

这种怎么是无限制输入呢。

但是仔细一想是Qt的Bug吗,网上查了下也有人遇到说是一个Bug,后来查看帮助说明

 一个是标准计数法,一个是科学计数法,默认是科学计数法,所以可以输入任何内容,

如果换成科学计数法呢

// 0.02-100.0  大于0的浮点数
QDoubleValidator *doubleValidator = new QDoubleValidator(this);
doubleValidator->setRange(0.02, 100.0, 2);
doubleValidator->setNotation(QDoubleValidator::StandardNotation);
ui->lineEdit_5->setValidator(doubleValidator);

运行效果:

但这有一个问题,就是最大值为100的时候输入999也可以,看来最大范围就是输入的位数

那么要消除字个顶端值越界的问题,只能使用自定义的方式,与QIntValidator一样,重写QDoubleValidator的两个函数validate和fixup,下面看代码实现:

#ifndef CUSTOMDOUBLEVALIDATOR_H
#define CUSTOMDOUBLEVALIDATOR_H

#include <QDoubleValidator>

class CustomDoubleValidator : public QDoubleValidator
{
    Q_OBJECT
public:
    explicit CustomDoubleValidator(QObject *parent = nullptr);

    CustomDoubleValidator(double bottom, double top,  int decimals = 0, QObject *parent = nullptr);
    void setRange(double bottom, double top, int decimals = 0) override;
    virtual State validate(QString &s, int &n) const override;
    virtual void fixup(QString &input) const override;

};

#endif // CUSTOMDOUBLEVALIDATOR_H

#include "customdoublevalidator.h"
#include <QDebug>

CustomDoubleValidator::CustomDoubleValidator(QObject *parent)
    : QDoubleValidator{parent}
{

}

CustomDoubleValidator::CustomDoubleValidator(double bottom, double top, int decimals, QObject *parent)
    :QDoubleValidator(bottom, top, decimals, parent)
{

}

void CustomDoubleValidator::setRange(double bottom, double top, int decimals)
{
    QDoubleValidator::setRange(bottom, top, decimals);
}
//validate()是验证的过程,返回结果是State。每当输入有变化时调用此方法
//enum State {
//    Invalid,        //验证通不过
//    Intermediate,   //输入未完成,不确定是否能通过验证
//    Acceptable      //验证通过
//}
QValidator::State CustomDoubleValidator::validate(QString &s, int &n) const
{
    //这里判断超过位数了,"0" , "0." , "0.0"都表示输入未完成
    if (s.isEmpty() || s == "0" || s == "0." || s == "0.0")
    {
        return QValidator::Intermediate;
    }

    int dotPos = s.indexOf(".");
    if (dotPos > 0)
    {
        int offset = s.length() - dotPos - 1;
        if (s.right(offset).length() > decimals())
        {
            return QValidator::Invalid;
        }
    }

    double value = s.toDouble();
    if(value > top() || value < bottom()) {
        return QValidator::Invalid;
    }

    return QValidator::Acceptable;
}
//当验证通不过时,通过调用 QValidator::fixedup()是这个函数修复错误。
void CustomDoubleValidator::fixup(QString &input) const
{
    qDebug() << "fixup============input==============" << input;
    //这里要做个判断,低于最小值取最小值,例如0.01转换为0.02,
    double value = input.toDouble();
    if(value < bottom())
        input = QString("%1").arg(bottom());
}

 

完整代码如下:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>



QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();
    void initView();

private:
    Ui::MainWindow *ui;
};
#endif // MAINWINDOW_H
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QRegExpValidator>
#include <QIntValidator>
#include <QDoubleValidator>
#include <QDebug>
#include "customintvalidator.h"
#include "customdoublevalidator.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    initView();
}

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

void MainWindow::initView()
{
    QRegExp regIntExp("^(([1-9][0-9]|[1-9])|100)$");
    QValidator *regIntValidator = new QRegExpValidator(regIntExp, ui->lineEdit);
    ui->lineEdit->setValidator(regIntValidator);
    //1-100整数
    QIntValidator *intValidator = new QIntValidator(1, 100, this);
    intValidator->setRange(1, 100);
    ui->lineEdit_2->setValidator(intValidator);

    //1-100整数
    CustomIntValidator *customIntValidator = new CustomIntValidator(1, 100, this);
    customIntValidator->setRange(1, 100);
    ui->lineEdit_3->setValidator(customIntValidator);

    // 0.01-100.0  大于0的浮点数
    QRegExp regExp("^(([1-9][0-9]|[1-9])(\\.\\d{1,2})?|0\\.\\d{1,2}|100)$");
    QValidator *validator = new QRegExpValidator(regExp, ui->lineEdit_4);
    ui->lineEdit_4->setValidator(validator);

    // 0.02-100.0  大于0的浮点数
    QDoubleValidator *doubleValidator = new QDoubleValidator(this);
    doubleValidator->setRange(0.02, 100.0, 2);
    doubleValidator->setNotation(QDoubleValidator::StandardNotation);
    ui->lineEdit_5->setValidator(doubleValidator);

    CustomDoubleValidator *customDoubleValidator = new CustomDoubleValidator(this);
    customDoubleValidator->setRange(0.02, 100.0, 2);
    customDoubleValidator->setNotation(QDoubleValidator::StandardNotation);
    ui->lineEdit_6->setValidator(customDoubleValidator);

}

运行结果:

参考:
https://blog.csdn.net/Joker__123/article/details/125517029
https://blog.csdn.net/u011391361/article/details/132101193
https://blog.csdn.net/whik1194/article/details/93504971

  • 20
    点赞
  • 31
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值