Shaped Clock Example▲

Widget masks are used to customize the shapes of top-level widgets by restricting the available area for painting. On some window systems, setting certain window flags will cause the window decoration (title bar, window frame, buttons) to be disabled, allowing specially-shaped windows to be created. In this example, we use this feature to create a circular window containing an analog clock.
Since this example's window does not provide a File menu or a close button, we provide a context menu with an Exit entry so that the example can be closed. Click the right mouse button over the window to open this menu.
ShapedClock Class Definition▲
The ShapedClock class is based on the AnalogClock class defined in the Analog Clock example. The whole class definition is presented below:
class
ShapedClock : public
QWidget
{
Q_OBJECT
public
:
ShapedClock(QWidget *
parent =
0
);
QSize sizeHint() const
override
;
protected
:
void
mouseMoveEvent(QMouseEvent *
event) override
;
void
mousePressEvent(QMouseEvent *
event) override
;
void
paintEvent(QPaintEvent *
event) override
;
void
resizeEvent(QResizeEvent *
event) override
;
private
:
QPoint dragPosition;
}
;
The paintEvent() implementation is the same as that found in the AnalogClock class. We implement sizeHint() so that we don't have to resize the widget explicitly. We also provide an event handler for resize events. This allows us to update the mask if the clock is resized.
Since the window containing the clock widget will have no title bar, we provide implementations for mouseMoveEvent() and mousePressEvent() to allow the clock to be dragged around the screen. The dragPosition variable lets us keep track of where the user last clicked on the widget.
ShapedClock Class Implementation▲
The ShapedClock constructor performs many of the same tasks as the AnalogClock constructor. We set up a timer and connect it to the widget's update() slot:
ShapedClock::
ShapedClock(QWidget *
parent)
:
QWidget(parent, Qt::
FramelessWindowHint |
Qt::
WindowSystemMenuHint)
{
QTimer *
timer =
new
QTimer(this
);
connect(timer, SIGNAL(timeout()), this
, SLOT(update()));
timer-&
gt;start(1000
);
QAction *
quitAction =
new
QAction(tr("E&xit"
), this
);
quitAction-&
gt;setShortcut(tr("Ctrl+Q"
));
connect(quitAction, SIGNAL(triggered()), qApp, SLOT(quit()));
addAction(quitAction);
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"
));
}
We inform the window manager that the widget is not to be decorated with a window frame by setting the Qt::FramelessWindowHint flag on the widget. As a result, we need to provide a way for the user to move the clock around the screen.
Mouse button events are delivered to the mousePressEvent() handler:
void
ShapedClock::
mousePressEvent(QMouseEvent *
event)
{
if
(event-&
gt;button() ==
Qt::
LeftButton) {
dragPosition =
event-&
gt;globalPos() -
frameGeometry().topLeft();
event-&
gt;accept();
}
}
If the left mouse button is pressed over the widget, we record the displacement in global (screen) coordinates between the top-left position of the widget's frame (even when hidden) and the point where the mouse click occurred. This displacement will be used if the user moves the mouse while holding down the left button. Since we acted on the event, we accept it by calling its accept() function.

The mouseMoveEvent() handler is called if the mouse is moved over the widget.
void
ShapedClock::
mouseMoveEvent(QMouseEvent *
event)
{
if
(event-&
gt;buttons() &
amp; Qt::
LeftButton) {
move(event-&
gt;globalPos() -
dragPosition);
event-&
gt;accept();
}
}
If the left button is held down while the mouse is moved, the top-left corner of the widget is moved to the point given by subtracting the dragPosition from the current cursor position in global coordinates. If we drag the widget, we also accept the event.
The paintEvent() function is given for completeness. See the Analog Clock example for a description of the process used to render the clock.
void
ShapedClock::
paintEvent(QPaintEvent *
)
{
static
const
QPoint hourHand[3
] =
{
QPoint(7
, 8
),
QPoint(-
7
, 8
),
QPoint(0
, -
40
)
}
;
static
const
QPoint minuteHand[3
] =
{
QPoint(7
, 8
),
QPoint(-
7
, 8
),
QPoint(0
, -
70
)
}
;
QColor hourColor(127
, 0
, 127
);
QColor minuteColor(0
, 127
, 127
, 191
);
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(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 &
lt; 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 &
lt; 60
; ++
j) {
if
((j %
5
) !=
0
)
painter.drawLine(92
, 0
, 96
, 0
);
painter.rotate(6.0
);
}
}
In the resizeEvent() handler, we re-use some of the code from the paintEvent() to determine the region of the widget that is visible to the user:
void
ShapedClock::
resizeEvent(QResizeEvent *
/* event */
)
{
int
side =
qMin(width(), height());
QRegion maskedRegion(width() /
2
-
side /
2
, height() /
2
-
side /
2
, side,