Qt浅谈之二:钟表(时分秒针)
一、简介
QT编写的模拟时钟,demo里的时钟只有时针和分针,在其基础上添加了秒针,构成了一个完整的时钟。能对2D绘图中坐标系统、平移变换(translate)、比例变换(scale)、旋转变换(rotate)、扭曲变换(shear)及其save()和restore()来保存和恢复坐标系的状态。
二、效果图
(1)时钟运行,秒针随系统时间移动。如图1。
三、详解
1、定时器
- Clock::Clock(QWidget *parent)
- : QWidget(parent)
- {
- QTimer *timer = new QTimer(this); //声明一个定时器
- //update()会自动产生重绘消息,调用paintEvent()
- connect(timer, SIGNAL(timeout()), this, SLOT(update())); //连接信号槽,定时器超时触发窗体更新
- timer->start(1000); //启动定时器
- setWindowTitle(tr("My Clock")); //设置窗体名称
- resize(300, 300); //设置窗体大小
- }
启动一个定时器,timer->start(1000);单位是ms,每一秒中update重绘一次窗口。
2、重绘事件
(1)先确定指针的颜色和形状大小。其坐标后面再确定。
- void Clock::paintEvent(QPaintEvent *event)
- {
- //下面三个数组用来定义表针的三个顶点,以便后面的填充
- static const QPoint hourHand[3] = {
- QPoint(3, 8),
- QPoint(-3, 8),
- QPoint(0, -40)
- };
- static const QPoint minuteHand[3] = {
- QPoint(3, 8),
- QPoint(-3, 8),
- QPoint(0, -70)
- };
- static const QPoint secondHand[3] = {
- QPoint(3, 8),
- QPoint(-3, 8),
- QPoint(0, -90)
- };//秒针
- //填充表针的颜色
- QColor hourColor(127, 0, 127); //分针颜色(第四个表示不透明度)
- QColor minuteColor(0, 127, 127, 191);
- QColor secondColor(127, 127, 0, 127);
- //...
- }
(2)qMin(width(), height());获取长宽的最小值,以确保绘制的时钟是圆形的,并使用painter.scale(side / 300.0, side / 300.0);来执行比例变换,实现缩放效果,比如窗口变成600,则600/300.0放大2倍。
painter.translate(width() / 2, height() / 2);将最标原点从(0, 0)移动到窗口中心则原来的原点最标就变成(-150, -150)。
坐标变换后具体的坐标如下图2:
- {
- int side = qMin(width(), height()); //绘制的范围(宽、高中最小值)
- QTime time = QTime::currentTime(); //获取当前的时间
- QPainter painter(this); //声明用来绘图用的painter
- painter.setRenderHint(QPainter::Antialiasing);//绘制的图像反锯齿
- painter.translate(width() / 2, height() / 2);//重新定位坐标起始点,把坐标原点放到窗体的中央
- painter.scale(side / 300.0, side / 300.0);//设定画布的边界,用窗体宽高的最小值来计算时钟的大小,防止窗体拉伸导致的时钟变形以及显示不全
- }
再看秒针的坐标(-3, 8)、(3, 8)、(0, 90)即确定了秒针的具体大小和位置。变换后的圆心在屏幕的中心。
(3)在坐标(-40, 30)处画出时间,随系统一秒更新一次。
- {
- painter.setPen(Qt::red); //填充时针,不需要边线所以NoPen
- QString timeStr= QTime::currentTime().toString(); //绘制当前的时间
- painter.drawText(-40,30,80,30,Qt::AlignHCenter | Qt::AlignTop, timeStr);
- }
(4)根据当前的时间,计算时针的移动角度6.0 * (time.minute() + time.second() / 60.0,使用rotate进行最标旋转,比如最标旋转30度如下图3所示。
painter.drawConvexPolygon(minuteHand, 3); 画出时针的三角形,如图2所示。然后painter.restore();恢复坐标到图2,不恢复的话无法确定分针的角度。
在分别计算分针的角度6.0 * (time.minute() + time.second() / 60.0)和秒针的角度6.0 * time.second()。
- {
- //...
- painter.setPen(Qt::NoPen); //填充时针,不需要边线所以NoPen
- painter.setBrush(hourColor); //画刷颜色设定
- painter.save(); //保存painter的状态,保存的是当前的坐标状态,如果不保存,画完之后坐标以改变不方便画下一个
- painter.rotate(30.0 * ((time.hour() + time.minute() / 60.0))); //将painter(的”视角“)根据时间参数转移(30° * (小时 + 分钟 / 60))
- painter.drawConvexPolygon(hourHand, 3); //填充时针的区域
- painter.restore();
- //后面的跟前面的类似,分别绘制了分针和秒针,及相应的刻度
- painter.setPen(Qt::NoPen);
- painter.setBrush(minuteColor);
- painter.save();
- painter.rotate(6.0 * (time.minute() + time.second() / 60.0)); //设旋转(角度 = 6° * (分钟 + 秒 / 60))
- painter.drawConvexPolygon(minuteHand, 3); //填充分针部分
- painter.restore();
- painter.setPen(Qt::NoPen);
- painter.setBrush(secondColor);
- painter.save();
- painter.rotate(6.0 * time.second()); //设置旋转(6° * 秒)
- painter.drawConvexPolygon(secondHand, 3); //设置填充
- painter.restore();
- //...
- }
(5)每次旋转6度,绘制长4个像素的直线,正点先不绘制,因为正点是8个像素的直线。接着绘制正点刻度和数字。不用保存画图的坐标,绘制都是旋转了360,回到了原来的最标系统中。
- { //...
- painter.setPen(minuteColor);
- for (int j = 0; j < 60; ++j) { //循环60次,绘制表盘(其实可以从1开始,到59,提高一点效率)
- if ((j % 5) != 0) //判断是否能被5整除(能被5整除表示是正点刻度,暂不绘制)
- painter.drawLine(0, -92, 0, -96); //不是正点刻度,绘制长4个像素的直线
- painter.rotate(6.0); //循环60次,每次旋转6度,所以不用save和restore
- }
- painter.setPen(hourColor); //下面画表示小时的刻度,此时要用到画笔(因为要划线)
- for (int i = 0; i < 12; ++i) {
- painter.drawLine(0, -88, 0, -96); //写上刻度数字
- if (i == 0) painter.drawText(-10,-88,20,20,Qt::AlignHCenter | Qt::AlignTop,QString::number(12));
- else painter.drawText(-10,-88,20,20,Qt::AlignHCenter | Qt::AlignTop,QString::number(i));
- painter.rotate(30.0);
- }
- }
(6)最后画上中心的小黑实心圆和外圈的空心圆,主要是计算下坐标。圆心分别为2和97。
- {
- //...
- painter.setPen(Qt::NoPen);
- painter.setBrush(secondColor);
- painter.save();
- painter.rotate(6.0 * time.second()); //设置旋转(6° * 秒)
- painter.drawConvexPolygon(secondHand, 3); //设置填充
- painter.restore();
- painter.setBrush(Qt::black);
- painter.drawEllipse(QPoint(0,0),2,2);
- painter.setBrush(Qt::NoBrush);
- painter.setPen(Qt::black);
- painter.drawEllipse(QPoint(0,0),97,97);
- //...
- }
四、总结
(1)时分秒也可以设置成四边形的,如
- static const QPoint hourHand[4] ={QPoint(0,10),QPoint(-1,-30),QPoint(0,-60),QPoint(1,-30)};
- static const QPoint minuteHand[4] ={QPoint(0,10),QPoint(-1,-40),QPoint(0,-70),QPoint(1,-40)};
- static const QPoint secondHand[4] ={QPoint(0,10),QPoint(-1,-60),QPoint(0,-90),QPoint(1,-60)};
- QColor hourColor(255,0,0);
- QColor minuteColor(0,127,127);
- QColor secondColor(0,0,255);
运行效果如下图4:
(2)解析顺序不是按代码正常顺序,请参看源码。
(3)源码已经打包上传到csdn上可登录下载(http://download.csdn.net/detail/taiyang1987912/7492657)。
(4)若需要沟通可以联系yang.ao@i-soft.com.cn
patch1
- void Clock::mouseMoveEvent(QMouseEvent *event)
- {
- if (m_pressMouse) { //移动窗口
- QPoint movePos = event->globalPos();
- move(movePos - m_movePoint);
- }
- }
- void Clock::contextMenuEvent(QContextMenuEvent *)
- {
- QCursor cur=this->cursor();
- QMenu *menu=new QMenu(this);
- QAction *deleteAction= new QAction(tr("关闭"), this);
- menu->addAction(deleteAction);
- connect(deleteAction, SIGNAL(triggered()), SLOT(close()));
- menu->exec(cur.pos());
- }
- void Clock::mousePressEvent(QMouseEvent *event)
- {
- if (event->button() == Qt::LeftButton) {
- m_pressMouse = true;
- }
- m_movePoint = event->globalPos() - pos(); //窗口移动距离
- }
- void Clock::mouseReleaseEvent(QMouseEvent * event)
- {
- m_pressMouse = false;
- }
实现效果如下图5所示:(无外边框)
http://blog.csdn.net/taiyang1987912/article/details/30272105
Qt浅谈之二:钟表(时分秒针)的更多相关文章
- Qt浅谈之二十七进程间通信之QtDBus
一.简介 DBus的出现,使得Linux进程间通信更加便捷,不仅可以和用户空间应用程序进行通信,而且还可以和内核的程序进行通信,DBus使得Linux变得更加智能,更加具有交互性. DB ...
- Qt浅谈之二十App自动重启及关闭子窗口
一.简介 最近因项目需求,Qt程序一旦检测到错误,要重新启动,自己是每次关闭主窗口的所有子窗口但有些模态框会出现问题,因此从网上总结了一些知识点,以备以后的应用. 二.详解 1.Qt结构 int ma ...
- Qt浅谈之二十App自动重启及关闭子窗口(六种方法)
一.简介 最近因项目需求,Qt程序一旦检测到错误,要重新启动,自己是每次关闭主窗口的所有子窗口但有些模态框会出现问题,因此从网上总结了一些知识点,以备以后的应用. 二.详解 1.Qt结构 int ma ...
- Qt浅谈之二十六图片滑动效果
一.简介 博客中发现有作者写的仿360的代码,觉得其中图片滑动的效果很有意思,特提取其中的代码.并加上类似mac的画面移动的动画效果. 二.详解 1.代码一:界面滑动(QWidget) (1)slid ...
- Qt浅谈之二十一log调试日志
一.简单介绍 近期因调试code时,想了解程序的流程,但苦于没有一个简易的日志记录,不停使用qDebug打印输出,而终于提交代码时得去多次删除信息打印,有时还会出现新改动的代码分不清是哪些部分.而使用 ...
- Qt浅谈之二十七进程间通信之QtDBus good
一.简介 DBus的出现,使得Linux进程间通信更加便捷,不仅可以和用户空间应用程序进行通信,而且还可以和内核的程序进行通信,DBus使得Linux变得更加智能,更加具有交互性. DB ...
- Qt浅谈内存泄露(总结)
Qt浅谈内存泄露(总结) 来源 http://blog.csdn.net/taiyang1987912/article/details/29271549 一.简介 Qt内存管理机制:Qt 在内部能够维 ...
- Qt浅谈之总结(整理)
Qt浅谈之总结(整理) 来源 http://blog.csdn.net/taiyang1987912/article/details/32713781 一.简介 QT的一些知识点总结,方便以后查阅. ...
- Android开发-浅谈架构(二)
写在前面的话 我记得有一期罗胖的<罗辑思维>中他提到 我们在这个碎片化 充满焦虑的时代该怎么学习--用30%的时间 了解70%该领域的知识然后迅速转移芳草鲜美的地方 像游牧民族那样.原话应 ...
随机推荐
- cordova-plugin-file-transfer 监听到下载成功,找不到文件 - 简书
原文:cordova-plugin-file-transfer 监听到下载成功,找不到文件 - 简书 下载成功后找不到下载文件 function download(fileEntry, uri) { ...
- 配置cordova的android开发环境(无android studio)
原文:配置cordova的android开发环境(无android studio) 趁元旦放假想试一下cordova,不想安装庞大的android studio,所以想最小化安装,居然花了一整天的时间 ...
- DevCloud项目管理与Leangoo使用对比
DevCloud(华为软件开发云)是集华为研发实践.前沿研发理念.先进研发工具为一体的研发云平台,面向开发者提供端到端的研发工具服务.项目管理是DevCloud服务之一,致力于为敏捷开发团队提供简单高 ...
- matlab 图像的保存
gcf:获取当前显示图像的句柄: 默认 plot 的 position 是 [232 246 560 420] 0. save >> A = randn(3, 4); >> B ...
- Spring+SpringMVC+MyBatis+easyUI
Spring+SpringMVC+MyBatis+easyUI 日常啰嗦 还好在第一篇文章里就列好了接下来的主线及要写的知识点,不然都不知道要写什么东西了,开篇里已经列了基础篇要讲svn和git的知识 ...
- Python 格式化输出 —— 小数转化为百分数
比如将 0.1234 转化为 12.34% 的形式: rate = .1234 print('%.2f%%' % (rate * 100)) 第一个百分号和 .2f 相连,表示浮点数类型保留小数点后两 ...
- Win7 32bit下一个hadoop2.5.1源代码编译平台的搭建各种错误遇到
从小白在安装hadoop困难和错误时遇到说起,同时,我们也希望能得到上帝的指示. 首先hadoop更新速度非常快,最新的是hadoop2.5.1,因此就介绍下在安装2.5.1时遇到的各种困难. 假设直 ...
- 关于iis里面 .net framework 版本的切换
然而,即便这样还是不够的 在iis上浏览网站会报以下错误: 这是什么原因呢?这是因为应用程序池中没有使用asp.net v4.0 网站程序是用v4.0写的,所以我们要用4.0的程序池版本,当前应用程序 ...
- Openstack+Kubernetes+Docker微服务实践
Openstack+Kubernetes+Docker微服务实践 ..... Openstack+Kubernetes+Docker微服务实践之路--选型 posted @ 2016-11-15 ...
- 零元学Expression Design 4 - Chapter 4 教你如何自制超炫笔刷
原文:零元学Expression Design 4 - Chapter 4 教你如何自制超炫笔刷 在Chapter 2 有稍微讲过Design内建笔刷的用法,本章将教大家如何自制独一无二的笔刷,并且重 ...