#ifndef EXAMDIALOG_H
#define EXAMDIALOG_H
#include <QDialog>
#include <QTimer>
#include <QTextEdit>
#include <QLabel>
#include <QRadioButton>
#include <QCheckBox>
#include <QGridLayout>
#include <QButtonGroup>
class ExamDialog : public QDialog
{
Q_OBJECT
public:
ExamDialog(QWidget* parent = 0);
void initTimer(); //初始化计时器
void initLayout(); //初始化布局管理器
bool initTextEdit();//初始化文本编辑器
void initButtons(); //初始化按钮及标签
bool hasNoSelect(); //判断题目是否有未完成的
private:
QTimer *m_timer; //计时器
int m_timeGo; //考试已用时
QTextEdit *m_textEdit; //考试题库显示
QLabel *m_titleLabels[10]; //题目标签
QButtonGroup *m_btnGroups[9]; //单项按钮分组
QRadioButton *m_radioBtns[32]; //单选题按钮
QCheckBox *m_checkBtns[4]; //多选题按钮
QRadioButton *m_radioA; //判断题A选项
QRadioButton *m_radioB; //判断题B选项
QGridLayout *m_layout; //布局管理器
QStringList m_answerList; //存放答案的链表
private slots:
void freshTime(); //刷新考试时间
void getScore(); //获取考试成绩
};
#endif // EXAMDIALOG_H
#include "examdialog.h"
#include <QFile>
#include <QTextStream>
#include <QMessageBox>
#include <QApplication>
#include <QPushButton>
ExamDialog::ExamDialog(QWidget* parent):QDialog(parent)
{
//设置字体大小
QFont font;
font.setPointSize(12);
setFont(font);
//设置窗体背景颜色
setPalette(QPalette(QColor(209,215,255)));
setWindowTitle("考试已用时:0分0秒");
setWindowFlags(Qt::Dialog | Qt::WindowCloseButtonHint); //基本对话框风格加上一个关闭按钮
resize(800,900);
// 调用初始化计时器
initTimer();
//初始化布局管理器
initLayout();
//初始化文本编辑器
if(!initTextEdit()){
QMessageBox::information(this,"提示","初始化题库数据文件失败!");
//因为初始化失败,所以系统就直接退出(发送信号)。
//第一个参数:多久来发送信号(0:立即退出) 第二个参数:想要那个对象相应槽方法(当前应用程序:QApp[应用程序的全局对象]) 第三个参数:响应方法
QTimer::singleShot(0,qApp,SLOT(quit()));//间隔0s,当前的应用程序执行退出操作
}
initButtons();
show();
}
void ExamDialog::initTimer()
{
m_timeGo = 0;
m_timer = new QTimer(this);
m_timer->setInterval(1000);
m_timer->start();
connect(m_timer,SIGNAL(timeout()),this,SLOT(freshTime()));
}
void ExamDialog::initLayout()
{
m_layout = new QGridLayout(this);
m_layout->setSpacing(10); //设置控件间的间距
m_layout->setMargin(10); //设置窗体与控件间的间隙
}
bool ExamDialog::initTextEdit()
{
QString strLine; //保存文件中读取到的一行数据
QStringList strList; //保存读取到的答案行
QString fileName("exam.txt");
QFile file(fileName);
QTextStream stream(&file);
stream.setCodec("UTF-8");
if( file.open(QIODevice::ReadOnly | QIODevice::Text) )
{
m_textEdit = new QTextEdit(this);
m_textEdit->setReadOnly(true); //将文本设置为只读属性,不然现实的文本框居然还能够被修改。
QString strText; //用于保存显示到文本编辑器的数据
int nLines = 0;
while(!stream.atEnd())
{
//过滤首行
if(nLines == 0){
stream.readLine();
nLines++;
continue;
}
//过滤答案行(题目、ABCD、答案一共是6行)。一共有10个题。第十题是判断题需要特殊的处理。
if( (nLines >= 6 && nLines <= 6 * 9 && (nLines % 6 == 0) ) /*选择题的答案*/
|| (nLines == 6 * 9 + 4)/*判断题的答案行*/){
//对于答案行的处理
strLine = stream.readLine(); //先读取这一行
strList = strLine.split(" "); //对答案进行处理——以空格进行分割(eg:答案 A)
m_answerList.append(strList.at(1));//把答案存放到答案链表里边 因为0:答案 1:A\b\c\d
strText += "\n";
nLines++;
continue;
}
//读取一行
strText += stream.readLine();
strText += "\n";
nLines++;
}
//添加布局
m_textEdit->setText(strText);
m_layout->addWidget(m_textEdit,0,0,1,10);//窗口对象,行、列、行宽、列宽
file.close();
return true;
}else{
return false;
}
}
void ExamDialog::initButtons()
{
//答案的文本
QStringList strList = {"A","B","C","D"};
//对十道题进行布局
for(int i = 0; i <10; i++){
//题目标签
m_titleLabels[i] = new QLabel(this);
m_titleLabels[i]->setText("第" + QString::number(i+1) + "题");
//添加到布局管理器中 参一:添加的部件的对象 参二:第几行(因为文本编辑器在第0行,所以题目标签在第1行) 参三:第几列
m_layout->addWidget(m_titleLabels[i],1,i);
//判断题(也就是最后一题了——需要特殊处理的)
if(i == 9){
//单选
m_radioA = new QRadioButton(this);//构造单选对象
m_radioB = new QRadioButton(this);//构造单选对象
//添加到布局里面(文本编辑器第0行、题目标签第1行) 正确地2行,错误第3行 判断题占据最后一列。
m_radioA->setText("正确");
m_radioB->setText("错误");
m_layout->addWidget(m_radioA,2,9); //正确:2行9列
m_layout->addWidget(m_radioB,3,9); //错误:3行9列
//将单选进行分组,不加这个的话,就会整个单选框只能选择一个
m_btnGroups[8] = new QButtonGroup(this);
m_btnGroups[8]->addButton(m_radioA);
m_btnGroups[8]->addButton(m_radioB);
//判断题是最后一个溜了,当他构造完了之后直接就能退出了
break;
}
if(i < 8) m_btnGroups[i] = new QButtonGroup(this);
//选择题(4个按钮需要布局)
for(int j = 0; j < 4; j++)
{
if( i == 8){//第八题为:多项多选题。所以需要特殊处理
m_checkBtns[j] = new QCheckBox(this);//构造多选对象
m_checkBtns[j]->setText(strList.at(j)); //拿到 A、B、C、D这四个选项
m_layout->addWidget(m_checkBtns[j],2+j,8); //A、B、C、D各占第2、3、4、5行8第列
}else{//单项选择题
m_radioBtns[4 * i + j] = new QRadioButton(this); //构造出所有的单选的按钮对象
m_radioBtns[4 * i + j]->setText(strList.at(j)); //拿到 A、B、C、D这四个选项
m_layout->addWidget(m_radioBtns[4 * i + j],2+j,i); //A、B、C、D各占第2、3、4、5行0~7第列
//将单选进行分组,不加这个的话,就会整个单选框只能选择一个
m_btnGroups[i]->addButton(m_radioBtns[4 * i + j]);
}
}
}
//构造提交按钮的对象
QPushButton *submitBtn = new QPushButton(this);
submitBtn->setText("提交");
submitBtn->setFixedSize(100,35);
//适用槽函数来获取分数
connect(submitBtn,SIGNAL(clicked(bool)),this,SLOT(getScore()));
m_layout->addWidget(submitBtn,6,9); //设置提交按钮的位置
}
//判断题目是否有未完成的
bool ExamDialog::hasNoSelect()
{
int radioSelects = 0;//单选题的计数
//通过按钮分组来判断,来检查按钮是否被选上了
for(int i = 0; i < 8; i++)
{
//如果有按钮选上了,计数+1
if( m_btnGroups[i]->checkedButton() )
radioSelects++;
}
//如果不足 8 说明有未完成的单选
if(radioSelects != 8)
return true;
//对多选按钮进行判断——只要有选中了的就行
int checkSelects = 0;
for(int i = 0; i < 4; i++)
{
if(m_checkBtns[i]->isChecked())
checkSelects++;
}
//如果多选只有1个或者没有那么就选的不完全,即又为完成的选项
if(checkSelects == 0 || checkSelects == 1)
return true;
//判断题——只有有一个选上了就ok
if(!m_radioA->isChecked() && !m_radioB->isChecked())
return true;
return false;
}
void ExamDialog::getScore()
{
if(hasNoSelect()){
QMessageBox::information(this,"提示","您有未完成的题目,请完成考试!","是");
return;
}
int scores = 0;
//对10个题进行判断
for(int i = 0; i < 10; i++)
{
//单选题计分
if( i < 8)
//与链表中的答案进行对比,如果正确,则+10
if(m_btnGroups[i]->checkedButton()->text() == m_answerList.at(i))
scores += 10;
//多项选择题计分
if(i == 8){
QString answer = m_answerList.at(i);
bool hasA = false;
bool hasB = false;
bool hasC = false;
bool hasD = false;
//contains:是否包含指定的字符串
//提取答案是否正确
if( answer.contains("A") ) hasA = true;
if( answer.contains("B") ) hasB = true;
if( answer.contains("C") ) hasC = true;
if( answer.contains("D") ) hasD = true;
//提取选择的按钮是否选中。选中了返回true,未选中false
bool checkA = m_checkBtns[0]->checkState();
bool checkB = m_checkBtns[1]->checkState();
bool checkC = m_checkBtns[2]->checkState();
bool checkD = m_checkBtns[3]->checkState();
//因为多选要所有答案都对才记分,如果状态不一致那么跳过本次循环
if( hasA != checkA) continue;
if( hasB != checkB) continue;
if( hasC != checkC) continue;
if( hasD != checkD) continue;
scores += 10;
}
//判断题计分
if(i == 9){
//选中项文本和答案文本进行对比
if(m_btnGroups[8]->checkedButton()->text() == m_answerList.at(i))
scores += 10;
}
}
//如果分数太低,用户想再考一次的话。
QString str = "您的分数是:" + QString::number(scores) + "分,是否重新考试?";
int res = QMessageBox::information(this,"提示",str,QMessageBox::Yes | QMessageBox::No);
if(res == QMessageBox::Yes)
return;
else
close();
}
void ExamDialog::freshTime()
{
m_timeGo++;
QString min = QString::number(m_timeGo / 60);
QString sec = QString::number(m_timeGo % 60);
setWindowTitle("考试已用时:" + min + "分" + sec + "秒");
}
//examdialog.h新增加 方法
public:
bool hasNoSelect(); //判断题目是否有未完成的
private slots:
void getScore(); //获取考试成绩
//examdialog.cpp中的实现
//判断题目是否有未完成的
bool ExamDialog::hasNoSelect()
{
int radioSelects = 0;//单选题的计数
//通过按钮分组来判断,来检查按钮是否被选上了
for(int i = 0; i < 8; i++)
{
//如果有按钮选上了,计数+1
if( m_btnGroups[i]->checkedButton() )
radioSelects++;
}
//如果不足 8 说明有未完成的单选
if(radioSelects != 8)
return true;
//对多选按钮进行判断——只要有选中了的就行
int checkSelects = 0;
for(int i = 0; i < 4; i++)
{
if(m_checkBtns[i]->isChecked())
checkSelects++;
}
//如果多选只有1个或者没有那么就选的不完全,即又为完成的选项
if(checkSelects == 0 || checkSelects == 1)
return true;
//判断题——只有有一个选上了就ok
if(!m_radioA->isChecked() && !m_radioB->isChecked())
return true;
return false;
}
void ExamDialog::getScore()
{
if(hasNoSelect()){
QMessageBox::information(this,"提示","您有未完成的题目,请完成考试!","是");
return;
}
int scores = 0;
//对10个题进行判断
for(int i = 0; i < 10; i++)
{
//单选题计分
if( i < 8)
//与链表中的答案进行对比,如果正确,则+10
if(m_btnGroups[i]->checkedButton()->text() == m_answerList.at(i))
scores += 10;
//多项选择题计分
if(i == 8){
QString answer = m_answerList.at(i);
bool hasA = false;
bool hasB = false;
bool hasC = false;
bool hasD = false;
//contains:是否包含指定的字符串
//提取答案是否正确
if( answer.contains("A") ) hasA = true;
if( answer.contains("B") ) hasB = true;
if( answer.contains("C") ) hasC = true;
if( answer.contains("D") ) hasD = true;
//提取选择的按钮是否选中。选中了返回true,未选中false
bool checkA = m_checkBtns[0]->checkState();
bool checkB = m_checkBtns[1]->checkState();
bool checkC = m_checkBtns[2]->checkState();
bool checkD = m_checkBtns[3]->checkState();
//因为多选要所有答案都对才记分,如果状态不一致那么跳过本次循环
if( hasA != checkA) continue;
if( hasB != checkB) continue;
if( hasC != checkC) continue;
if( hasD != checkD) continue;
scores += 10;
}
//判断题计分
if(i == 9){
//选中项文本和答案文本进行对比
if(m_btnGroups[8]->checkedButton()->text() == m_answerList.at(i))
scores += 10;
}
}
//如果分数太低,用户想再考一次的话。
QString str = "您的分数是:" + QString::number(scores) + "分,是否重新考试?";
int res = QMessageBox::information(this,"提示",str,QMessageBox::Yes | QMessageBox::No);
if(res == QMessageBox::Yes)
return;
else
close();
}
private slots://槽函数——>实现具体的操作:参数4
void getScore(); //获取考试成绩
connect(submitBtn,SIGNAL(clicked(bool)),this,SLOT(getScore()));
//单选
m_btnGroups[i]->checkedButton();//成功返回 true,失败返回 false
//多选
m_checkBtns[i]->isChecked();//成功返回 true,失败返回 false
//判断
m_radioA->isChecked();//成功返回 true,失败返回 false
//contains:是否包含指定的字符串 承购返回:true;失败返回:false
answer.contains("A")
目前我们分别实现了——登录和获取分数的功能,但是两者没有任何联系,那么我们如何实现两个页面之间的交互呢?
驾校科目一考试系统——窗口交互。