百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术教程 > 正文

从零开始学Qt(50):进阶!自定义用户名密码登录窗口

csdh11 2025-03-05 15:01 26 浏览

一般的大型应程序在启动时会显示一个启动画面,即Splash窗口。Splash窗口是一个无边对话框,一般显示一个图片,展示软件的信息。Splash窗口显示时,程序在后台做一些比较 耗时的启动准备工作,Splash窗口显示一段时间后自动关闭,然后软件的主窗口显示出来。Qt有一个QSplashScreen类可以实现Splash窗口的功能,它提供了载入图片,自动设置窗口无边框效果等功能。

有的应用程序还有软件登录界面,要求用户输入用户名和密码才可以进入软件。

Splash窗口和登录界面实质都是对话框,它们在程序启动时显示。本节设计的实例是一个 Splash登录对话框,这个对话框结合了 Splash 窗口和登录界面两者的功能,示例运行时的启动界面如图所示。

这个实例演示如下的一些功能的实现方法:

  • 如何实现Splash特点的无边框对话框:
  • 如何设计用鼠标拖动无边框的对话框;
  • 如何使用QSettings类存储用户名、密码等信息;
  • 如何用QCryptographicHash类进行字符串加密;
  • 如何根据登录输入状况确定启动主窗口或终止程序运行。

对话框界面设计和类定义

采用新建Qt Designer Form Class的方法创建启动登录对话框,它从QDialog继承而来,设置类名称为QDlgLogin。界面设计在UI设计器里进行,主要区域是一个用于显示图片的QLabel组件,由文件载入图片,为QLabel组件的pixmap指定图片。

对话框下方是用于用户名和密码输入的QLineEdit组件,两个按钮用于选择用户输入,设置 “取消”按钮的clicked()信号与对话框的reject()槽函数关联。但是“确定”按钮的clicked()信号不要设置为与对话框的任何槽函数关联,需要对其编写自定义的槽函数代码,因为需要根据用户输入确定对话框返回结果。为对话框界面上的组件设置好布局。

下面是dlglogin.h文件中DlgLogin类的定义:

class DlgLogin : public QDialog
{
	Q_OBJECT
private:
  bool m_moving=false; //表窗口是否在鼠标操作下移动
  QPoint m_lastPos; //上一次的鼠标位置
  QString m_user="user"; //初始化用户名
  QString m_pswd="12345";//初始化密码,未加密的
  int m_tryCount=0; //试错次数
  void readSettings(); //读取设置,注册表
  void writeSettings(); //写入设置,注册表
  QString encrypt(const QString& str); //字符串加密
protected:
  //用于鼠标拖动窗口的鼠标事件
  void mousePressEvent(QMouseEvent * event);
  void mouseMoveEvent(QMouseEvent * event);
  void mouseReleaseEvent(QMouseEvent *event);
public:
  explicit DlgLogin(QWidget *parent = nullptr);
  ~DlgLogin();
private slots:
	void on_btnOk_clicked();
private:
	Ui::DlgLogin *ui;
};

在DlgLogin类中,定义了一些私有成员变量。

  • m_moving和m_lastPos用于在拖动窗口时记录移动状态和上次的位置,由于Splash窗口 没有标题栏,所以采用在图片上拖拉的方式移动窗口,使用了3个鼠标事件来实现窗口拖 动操作。
  • m_user, m_pswd, mtryCount用于记录用户名、密码和试错次数。
  • readSettings()用于读取存储的设置,writeSettings()用于将设置存储,在Windows系统下, 这些信息是存储在注册表里的。
  • encrypt()函数用于对—个字符串进行加密。

DIgLogin类功能实现

(1)构造函数里的初始化

DIgLogin类的构造函数代码如下:

