一、前言
qt对人员管理部分个人总结的模块化程序,直接按照步骤复制粘贴程序,直接实现人员管理功能,无需花费脑筋在理清各个思路,适合快速编写组装程序
二、环境
windows
qt5.7
sqlite3
三、正文
思来想去大半天,不知道做成什么样的模块化能更好一些,想过方式一:直接通过接口调用外部界面,外部界面单独与数据库通讯交互,关闭界面之后回到主程序,这样优点就是程序无需特别多的嵌入,简单直白,缺点就是不容易统一样式和程序风格, 在不同程序结构下感觉会格格不入,所以本文最终选择了方式二:就是将所有使用的控件和代码调用方式开源,这样在写新的程序,在人员管理的位置,只要留好界面,然后放入模板的控件(3个按键,1个表格),在复制粘贴写好的程序就OK了。
话不多说直接看正文。
这里我跟着新建一个空程序,最后会把程序放到下载连接中,提供参考,正文中介绍的也很详细,跟着文章也会实现
1.数据库更改
数据库使用的是sqlite3,按下图创建两个表格,一个是所有用户信息表格,一个是上次登录信息表格
2.更改pro文件
新建一个qt工程,在pro文件中添加如下信息
QT += core gui sql
当然也可以多添加点库,因为都会用到
QT += core gui axcontainer printsupport serialport sql
添加完pro文件对应就得引用相关头文件,这里建议使用一个通用.h文件用于方式公共信息,不用再每个文件中放入一堆头文件,这里我是用一个common.h的头文件,如下所示
#ifndef COMMON_H
#define COMMON_H
#include <QVector>
#include <QMap>
#include <QApplication>
#include <QMainWindow>
#include <QtSql>
#include <QSqlQuery>
#include <QSqlTableModel>
#include <QSerialPort>
#include <QSerialPortInfo>
#include <QDebug>
#include <QFile>
#include <QFileDialog>
#include <QMessageBox>
#include <QProcess>
#include <QPixmap>
#include <QPaintEvent>
#include <QPainter>
#include <windows.h>
#include <QScrollBar>
#include <QTableWidget>
#include <QListWidgetItem>
#include <QTimer>
#include <QTime>
#include <QSystemTrayIcon>
#include <QGridLayout>
#include <QPushButton>
#include <ActiveQt/QAxObject> //Excel
#include <QDoubleSpinBox>
#include <QAbstractItemView>
#include <QCheckBox>
#define WINDOWS
extern QSqlDatabase db;
extern bool massage_dialog(int button_num,QString tile_text,QString massage_content,char ff);
class CommonHelper
{
public:
static void setStyle(const QString &style) {
QFile qss(style);
qss.open(QFile::ReadOnly);
qApp->setStyleSheet(qss.readAll());
qss.close();
}
};
//用户管理结构体
typedef struct
{
QString now_username;//当前用户
QString now_password;//当前密码
bool now_root;//当前权限
QString now_noted;//当前备注
QStringList usernamelist;//用户列表
QStringList userpswdlist;//用户密码
QVector<bool> userroot;//用户权限
QStringList usernote;//用户备注
}UserList;
#endif // COMMON_H
3.更改main文件
之后就是写入口文件了,这里还没到正文主要部分,主要是先说明一下引用的相关环境内容,再main文件中我重定义了一个messagebox函数,可以自己控制消息提示框,当然不习惯的也可以用自己创建的界面代替,可以参照我之前的帖子,共创建了10余种不同功能的messagebox自定义界面。
这里我新建的数据库叫user.db main.cpp如下
#include "common.h"
#include "login.h"
QSqlDatabase db;
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
///更换主题
// CommonHelper::setStyle(":/PIC/qss/white.qss");
//打开数据库
db = QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName(qApp->applicationDirPath()+"/user.db");
db.open();
login y;
y.show();
return a.exec();
}
bool massage_dialog(int button_num,QString tile_text,QString massage_content,char ff)
{
int flag;
QMessageBox *msgBox = new QMessageBox();
QTimer::singleShot(15000,msgBox,SLOT(close())); //也可将accept改为close,定时关闭提示框,防止信号阻塞和假死
/*调整提示框样式*/
switch (button_num) {
case 1:
msgBox->addButton(QObject::tr("确定"), QMessageBox::YesRole);
break;
case 2:
msgBox->addButton(QObject::tr("确定"), QMessageBox::YesRole);
msgBox->addButton(QObject::tr("取消"), QMessageBox::NoRole);
break;
default:
break;
}
msgBox->setWindowTitle(tile_text);
msgBox->setText(massage_content);
if(ff==0)msgBox->setIcon(QMessageBox::NoIcon);
else if(ff==1)msgBox->setIcon(QMessageBox::Information);
else if(ff==2)msgBox->setIcon(QMessageBox::Warning);
else if(ff==3)msgBox->setIcon(QMessageBox::Critical);
else if(ff==4)msgBox->setIcon(QMessageBox::Question);
//msgBox->setStyleSheet("background-color:white");
/*显示提示框*/
flag = msgBox->exec();
return flag;
}
4.登录界面login创建
登录界面login直接复制程序,ui就2个lineedit和2个按键以及两个checkbox,比较基础的人员管理,引用了common的人员部分结构体
login.cpp
#include "login.h"
#include "ui_login.h"
#include "mainwindow.h"
UserList Data_user;//用户信息结构体
login::login(QWidget *parent) :
QWidget(parent),
ui(new Ui::login)
{
ui->setupUi(this);
// this->setWindowFlags(Qt::WindowStaysOnTopHint); //设置置顶
this->setWindowModality(Qt::ApplicationModal);//设置一直保持在顶端,不可切换其他界面,除非被新界面带此属性覆盖
// this->setWindowFlags(Qt::FramelessWindowHint);//设置界面无边框
// this->setFocusPolicy(Qt::StrongFocus);//设置强焦点策略,这样输入法面板隐藏以后单击界面键盘隐藏
this->setWindowFlags(Qt::Dialog);//取消窗口最大化最小化按钮
///获取数据库中用户信息
QSqlQuery qry(db);
if(qry.exec("select * from Sys_user")){//将所有用户信息遍历赋值到Data_user结构体中
for(int i=0;qry.next()&&i<1000;i++){
Data_user.usernamelist.append(qry.value(0).toString());//赋值用户账号
Data_user.userpswdlist.append(qry.value(1).toString());//赋值用户密码
Data_user.userroot.append(qry.value(2).toBool());//赋值当前学员列表(字符串未拆分)
Data_user.usernote.append(qry.value(3).toString());//学员列表是否变更标志
}
}
if(qry.exec("select * from Sys_userlast")&&qry.next()){//读取上次用户登录信息,自动赋值
QString lastusername=qry.value(1).toString();//上次用户账号
QString lastuserpswd=qry.value(2).toString();//上次用户密码
bool savepassword_flag=qry.value(3).toBool();//是否保存密码
bool autologin_flag=qry.value(4).toBool();//是否自动登录
ui->username->setText(lastusername);//自动填入上次用户账号
if(savepassword_flag)ui->userpassword->setText(lastuserpswd);//是否自动填入上次用户密码
ui->checkBox1->setChecked(savepassword_flag);//自动填入上次用户是否保存密码信息
ui->checkBox2->setChecked(autologin_flag);//自动填入上次用户是否自动登录
if(autologin_flag==true){//自动登录上次用户
QTimer::singleShot(10,this,SLOT(on_btn_login_clicked()));//延迟10ms在登录,否则登录界面在最前端,而不是菜单界面
}
}
}
login::~login()
{
delete ui;
}
//记住密码
void login::on_checkBox1_clicked(bool checked)
{
if(!checked)//如果取消记住密码,则取消自动登录选择
ui->checkBox2->setChecked(false);
}
//自动登录
void login::on_checkBox2_clicked(bool checked)
{
if(checked&&ui->checkBox1->isChecked());//只有在记住密码状态才可以选择自动登录
else ui->checkBox2->setChecked(false);//否则不可选择自动登录
}
//登录
void login::on_btn_login_clicked()
{
/*if(ui->username->text()=="admin"&&ui->userpassword->text()=="admin"){//判断超级账户
Data_user.now_username=ui->username->text();//当前用户
Data_user.now_password=ui->userpassword->text();//当前密码
Data_user.now_root=1;
Data_user.now_noted="超级管理员";
menu *menu1 = new menu();
menu1->show();
menu1->setAttribute(Qt::WA_DeleteOnClose);//若是关闭界面,则彻底释放资源
connect(this,SIGNAL(sendlogin(QString,QString,bool,QString)),menu1,SLOT(receivelogin(QString,QString,bool,QString)));
emit sendlogin(Data_user.now_username,Data_user.now_password,Data_user.now_root,Data_user.now_noted);
}
else */if(Data_user.usernamelist.contains(ui->username->text())){//判断普通账户
if(ui->userpassword->text()==Data_user.userpswdlist.at(Data_user.usernamelist.indexOf(ui->username->text()))){//判断密码
//赋值登录信息到当前用户信息缓存
Data_user.now_username=ui->username->text();//当前用户
Data_user.now_password=ui->userpassword->text();//当前密码
Data_user.now_root=Data_user.userroot.at(Data_user.usernamelist.indexOf(ui->username->text()));//当前登录者管理员状态
Data_user.now_noted=Data_user.userroot.at(Data_user.usernamelist.indexOf(ui->username->text()));//当前登录者备注
//存储当前登录界面配置信息到数据库
QSqlQuery qry(db);//数据库保存
qry.exec(QString("UPDATE Sys_userlast set username='%1',userpassword='%2',savepwd='%3',autologin='%4' where id = 1;").arg(ui->username->text()).arg(ui->userpassword->text()).arg(ui->checkBox1->isChecked()).arg(ui->checkBox2->isChecked()));
//登录到主界面
MainWindow *menu1 = new MainWindow();
menu1->show();
menu1->setAttribute(Qt::WA_DeleteOnClose);//若是关闭界面,则彻底释放资源
}
else{
massage_dialog(1,"提示","密码错误!请重新输入!",1);
}
}
else{
massage_dialog(1,"提示","用户不存在!",1);
}
}
//关机
void login::on_btn_pushout_clicked()
{
this->close();
}
login.h
#ifndef LOGIN_H
#define LOGIN_H
#include "common.h"
extern UserList Data_user;
namespace Ui {
class login;
}
class login : public QWidget
{
Q_OBJECT
public:
explicit login(QWidget *parent = 0);
~login();
signals:
private slots:
void on_checkBox1_clicked(bool checked);
void on_checkBox2_clicked(bool checked);
void on_btn_login_clicked();
void on_btn_pushout_clicked();
private:
Ui::login *ui;
};
#endif // LOGIN_H
这里还建立了一个picture的资源文件,添加了2张选择框样式图片
5.主界面mainwindow创建
在登录界面登录成功就是进入到主界面了,在主界面中只需创建3个按键和一个表格即可,前文的数据库调用,common.h文件的人员结构体,main文件的自定义messagebox都是铺垫,都是这里需要用到的资源,如果项目中包含这些,就不用从前再缕过来了,当然这是经历过几个程序叠加后的说,不然第一次用肯定不是按照我的思路,继续。
在mainwindow.ui文件中创建控件
其中表格的样式表可以复制下面的
QTableView{/*设置常规*/
alternate-background-color:rgb(255,255,255,20);
background:black;
color:white;
gridline-color : rgb(120, 120, 120);
font: 75 10pt "宋体";
}
QHeaderView::section{/*设置表头*/
font: 75 16pt "宋体";
color: white;
background-color: rgb(42, 63, 22);
}
QTableView::Item:Selected{/*设置选中*/
alternate-background-color:rgb(255,255,255,20);
background:rgb(255,255,255,40);
color: rgb(0, 0, 255);
}
QScrollBar:vertical {/*设置滚动条背景*/
border: none;
background-color: rgb(42,63,22);
width: 3px;
}
QScrollBar::handle:vertical {/*设置滑动条*/
border: none;
border-radius:2px;
background: rgb(120, 120, 120);
}
QScrollBar::sub-line:vertical {
border: none;
height: 0px;
subcontrol-position: top;
subcontrol-origin: margin;
}
QScrollBar::add-line:vertical {
border: none;
height: 0px;
subcontrol-position: bottom;
subcontrol-origin: margin;
}
QScrollBar::up-arrow:vertical, QScrollBar::down-arrow:vertical {
border:none;
width: 0px;
height: 0px;
}
QScrollBar::add-page:vertical, QScrollBar::sub-page:vertical {
background: none;
}
控件名称tableWidget_user,btn_user_add,btn_user_change,btn_user_delete分别对应人员显示表格,人员添加按键,人员修改按键,人员删除按键。
mainwindow.h文件,这里增加了表格滑动功能,当表格放满一页后,通过触摸或者鼠标可以模拟手机滑动界面上下滑动
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include "common.h"
namespace Ui {
class MainWindow;
}
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
explicit MainWindow(QWidget *parent = 0);
~MainWindow();
signals:
//人员管理
void sendtext(QString,QString,bool,QString,QStringList);//messagebox_user
protected:
//表格滑动刷新
bool eventFilter(QObject *obj, QEvent *event);
private slots:
//人员管理
void on_btn_user_add_clicked();//用户增加
void on_btn_user_change_clicked();//用户修改
void on_btn_user_delete_clicked();//用户删除
void tableWidget_user_init(QTableWidget *tab,QVector<int> line,QStringList name,bool push);
void tableWidget_user_refuse(QTableWidget *TableWidget);//更新用户信息
private:
Ui::MainWindow *ui;
QScrollBar *m_scrollBarV;//表格滚动实现
};
#endif // MAINWINDOW_H
mainwindow.cpp文件
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "login.h"
#include "messagebox/messagebox_user.h"
//风格:文字白色,图片选择样式切换
QString QCheckBoxstyle=
"QCheckBox{color: rgb(255, 255, 255);background-color: rgb(0, 0, 0);}"
"QCheckBox::indicator{background-color: rgba(255, 255, 255, 0);border: 0px solid #b1b1b1;width: 30px;height: 30px;}"
"QCheckBox::indicator:unchecked {image:url(:/PIC/未选择.png);}"
"QCheckBox::indicator:unchecked:hover {image:url(:/PIC/未选择.png);}"
"QCheckBox::indicator:unchecked:pressed {image:url(:/PIC/未选择.png);}"
"QCheckBox::indicator:checked {image:url(:/PIC/选择.png);}"
"QCheckBox::indicator:checked:hover {image:url(:/PIC/选择.png);}"
"QCheckBox::indicator:checked:pressed {image:url(:/PIC/选择.png);}";
QString btnwhite="color: rgb(255, 255, 255);";
QString btngreen="color: rgb(0, 255, 0);";
QString btnblue="color: rgb(20, 50, 255);";
QString btngray="color: rgb(160, 160, 160);";
QString btnstyle=
"background-color: rgba(255, 255, 255, 0);"
"border-right:0px solid #9eca56; "
"border-bottom:0px solid #9eca56;"
"border-left:0px solid #9eca56;"
"border-top:0px solid #9eca56; "
"font: 75 italic 12pt '宋体';"
"text-decoration: underline;";
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
this->setWindowModality(Qt::ApplicationModal);//设置一直保持在顶端,不可切换其他界面,除非被新界面带此属性覆盖
this->setWindowFlags(Qt::Dialog);//取消窗口最大化最小化按钮
//用户管理初始化
QVector<int> table_line;
table_line.append({140,600,400,100,300});//配置表格宽度5列
QStringList table_title;
table_title.append({"序号","账号(操作人员)","密码","管理员","备注"});//配置表格表头名称5列
m_scrollBarV = ui->tableWidget_user->verticalScrollBar();//绑定表格滑动效//在表格第一次初始化前必须初始化一次,否则程序崩溃
tableWidget_user_init(ui->tableWidget_user,table_line,table_title,true);//初始化表格表头
tableWidget_user_refuse(ui->tableWidget_user);//更新全部用户信息
connect(ui->tableWidget_user,&QTableWidget::cellPressed,[=](){//点击表格某行生效,当需要滑动时必定点击到表格某行。
m_scrollBarV = ui->tableWidget_user->verticalScrollBar();//绑定表格滑动效
});
}
MainWindow::~MainWindow()
{
delete ui;
}
/*****************************人员管理处理函数**********************************/
//用户增加
void MainWindow::on_btn_user_add_clicked()
{
MessageBox_user * box = new MessageBox_user();
box->show();
box->setAttribute(Qt::WA_DeleteOnClose);//若是关闭界面,则彻底释放资源
connect(this,SIGNAL(sendtext(QString,QString,bool,QString,QStringList)),box,SLOT(receivetext(QString,QString,bool,QString,QStringList)));
emit sendtext("","",false,"",Data_user.usernamelist);
connect(box,&MessageBox_user::sendtext,[=](QString res1,QString res2,bool res3,QString res4){
qDebug()<<res1<<res2<<res3<<res4;
QSqlQuery qry(db);//数据库保存
qry.exec(QString("INSERT INTO Sys_user(name,password,root,note) VALUES ('%1','%2','%3','%4');").arg(res1).arg(res2).arg(res3).arg(res4));
tableWidget_user_refuse(ui->tableWidget_user);//更新全部用户信息
});
}
//用户修改
void MainWindow::on_btn_user_change_clicked()
{
int checkid=0;
QList<QCheckBox*> checkList = ui->tableWidget_user->findChildren<QCheckBox*>();
for(int i=0;i<checkList.size();i++){//遍历寻找是否有选中
if(checkList.at(i)->isChecked()){//如果找到选中,则跳出循环
checkid=checkList.at(i)->objectName().replace("QCheckBox_user_","").toInt();//替换控件名称转换为表格所在的行
break;//跳出寻找
}
}
if(checkid>0){//有选中的行
MessageBox_user * box = new MessageBox_user();
box->show();
box->setAttribute(Qt::WA_DeleteOnClose);//若是关闭界面,则彻底释放资源
connect(this,SIGNAL(sendtext(QString,QString,bool,QString,QStringList)),box,SLOT(receivetext(QString,QString,bool,QString,QStringList)));
QString change_username=Data_user.usernamelist.at(checkid-1);//赋值选中用户名称
QString change_userpassword=Data_user.userpswdlist.at(checkid-1);//赋值选中用户密码
bool change_userroot=Data_user.userroot.at(checkid-1);//赋值选中用户权限
QString change_usernote=Data_user.usernote.at(checkid-1);//赋值选中用户备注
emit sendtext(change_username,change_userpassword,change_userroot,change_usernote,Data_user.usernamelist);
connect(box,&MessageBox_user::sendtext,[=](QString res1,QString res2,bool res3,QString res4){
// qDebug()<<res1<<res2<<res3<<res4;
//判断修改的用户是否为当前用户
int err_flag=0;//错误编号
if(change_username==Data_user.now_username){
if(res3==0){
res3=1;
err_flag=1;//降低本身权限,可能导致没有账号能进入管理员系统
}
//赋值登录信息到当前用户信息缓存
Data_user.now_username=res1;//当前用户
Data_user.now_password=res2;//当前密码
Data_user.now_root=res3;//赋值权限
Data_user.now_noted=res4;//赋值备注
//存储当前登录界面配置信息到数据库
QSqlQuery qry(db);//数据库保存
qry.exec(QString("UPDATE Sys_userlast set username='%1',userpassword='%2' where id = 1;").arg(res1).arg(res2));
//刷新当前用户名称
//ui->label_name->setText(QString("当前用户:%1").arg(Data_user.now_username));
}
QSqlQuery qry(db);//数据库保存
qry.exec(QString("UPDATE Sys_user set name='%1',password='%2',root='%3' ,note='%4'where name = '%5';").arg(res1).arg(res2).arg(res3).arg(res4).arg(change_username));
tableWidget_user_refuse(ui->tableWidget_user);//更新全部用户信息
if(err_flag==1)
massage_dialog(1,"提示","用户信息修改成功!\r\n(不可修改自身权限,已自动恢复)",1);
else
massage_dialog(1,"提示","用户信息修改成功!",1);
});
}
else{
massage_dialog(1,"提示","请选择要修改的用户列表序号!",1);
}
}
//用户删除
void MainWindow::on_btn_user_delete_clicked()
{
int checkid=0;
QList<QCheckBox*> checkList = ui->tableWidget_user->findChildren<QCheckBox*>();
for(int i=0;i<checkList.size();i++){//遍历寻找是否有选中
if(checkList.at(i)->isChecked()){//如果找到选中,则跳出循环
checkid=checkList.at(i)->objectName().replace("QCheckBox_user_","").toInt();//替换控件名称转换为表格所在的行
break;//跳出寻找
}
}
if(checkid>0){//有选中的行
if(Data_user.usernamelist.at(checkid-1)==Data_user.now_username){
massage_dialog(1,"提示","不可以删除当前登录用户!",1);
}
else{
int res=massage_dialog(2,"提示",QString("是否确认删除用户%1").arg(Data_user.usernamelist.at(checkid-1)),4);
if(res==0){//点击确认返回0,点击取消或关闭返回1
QSqlQuery qry(db);
qry.exec(QString("delete from Sys_user where name = '%1'").arg(Data_user.usernamelist.at(checkid-1)));
qry.prepare("vacuum");//删除成功之后释放内存碎片
qry.exec();
tableWidget_user_refuse(ui->tableWidget_user);//更新全部用户信息
massage_dialog(1,"提示","删除用户成功!",1);
}
}
}
else{
massage_dialog(1,"提示","请选择要删除的用户列表序号!",1);
}
}
//通用表格初始化函数,建立表格列,刷新表头和列宽
void MainWindow::tableWidget_user_init(QTableWidget *tab,QVector<int> line,QStringList name,bool push)
{
tab->clearContents();//清空内容
tab->verticalHeader()->setVisible(false);//去掉行序号
tab->horizontalHeader()->setFixedHeight(40); //设置表头的高度
tab->horizontalHeader()->setStretchLastSection(true);//设置表格是否充满,即行末不留空
//tab->horizontalHeader()->setSectionResizeMode(QHeaderView::Fixed); //禁止鼠标拖放列宽度
tab->horizontalHeader()->setFocusPolicy(Qt::NoFocus); //设置表头不可选
tab->horizontalHeader()->setHighlightSections(false); //设置表头不可选//QTableWidget表头塌陷问题解决
tab->setEditTriggers(QAbstractItemView::NoEditTriggers);//设置表格内容不可修改
tab->setSelectionBehavior(QAbstractItemView::SelectRows);//设置选中就是一行选中
tab->setSelectionMode(QAbstractItemView::SingleSelection);//设置只能选中一行
tab->setFocusPolicy(Qt::NoFocus);//设置去掉选中虚线框
//tab->setAlternatingRowColors(true);//设置表格颜色交替
//表格滚动部分实现函数:初始化函数+以下俩函数+ (QObject *obj, QEvent *event)
if(push)tab->viewport()->installEventFilter(this);//对此对象安装事件过滤器
if(push)tab->setVerticalScrollMode(QAbstractItemView::ScrollPerPixel);//设置滚动模式按照像素滑动,做表格滑动时使用
tab->setColumnCount(line.size());//设置列数量
for(int i=0;i<line.size();i++)
tab->setColumnWidth(i,line.at(i));
tab->setHorizontalHeaderLabels(name);//设置table列表各项标题
tab->setRowCount(0);
}
//更新用户信息
void MainWindow::tableWidget_user_refuse(QTableWidget *TableWidget)
{
Data_user.usernamelist.clear();//清空用户账号
Data_user.userpswdlist.clear();//清空用户密码
Data_user.userroot.clear();//清空用户权限
Data_user.usernote.clear();//清空用户备注
QSqlQuery qry(db);
if(qry.exec("select * from Sys_user")){
for(int i=0;qry.next()&&i<1000;i++){
Data_user.usernamelist.append(qry.value(0).toString());//赋值用户账号
Data_user.userpswdlist.append(qry.value(1).toString());//赋值用户密码
Data_user.userroot.append(qry.value(2).toBool());//赋值用户权限
Data_user.usernote.append(qry.value(3).toString());//赋值用户备注
}
}
TableWidget->clearContents();//清空内容
TableWidget->setRowCount(Data_user.usernamelist.size());//设置表格行数
//删除上次表格的复选框控件
QList<QCheckBox*> checkList = TableWidget->findChildren<QCheckBox*>();
for(int i=0;i<checkList.size();i++)delete checkList[i];
//删除上次表格的按键控件
QList<QPushButton*> btnList = TableWidget->findChildren<QPushButton*>();
for(int i=0;i<btnList.size();i++)delete btnList[i];
for(int i=0;i<Data_user.usernamelist.size();i++){ //刷新信号列表
//表格插入复选框
QCheckBox *lineid=new QCheckBox;
lineid->setObjectName(tr("QCheckBox_user_%1").arg(i+1));
TableWidget->setCellWidget(i,0,lineid);//插入按键到表格指定行列
lineid->setText(QString("%1").arg(i+1,2,10,QChar('0')));//设置文本
lineid->setStyleSheet(QCheckBoxstyle);//设置风格
lineid->setFont(QFont("隶书", 12, QFont::Bold));//设置文字
lineid->setAutoExclusive(true);//设置只有一个复选框可以选择,其他的自动失效
TableWidget->setItem(i,1,new QTableWidgetItem(Data_user.usernamelist.at(i)));//更新账号
TableWidget->item(i,1)->setTextAlignment(Qt::AlignHCenter|Qt::AlignVCenter);//数据居中
TableWidget->setItem(i,2,new QTableWidgetItem(Data_user.userpswdlist.at(i)));//更新密码
TableWidget->item(i,2)->setTextAlignment(Qt::AlignHCenter|Qt::AlignVCenter);//数据居中
if(Data_user.userroot.at(i))
TableWidget->setItem(i,3,new QTableWidgetItem("是"));//更新管理员
else
TableWidget->setItem(i,3,new QTableWidgetItem("否"));//更新管理员
TableWidget->item(i,3)->setTextAlignment(Qt::AlignHCenter|Qt::AlignVCenter);//数据居中
QPushButton *aaa=new QPushButton;
aaa->setObjectName(tr("btn_page5_usertable_1_%1").arg(i+1));
TableWidget->setCellWidget(i,4,aaa);//插入按键到表格指定行列
aaa->setText("备注信息");//设置文本
QString styles=btnwhite+btnstyle;styles.replace("12pt","16pt");
aaa->setStyleSheet(styles);//设置风格,加大字号
connect(aaa,&QPushButton::clicked,[=](){//绑定按键点击回调槽函数
massage_dialog(1,"备注信息",Data_user.usernote.at(i),1);
});
TableWidget->setRowHeight(i,50);//设置行高
}
// TableWidget->resizeRowsToContents();//根据单元格内容自动换行
}
/*****************************通用处理函数**********************************/
//滑动表格实现函数
bool MainWindow::eventFilter(QObject *obj, QEvent *event)
{
static signed int press_y = 0;
static signed int move_y = -1;
static signed int release_y = 0;
static QDateTime pressDateTime;
static QPropertyAnimation *animation = new QPropertyAnimation();
// qDebug()<<obj->objectName()<<event->type();
if("qt_scrollarea_viewport" != obj->objectName())
return false;
int scrollV_max = m_scrollBarV->maximum();//获取当前滚动控件的最大进度条值
int scrollV_min = m_scrollBarV->minimum();//获取当前滚动控件的最小进度条值,基本是0
#ifdef WINDOWS
int global_y=QCursor::pos().y();//相对于正屏幕的y坐标,window下此方法即可获取
#else
QMouseEvent *event1 = (QMouseEvent *)event;
int global_y=event1->globalPos().y();//相对于正屏幕的y坐标,linux下用此方法获取
#endif
//根据鼠标的动作——按下、放开、拖动,执行相应的操作
if(event->type() == QEvent::MouseButtonPress){//记录按下的时间、坐标
pressDateTime = QDateTime::currentDateTime();
move_y = global_y;
press_y = move_y;
animation->stop();
}
else if(event->type() == QEvent::MouseButtonRelease){//鼠标放开,根据鼠标拖动的垂直距离和持续时间,设置窗口滚动快慢程度和距离
if(animation->targetObject() != m_scrollBarV){
animation->setTargetObject(m_scrollBarV);
animation->setPropertyName("value");
}
move_y = -1;
release_y = global_y;
QObject *parent_obj = obj->parent();
if(parent_obj != 0 || parent_obj->inherits("QAbstractItemView"))
QTimer::singleShot(150, (QAbstractItemView *)parent_obj, SLOT(clearSelection()));
int endValue;
int pageStep;
if(release_y - press_y != 0 && qAbs(release_y - press_y) > 45){
int mseconds = pressDateTime.msecsTo(QDateTime::currentDateTime());
int limit = 440;
pageStep = 240;//scrollBarV->pageStep();
if(mseconds > limit)//滑动的时间大于某个值的时候,不再滚动(通过增加分母)
mseconds = mseconds + (mseconds - limit) * 20;
if(release_y - press_y > 0){
endValue = m_scrollBarV->value()- pageStep * (200.0 / mseconds);//.0避免避免强制转换为整形
if(scrollV_min > endValue)endValue = scrollV_min;
}
else if(release_y - press_y < 0){
endValue = m_scrollBarV->value() + pageStep * (200.0 / mseconds);
if(endValue > scrollV_max)
endValue = scrollV_max;
}
if(mseconds > limit)mseconds = 0;//滑动的时间大于某个值的时候,滚动距离变小,减小滑动的时间
animation->setDuration(mseconds+550);
animation->setEndValue(endValue);
animation->setEasingCurve(QEasingCurve::OutQuad);
animation->start();
return true;
}
}
else if(event->type() == QEvent::MouseMove && move_y >= 0){//窗口跟着鼠标移动
int move_distance = global_y - move_y;
int endValue = m_scrollBarV->value() - move_distance;
if(scrollV_min > endValue)endValue = scrollV_min;
if(endValue > scrollV_max)endValue = scrollV_max;
m_scrollBarV->setValue(endValue);
move_y = global_y;
}
return false;
}
这里引用了MessageBox_user这个类,用于与弹出界面交互,这个直接下载就行,开放一下就是如下:
messagebox_user.h
#ifndef MESSAGEBOX_USER_H
#define MESSAGEBOX_USER_H
#include <QDialog>
#include "common.h"
//The code from david!!!
#include <Windows.h>
#pragma comment(lib, "user32.lib")
namespace Ui {
class MessageBox_user;
}
class MessageBox_user : public QDialog
{
Q_OBJECT
public:
explicit MessageBox_user(QWidget *parent = 0);
~MessageBox_user();
protected:
void paintEvent(QPaintEvent *paintevent);
signals:
void sendtext(QString,QString,bool,QString);
private slots:
void on_btn_sure_clicked();
void on_btn_false_clicked();
void receivetext(QString res1,QString res2,bool res3,QString res4,QStringList has);
void closeEvent(QCloseEvent *event);//关闭程序
void widgetShake(QWidget *pWidget, int nRange);//控件振动
private:
Ui::MessageBox_user *ui;
QPixmap pic;
QStringList m_hascode;//已有的账号
};
#endif // MESSAGEBOX1_H
messagebox_user.cpp
#include "messagebox_user.h"
#include "ui_messagebox_user.h"
MessageBox_user::MessageBox_user(QWidget *parent) :
QDialog(parent),
ui(new Ui::MessageBox_user)
{
ui->setupUi(this);
this->setWindowModality(Qt::ApplicationModal);//设置一直保持在顶端,不可切换其他界面,除非被新界面带此属性覆盖
this->setWindowFlags(Qt::Dialog);//取消窗口最大化最小化按钮
//如果设置背景为指定图片,打开下面两句
// this->setWindowFlags(Qt::FramelessWindowHint);//设置窗体无边框
// this->setAttribute(Qt::WA_TranslucentBackground);//设置背景透明
// pic.load(tr(":/PIC/background/message5.png"));//设置背景
///设置按键调出键盘事件
connect(ui->pushButton_keyboard,&QPushButton::clicked,[=](){
//The code from david!!!
PVOID OldValue;
BOOL bRet = Wow64DisableWow64FsRedirection (&OldValue);
QString csProcess="C:\\Windows\\System32\\osk.exe";
QString params="";
ShellExecute(NULL, L"open", (LPCWSTR)csProcess.utf16(), (LPCWSTR)params.utf16(), NULL, SW_SHOWNORMAL);
if ( bRet ) Wow64RevertWow64FsRedirection(OldValue);
});
//倒计时关闭界面
// for(int i=0;i<60;i++)
// QTimer::singleShot(i*1000,this,[=](){ui->btn_false->setText(QString("取消(%1)").arg(59-i));});
// QTimer::singleShot(60*1000,this,[=](){this->close();});
// ui->btn_false->setFocus();//设置聚焦,作用你懂的
}
MessageBox_user::~MessageBox_user()
{
delete ui;
}
void MessageBox_user::paintEvent(QPaintEvent *paintevent)
{
paintevent->ignore();
QPainter painter(this);
painter.drawPixmap(0, 0, pic);//绘制图像
}
void MessageBox_user::on_btn_sure_clicked()
{
if(ui->lineEdit_1->text().isEmpty()){
ui->label_5->setText("不能为空");
widgetShake(ui->label_5,5);
ui->lineEdit_1->setFocus();
return;
}
else if(m_hascode.contains(ui->lineEdit_1->text())){
ui->label_5->setText("账号存在");
widgetShake(ui->label_5,5);
ui->lineEdit_1->setFocus();
return;
}
else if(ui->lineEdit_1->text()=="不限"){
ui->label_5->setText("禁用名称");
widgetShake(ui->label_5,5);
ui->lineEdit_1->setFocus();
return;
}
else{
ui->label_5->setText("");
}
if(ui->lineEdit_2->text().isEmpty()){
ui->label_6->setText("不能为空");
widgetShake(ui->label_6,5);
ui->lineEdit_2->setFocus();
return;
}
else{
ui->label_6->setText("");
}
emit sendtext(ui->lineEdit_1->text(),ui->lineEdit_2->text(),ui->checkBox_gly->isChecked(),ui->textEdit->toPlainText());
this->close();
}
void MessageBox_user::on_btn_false_clicked()
{
this->close();
}
void MessageBox_user::receivetext(QString res1,QString res2,bool res3,QString res4,QStringList has)
{
ui->lineEdit_1->setText(res1);
ui->lineEdit_2->setText(res2);
ui->checkBox_gly->setChecked(res3);
ui->textEdit->setText(res4);
m_hascode=has;
m_hascode.removeAt(m_hascode.indexOf(res1));
// qDebug()<<m_hascode;
if(res1.isEmpty()&&res2.isEmpty())
ui->lab_title->setText("人员增加");
else
ui->lab_title->setText("人员修改");
}
//振动效果
void MessageBox_user::widgetShake(QWidget *pWidget, int nRange)
{
int nX = pWidget->x();
int nY = pWidget->y();
QPropertyAnimation *pAnimation = new QPropertyAnimation(pWidget,"geometry");
pAnimation->setEasingCurve(QEasingCurve::InOutSine);
pAnimation->setDuration(300);
pAnimation->setStartValue(QRect(QPoint(nX,nY),pWidget->size()));
int nShakeCount = 20; //抖动次数
double nStep = 1.0/nShakeCount;
for(int i = 1; i < nShakeCount; i++){
nRange = i&1 ? -nRange : nRange;
pAnimation->setKeyValueAt(nStep*i,QRect(QPoint(nX + nRange,nY),pWidget->size()));
}
pAnimation->setEndValue(QRect(QPoint(nX,nY),pWidget->size()));
pAnimation->start(QAbstractAnimation::DeleteWhenStopped);
}
void MessageBox_user::closeEvent(QCloseEvent *event)
{
HWND appWnd;
appWnd = ::FindWindow(L"OSKMainClass", NULL);
if (appWnd){
SendMessage(appWnd, WM_SYSCOMMAND, SC_CLOSE, 0);
}
}
messagebox_user.ui
最后编译运行即可,总结前文,主要就是准备阶段连套的稍微多了一些,这个基本每个程序都是需要的,都这么写就行,人员管理部分按照上面头文件和源文件两大块直接复制粘贴,创建4个控件放在界面中,就实现功能了,妈妈再也不用担心我写程序在人员管理这块功能浪费时间了。
实现效果:
未点击自动登录下,点击记住密码,自动刷新上次账号密码,不登陆
点击自动登录,和记住密码,下次打开软件自动登录到mainwindow界面
取消记住密码自动取消自动登录
登录后自动刷新数据库中存储的所有人员信息
点击人员添加界面弹出新建人员提示框
输入信息重复或输错信息会有提示
录入成功后自动更新表格,可以点击序号列进行人员修改
修改当前登录人员会自动更新记录的上次登录信息,下次还是自动登录当前人员
删除人员具有消息选择框提示选择是否删除,15秒无操作自动关闭界面,只有点击确认才删除
动态演示如下:
四、结语
本文例程下载链接