qt example – shapedclock

鉴于已经被我改的面目全非了,我也分不清原来是什么样子的了。
要做到透明效果,需要设置2个地方,一是背景色,二是把表盘无关的区域Mask掉。
shapedclock.h
#ifndef SHAPEDCLOCK_H
#define SHAPEDCLOCK_H

#include <QWidget>

class ShapedClock : public QWidget
{
    Q_OBJECT

public:
    ShapedClock(QWidget *parent = 0);
    QSize sizeHint() const;

protected:
    void mouseMoveEvent(QMouseEvent *event);
    void mousePressEvent(QMouseEvent *event);
    void paintEvent(QPaintEvent *event);
    void resizeEvent(QResizeEvent *event);

private:
    QPoint dragPosition;
};

#endif
shapedclock.cpp
#include <QtGui>

#include "shapedclock.h"

ShapedClock::ShapedClock(QWidget *parent)
    : QWidget(parent, Qt::FramelessWindowHint | Qt::WindowSystemMenuHint | Qt::WindowStaysOnTopHint)
      //窗口的种类,Frameless是透明背景必须, staysOnTop是窗口总在最前
{
    QTimer *timer = new QTimer(this);
    connect(timer, SIGNAL(timeout()), this, SLOT(update()));
    timer->start(100);
    //和原来一样的计时器

    move(1100,50);
    //这是我自己加的初始位置

    setAttribute( Qt::WA_TranslucentBackground, true);
    //背景透明

    QAction *quitAction = new QAction(tr("E&xit"), this);
    quitAction->setShortcut(tr("Ctrl+Q"));
    connect(quitAction, SIGNAL(triggered()), qApp, SLOT(quit()));
    addAction(quitAction);
    //加一个action,快捷键是Ctrl+Q,实现退出功能,我很奇怪的是设定快捷键他的类型是QString,难道可以自动识别?

    setContextMenuPolicy(Qt::ActionsContextMenu);
    setToolTip(tr("Drag the clock with the left mouse button.\n"
                  "Use the right mouse button to open a context menu."));
    setWindowTitle(tr("Shaped Analog Clock"));
    //鼠标悬置的文本提示和标题
}

void ShapedClock::mousePressEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) {
        dragPosition = event->globalPos() - frameGeometry().topLeft();
        event->accept();
    }
    //当鼠标点下的时候,得到鼠标相对窗口左上角点的相对坐标
}

void ShapedClock::mouseMoveEvent(QMouseEvent *event)
{
    if (event->buttons() & Qt::LeftButton) {
        move(event->globalPos() - dragPosition);
        event->accept();
    }
    //当鼠标移动的时候,用当时鼠标全局位置减去上面的相对坐标得到现在的角点坐标,并移动
}

void ShapedClock::paintEvent(QPaintEvent *)
{
    /* 后面的就和昨天类似了*/

    static const QPoint hourHand[3] = {
        QPoint(5, 8),
        QPoint(-5, 8),
        QPoint(0, -40)
    };
    static const QPoint minuteHand[3] = {
        QPoint(5, 8),
        QPoint(-5, 8),
        QPoint(0, -70)
    };
    static const QPoint secondHand[4] = {
	QPoint(1, 8),
	QPoint(-1, 8),
	QPoint(-1, -90),
	QPoint(1,-90)
    };
    static const QPoint background[4] = {
	QPoint( -100, -100),
	QPoint( -100, 100),
	QPoint( 100, 100),
	QPoint( 100, -100),
    };
    //多了一个画背景

    QColor hourColor(255, 0, 255);
    QColor minuteColor(0, 255, 255, 191);
    QColor secondColor(255, 255, 0);
    QColor BGColor(255, 255, 255, 5);    //这是背景色

    int side = qMin(width(), height());
    QTime time = QTime::currentTime();

    QPainter painter(this);
    painter.setRenderHint(QPainter::Antialiasing);
    painter.translate(width() / 2, height() / 2);
    painter.scale(side / 200.0, side / 200.0);

    painter.setPen(Qt::NoPen);
    painter.setBrush(BGColor);
    painter.save();
    painter.drawConvexPolygon(background,4);
    painter.restore();
    //画背景

    painter.setPen(Qt::NoPen);
    painter.setBrush(hourColor);

    painter.save();
    painter.rotate(30.0 * ((time.hour() + time.minute() / 60.0)));
    painter.drawConvexPolygon(hourHand, 3);
    painter.restore();

    painter.setPen(hourColor);

    for (int i = 0; i < 12; ++i) {
        painter.drawLine(88, 0, 96, 0);
        painter.rotate(30.0);
    }

    painter.setPen(Qt::NoPen);
    painter.setBrush(minuteColor);

    painter.save();
    painter.rotate(6.0 * (time.minute() + time.second() / 60.0));
    painter.drawConvexPolygon(minuteHand, 3);
    painter.restore();

    painter.setPen(minuteColor);

    for (int j = 0; j < 60; ++j) {
        if ((j % 5) != 0)
            painter.drawLine(92, 0, 96, 0);
        painter.rotate(6.0);
    }
    QPainter painter2(this);
    painter2.setRenderHint(QPainter::Antialiasing);
    painter2.translate(width() / 4, height() / 8 * 5);
    painter2.scale(side / 600.0, side / 600.0);
    painter2.setPen(Qt::NoPen);
    painter2.setBrush(secondColor);
    painter2.save();
    if(time.second()<30)
	painter2.rotate(6.0 * (time.second() + time.msec()/1000.0 ));
    else if(time.second()==30 && time.msec()<550)
	painter2.rotate(180 - 36.0 * time.msec()/100.0);

    painter2.drawConvexPolygon(secondHand,4);
    painter2.restore();

    painter2.setPen(secondColor);
    painter2.rotate(270);
    for (int j = 0; j < 30; ++j) {
	painter2.drawLine(92, 0, 96, 0);
        painter2.rotate(6.0);
    }

    QPainter painter3(this);
    painter3.setRenderHint(QPainter::Antialiasing);
    painter3.translate(width() / 4 * 3, height() / 8 * 3);
    painter3.scale(side / 600.0, side / 600.0);
    painter3.setPen(Qt::NoPen);
    painter3.setBrush(secondColor);
    painter3.save();
    if(time.second()>30)
	painter3.rotate(360 - (6.0 * (time.second()-30 + time.msec()/1000.0 )));
    else if (time.second()==0 && time.msec()<550)
	painter3.rotate(180 + 36.0 * time.msec()/100.0);
    painter3.drawConvexPolygon(secondHand,4);
    painter3.restore();
    painter3.setPen(secondColor);
    painter3.rotate(90);
    for (int j = 0; j < 30; ++j) {
	painter3.drawLine(92, 0, 96, 0);
        painter3.rotate(6.0);
    }
    //这些都和昨天一样了

}

void ShapedClock::resizeEvent(QResizeEvent * /* event */)
{
    int side = qMin(width(), height());
    QRegion maskedRegion(width() / 2 - side / 2, height() / 2 - side / 2, side,
                         side, QRegion::Ellipse);
    setMask(maskedRegion);
    //将表盘以外的Mask
}

QSize ShapedClock::sizeHint() const
{
    return QSize(200, 200);
    //默认尺寸
}
main.cpp
#include <QApplication>

#include "shapedclock.h"

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    ShapedClock clock;
    clock.show();
    return app.exec();
}
效果图
shapedclock
可执行文件,说明,这次我没有编译静态库的Qt,所以这个只有装了Qt的机器才能用-_-b,静态库的估计要10几M大小吧。
就昨天写的十几行就有2个小BUG,哎-_-b

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注