DlgLogin::DlgLogin(QWidget *parent) :
QDialog(parent), ui(new Ui::DlgLogin)
{
  ui->setupUi(this);
  ui->edtPswd->setEchoMode(QLineEdit::Password);//设置为密码输入模式
  this->setAttribute(Qt::WA_DeleteOnClose); //设置为关闭时删除
  this->setWindowFlags(Qt::SplashScreen); //设置为 SplashScreen
  // this->setWindowFlags(Qt::FramelessWindowHint); //无边框,但在任务栏显示标题
  readSettings(); //读取存储的用户名和密码
}

QLineEdit::setEchoMode()函数设置编辑框回显方式,参数为QLineEdit::EchoMode枚举类型, 这里设置为QLineEdit::Password回显方式,用于将密码输入回显为一个符号,而不显示真实字符,使用setWndowFlags()函数将窗口标志设置为Qt::SplashScreen,这样对话框显示为Splash窗口,无边框,且在Windows任务栏上没有显示。另外一个类似的标志是Qt::FramelessWindowHint, 它会使对话框无边框,但是会在任务栏上显示对话框的标题。

初始设置后调用readSettings()函数读取存储的设置,根据存储的情况将用户名显示到窗口上的编辑框里。

(2)应用程序设置的存储

自定义成员函数readSettings()用于读取应用程序设置,writeSettings()用于保存设置,实现代码如下:

void DlgLogin::readSettings()
{
  //读取存储的用户名和密码,密码是经过加密的
  QString organization="FutureSoft"; //用于注册表,
  QString appName="Demo-Login";
  QSettings settings(organization, appName);
  bool saved=settings.value("saved",false).toBool() ;//读取 saved
  m_user=settings.value("Username", "user").toString();//读取 username
  QString defaultPSWD=encrypt("12345");//缺省密码"12345"加密后的数据
  m_pswd=settings.value("PSWD",defaultPSWD).toString() ;//读取 PSWD
  if (saved){
  	ui->edtUser->setText(m_user);
  }
  ui->chkBoxSave->setChecked(saved);
}

void DlgLogin::writeSettings()
{
  //保存用户名,密码等设置
  QSettings settings("FutureSoft","Demo-Login");//注册表键组
  settings.setValue("Username",m_user);//用户名
  settings.setValue ("PSWD", m_pswd); //密码,经过加密的
  settings.setValue ("saved", ui->chkBoxSave->isChecked());
}

应用程序的设置是指应用程序需要保存的—些信息,在Windows系统下,这些信息保存在注册表里。使用QSettings类可以实现设置信息的读取和写入。

创建QSettings对象时,需要传递organization和appName,例如:

QSettings settings("FutureRaider","Demo-Login");//注册表键组

指向的注册表目录是
HKEY_CURRENT_USER/Software/
FutureRaider/Demo-Login

注册表里参数是以“键—键值”对来保存的。writeSettings()函数里使用setValue()函数写入键值,readSettings()里使用value()函数读取键值。读取键值时可以指定缺省值,即如果键不存在,就用缺省值作为读取的值。

在Windows的开始菜单的输入框里输入regedit,打开注册表,査找到目录HKEY_CURRENT_ USER/Software/FutureRaider/Demo-Login,可以看到注册表里参数存储情况。其中,存储的密码是加密后的字符串。

(3)字符串加密

本实例中密码采用加密后的字符串保存,这样在实际应用中具有安全性。Qt提供了用于加密的类QCryptographicHash。自定义函数encrypt()就利用这个类进行字符串加密,实现代码如下:

QString DlgLogin::encrypt(const QString &str)
{
  //字符串MD5算法加密
  QByteArray btArray;
  btArray.append(str);
  QCryptographicHash hash(QCryptographicHash::Md5); //Md5加密算法
  hash.addData(btArray); // 添加数据
  QByteArray resultArray=hash.result();//返回最终的散列值
  QString md5=resultArray.toHex();//转换为 16 进制字符串
  return md5;
}

QCryptographicHash创建时需要指定一种加密算法,加密算法变量是枚举类型
QCryptographicHash::Algorithm,常用的常量值有 QCryptographicHash::Md4、QCryptographicHash::Md5、
QCryptographicHash::Sha512等,完整的描述可参考Qt的帮助文档。

