播放效果图如下:
实现方法:
源代码如下:
#include "mylrc.h"
#include <QPainter>
#include <QTimer>
#include <QMouseEvent>
#include <QContextMenuEvent>
#include <QMenu>
MyLrc::MyLrc(QWidget *parent) :
QLabel(parent)
{
setWindowFlags(Qt::Window | Qt::FramelessWindowHint);
// 设置背景透明
setAttribute(Qt::WA_TranslucentBackground);
setText(tr("MyPlayer音乐播放器 --- yafeilinux作品"));
// 固定部件大小
setMaximumSize(800, 60);
setMinimumSize(800, 60);
// 歌词的线性渐变填充
linearGradient.setStart(0, 10);
linearGradient.setFinalStop(0, 40);
linearGradient.setColorAt(0.1, QColor(14, 179, 255));
linearGradient.setColorAt(0.5, QColor(114, 232, 255));
linearGradient.setColorAt(0.9, QColor(14, 179, 255));
// 遮罩的线性渐变填充
maskLinearGradient.setStart(0, 10);
maskLinearGradient.setFinalStop(0, 40);
maskLinearGradient.setColorAt(0.1, QColor(222, 54, 4));
maskLinearGradient.setColorAt(0.5, QColor(255, 72, 16));
maskLinearGradient.setColorAt(0.9, QColor(222, 54, 4));
// 设置字体
font.setFamily("Times New Roman");
font.setBold(true);
font.setPointSize(30);
// 设置定时器
timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), this, SLOT(timeout()));
lrcMaskWidth = 0;
lrcMaskWidthInterval = 0;
}
void MyLrc::paintEvent(QPaintEvent *)
{
QPainter painter(this);
painter.setFont(font);
// 先绘制底层文字,作为阴影,这样会使显示效果更加清晰,且更有质感
painter.setPen(QColor(0, 0, 0, 200));
painter.drawText(1, 1, 800, 60, Qt::AlignLeft, text());
// 再在上面绘制渐变文字
painter.setPen(QPen(linearGradient, 0));
painter.drawText(0, 0, 800, 60, Qt::AlignLeft, text());
// 设置歌词遮罩
painter.setPen(QPen(maskLinearGradient, 0));
painter.drawText(0, 0, lrcMaskWidth, 60, Qt::AlignLeft, text());
}
// 开启遮罩,需要指定当前歌词开始与结束之间的时间间隔
void MyLrc::startLrcMask(qint64 intervalTime)
{
// 这里设置每隔30毫秒更新一次遮罩的宽度,因为如果更新太频繁
// 会增加CPU占用率,而如果时间间隔太大,则动画效果就不流畅了
qreal count = intervalTime / 30;
// 获取遮罩每次需要增加的宽度,这里的800是部件的固定宽度
lrcMaskWidthInterval = 800 / count;
lrcMaskWidth = 0;
timer->start(30);
}
// 停止遮罩
void MyLrc::stopLrcMask()
{
timer->stop();
lrcMaskWidth = 0;
update();
}
// 两个鼠标事件处理函数实现了部件的拖动
void MyLrc::mousePressEvent(QMouseEvent *event)
{
if (event->button() == Qt::LeftButton)
offset = event->globalPos() - frameGeometry().topLeft();
}
void MyLrc::mouseMoveEvent(QMouseEvent *event)
{
if (event->buttons() & Qt::LeftButton) {
setCursor(Qt::PointingHandCursor);
move(event->globalPos() - offset);
}
}
// 实现右键菜单来隐藏部件
void MyLrc::contextMenuEvent(QContextMenuEvent *event)
{
QMenu menu;
menu.addAction(tr("隐藏"), this, SLOT(hide()));
menu.exec(event->globalPos());
}
// 定时器溢出时增加遮罩的宽度,并更新显示
void MyLrc::timeout()
{
lrcMaskWidth += lrcMaskWidthInterval;
update();
}
实现思路:使用一个QLabel类将其背景设置为透明,然后重新实现其重绘事件处理函数来自定义文本的显示,可以使用渐变填充来实现多彩的文字。然后再使用定时器,在已经绘制的歌词上面再绘制一个不断变宽的相同的歌词来实现动态遮罩效果。
绘制一共将相同歌词绘制3次,第一次绘制深黑色的歌词,第二次绘制渐变填充的歌词,它是要正常显示的歌词,第3次绘制的是用于实现遮罩的歌词,他的颜色与前两次不同,且宽度是变化的,有歌词的长度和定时器来决定他的宽度。这样就实现了在播放时歌词从左向右动态改变颜色的效果。