QCryptographicHash只提供了加密功能,没有提供解密功能。

(4)用户名和密码输入判断

登录窗口运行后,单击“确定”按钮,程序会对输入内容进行判断。“确定”按钮的槽函数代码如下:

void DlgLogin::on_btnOk_clicked()
{//确定按钮
  QString user=ui->edtUser->text().trimmed();//输入用户名
  QString pswd=ui->edtPswd->text().trimmed(); //输入密码
  QString encrptPSWD=encrypt(pswd); //对输入密码进行加密
  if((user==m_user) && (encrptPSWD==m_pswd)){
  	writeSettings();
  	this->accept(); //对话框accept(),关闭对话框
  }else{
  	m_tryCount++; //错误次数
  	if(m_tryCount>3){
  		QMessageBox::critical(this,"错误","输入错误次数太多,强行退出");
  		this->reject(); //退出
  	}else{
  		QMessageBox::warning(this,"提示","用户名或密码错误");
  	}
  }
}

由于QCryptographicHash只提供了加密功能,没有提供解密功能,所以,在读取应用程序设定后,无法将加密后的密码解密并显示在窗口上,程序只能回显用户名,而不能回显密码。

这段程序会对输入的密码进行加密,因为从注册表读取的是加密后的密码,所以能够对比输入的用户名和密码与存储的用户名和密码是否匹配。

如果输入正确,调用窗口的accept()槽函数关闭对话框,对话框返回值为QDialog::Accepted, 否则试错次数加1;如果试错次数大于3次,就调用窗口的reject()槽函数关闭对话框,对话框返回值为QDialog::Rejected。

(5)窗口拖动功能的实现

由于Splash窗口没有边框,因此不能像普通的窗口那样通过拖动窗口的标题栏来拖动窗口。 为了实现窗口的拖动功能,对窗口的3个鼠标事件进行处理,实现的代码如下:

void DlgLogin::mousePressEvent(QMouseEvent *event)
{//鼠标按键被按下
  if(event->button()== Qt::LeftButton){
  	m_moving = true;
  	m_lastPos = event->globalPos()-pos(); //记录下鼠标相对于窗口的位置
  }
  return QDialog::mousePressEvent(event);
}

void DlgLogin::mouseMoveEvent(QMouseEvent *event)
{//鼠标按下左键移动
  if( m_moving && (event->buttons() && Qt::LeftButton) &&
  (event->globalPos()-m_lastPos).manhattanLength() > QApplication::startDragDistance() ){
  	move(event->globalPos()-m_lastPos);
  	m_lastPos=event->globalPos()-pos();
  }
  	return QDialog::mouseMoveEvent(event);
}

void DlgLogin::mouseReleaseEvent(QMouseEvent *event)
{//鼠标按键释放
	m_moving=false; //停止移动
}

mousePressEvent(QMouseEvent *event)事件在鼠标按键按下时发生,传递的参数event有鼠标按键和坐标信息,判断如果是鼠标左键按下,就设置变量m_moving值为true,表示开始移动,并记录下鼠标坐标。event->globalPos()与对话框的pos()是不同坐标系下的坐标,在绘图部分再详细介绍。

mouseMoveEvent(QMouseEvent *event)事件在鼠标移动时发射,程序里判断是否已经开始移动并且按下鼠标左键;如果是,则调用窗口move()函数,横向和纵向移动一定的距离,并再次记录坐标点。

mouseReleaseEvent(QMouseEvent *event)事件在鼠标按键释放时发生,左键释放时停止窗口移动。

所以,当在窗口上按下鼠标左键并移动时,窗口就会随之移动。

登录窗口的使用

设计好启动和登录窗口 DlgLogin之后,在main()函数里使用启动与登录对话框。main()函 数的代码如下:

int main(int argc, char *argv[])
{
  QApplication a(argc, argv);
  DlgLogin *login=new DlgLogin;
  if(login->exec()==QDialog::Accepted){
  	MainWindow w;
  	w.show();
  	return a.exec();
  }else{
  	return 0;
  }
}

在主窗口之前创建登录对话框对象login,并以模态显示的方式调用此对话框。如果对话框返回的是QDialog::Accepted,说明通过了用户名和密码验证,就创建主窗口并显示;否则退出应用程序。由于DlgLogin设置为关闭时删除,验证关闭登录窗口启,对象会自动删除。

相关推荐

探索Java项目中日志系统最佳实践:从入门到精通

探索Java项目中日志系统最佳实践:从入门到精通在现代软件开发中,日志系统如同一位默默无闻却至关重要的管家,它记录了程序运行中的各种事件,为我们排查问题、监控性能和优化系统提供了宝贵的依据。在Java...

用了这么多年的java日志框架,你真的弄懂了吗?

在项目开发过程中,有一个必不可少的环节就是记录日志,相信只要是个程序员都用过,可是咱们自问下,用了这么多年的日志框架,你确定自己真弄懂了日志框架的来龙去脉嘛?下面笔者就详细聊聊java中常用日志框架的...

物理老师教你学Java语言(中篇)(物理专业学编程)

第四章物质的基本结构——类与对象...

一文搞定!Spring Boot3 定时任务操作全攻略

各位互联网大厂的后端开发小伙伴们,在使用SpringBoot3开发项目时,你是否遇到过定时任务实现的难题呢?比如任务调度时间不准确,代码报错却找不到方向,是不是特别头疼?如今,随着互联网业务规模...

你还不懂java的日志系统吗 ?(java的日志类)

一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...

谈谈枚举的新用法--java(java枚举的作用与好处)

问题的由来前段时间改游戏buff功能,干了一件愚蠢的事情,那就是把枚举和运算集合在一起,然后运行一段时间后buff就出现各种问题,我当时懵逼了!事情是这样的,做过游戏的都知道,buff,需要分类型,且...

你还不懂java的日志系统吗(javaw 日志)

一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...

Java 8之后的那些新特性(三):Java System Logger

去年12月份log4j日志框架的一个漏洞,给Java整个行业造成了非常大的影响。这个事情也顺带把log4j这个日志框架推到了争议的最前线。在Java领域,log4j可能相对比较流行。而在log4j之外...

Java开发中的日志管理:让程序“开口说话”

Java开发中的日志管理:让程序“开口说话”日志是程序员的朋友,也是程序的“嘴巴”。它能让程序在运行过程中“开口说话”,告诉我们它的状态、行为以及遇到的问题。在Java开发中,良好的日志管理不仅能帮助...

吊打面试官(十二)--Java语言中ArrayList类一文全掌握

导读...

OS X 效率启动器 Alfred 详解与使用技巧

问:为什么要在Mac上使用效率启动器类应用?答:在非特殊专业用户的环境下,(每天)用户一般可以在系统中进行上百次操作,可以是点击,也可以是拖拽,但这些只是过程,而我们的真正目的是想获得结果,也就是...

Java中 高级的异常处理(java中异常处理的两种方式)

介绍异常处理是软件开发的一个关键方面,尤其是在Java中,这种语言以其稳健性和平台独立性而闻名。正确的异常处理不仅可以防止应用程序崩溃,还有助于调试并向用户提供有意义的反馈。...

【性能调优】全方位教你定位慢SQL,方法介绍下!

1.使用数据库自带工具...

全面了解mysql锁机制(InnoDB)与问题排查

MySQL/InnoDB的加锁,一直是一个常见的话题。例如,数据库如果有高并发请求,如何保证数据完整性?产生死锁问题如何排查并解决?下面是不同锁等级的区别表级锁:开销小,加锁快;不会出现死锁;锁定粒度...

看懂这篇文章,你就懂了数据库死锁产生的场景和解决方法

一、什么是死锁加锁(Locking)是数据库在并发访问时保证数据一致性和完整性的主要机制。任何事务都需要获得相应对象上的锁才能访问数据,读取数据的事务通常只需要获得读锁(共享锁),修改数据的事务需要获...