基于Qt的网络音乐播放器(五)实现歌词滚动显示

2020博客之星年度总评选进行中:请为74号的狗子投上宝贵的一票!
我的投票地址:点击为我投票
在这里插入图片描述



网络播放器系列:


1.思路和效果图

先说一下大体思路:
json解析出来的lyrics歌词(字符串形式:[00:18.26]毕竟我们深爱过\r\n[00:21.74]有你陪的日子里)中每句和每句之间有\n,所以我们利用这个换行符标识来分割字符串,放在list中,这样,我们得到的每一个字符串都是时间戳+歌词的形式,接下来,我们再继续解析单个字符串,用Qmap<int,QString>来保存,时间作为键值,歌词作为值,这样就构成了时间对应歌词的形式,然后通过QMediaPlayer类中positionChanged(qint64 duration)信号调用槽函数onDurationChanged(qint64 duration)来显示歌词,positionChanged信号会返回当前歌曲的进度,这个进度是毫秒级别的,将返回的时间与map的键值做对比,从而在适当的时间显示对应的歌词,歌词用Label显示。大体思路就是这样,然后具体实现的时候,还是有许多细节需要注意的,遇到再说,还有就是上面提到的函数等等,在前面的文章中已经建立,下面的代码是直接写实现,如果不知道在哪里写,可查看前面几篇文章。

在这里插入图片描述


2.歌词的解析与存储

mainwindow.h

//类成员
QMap<int,QString> lrcMap;

mianwindow.cpp

	if (valuedataObject.contains("lyrics")) //lrc
	{
		QJsonValue play_url_value = valuedataObject.take("lyrics");
		if (play_url_value.isString())
		{
			QString play_lrcStr = play_url_value.toString();
			if (play_urlStr != "")
			{
				if (play_lrcStr != "")
				{	//将整个歌词给s
					QString s = play_lrcStr;
					// s1 用列表的形式保存每一句歌词
					QStringList s1 = s.split("\n");
					for (int i = 3; i < s1.size() - 1; i++)
					{
						QString ss1 = s1.at(i);
						//歌词中开头有一些是无意义的字符,用正则表达式判断,只保存包含有时间戳的字符串。
						QRegExp ipRegExp = QRegExp("\\[\\d\\d\\S\\d\\d\\S\\d\\d\\]");
						//若包含则返回flase
						bool match = ipRegExp.indexIn(ss1);
						if (match == false)
						{
							//时间解析格式(分*60+秒)*100+厘秒
							int s_1 = ss1.mid(1, 2).toInt();      //分
							int s_2 = ss1.mid(4, 2).toInt();      //秒
							int s_3 = ss1.mid(7, 2).toInt();      //厘秒
							int s_count = (s_1 * 60 + s_2) * 100 + s_3;   //规定写法
							int lrctime = s_count;
							QString lrcstr = ss1.mid(10);
							//用Qmap来保存
							lrcMap.insert(lrctime, lrcstr);
						}
					}
				}
				else
				{
					//没有歌词;
				}
			}
		}
	}

由于json返回的歌词里面的时间表示是[02:12.85](.后面的数字表示85/100秒)这种形式,而positionChanged返回的是以毫秒的形式,为了能够做对比,我们规定一种通用的表示方法:

  • 时间解析格式(分*60+秒)*100+厘秒,这个厘秒就是小数点后面的数。

3.onDurationChanged()

void MainWindow::onPositionChanged(qint64 position)
{
    //时间标签得法
    //(分*60+秒)*100+厘秒
    int pos = position/10;
    QMap<int, QString>::iterator iter = lrcMap.begin();
        while (iter != lrcMap.end())
        {
            if(pos-50<=iter.key()&& pos+50>=iter.key())
            {
                    int j=0;
                    if(iter != lrcMap.begin())
                    {
                        iter--;
                        ui->label_20->setText(iter.value());
                        j++;
                    }
                    if(iter != lrcMap.begin())
                    {
                        iter--;
                        ui->label_19->setText(iter.value());
                        j++;
                    }

                    if(iter != lrcMap.begin())
                    {
                        iter--;
                        ui->label_6->setText(iter.value());
                        j++;
                    }
                    for(;j>0;j--)
                    {
                        iter++;
                    }
               //中间
               ui->label_21->setText(iter.value());
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_22->setText(iter.value());
               }
               else
               {
                   ui->label_22->setText("");
                   return;
               }
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_23->setText(iter.value());
               }
               else
               {
                   ui->label_23->setText("");
                   return;
               }
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_24->setText(iter.value());
               }
               else
               {
                   ui->label_24->setText("");
                   return;
               }
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_25->setText(iter.value());
               }
               else
               {
                   ui->label_25->setText("");
                   return;
               }
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_26->setText(iter.value());
               }
               else
               {
                   ui->label_26->setText("");
                   return;
               }
               iter++;
               if(iter != lrcMap.end())
               {
                   ui->label_27->setText(iter.value());
               }
               else
               {
                   ui->label_27->setText("");
                   return;
               }
            }
            iter++;
        }
}

label_21等10标签用于显示歌词,label_21匹配当前时间显示的歌词,并且把该歌词前面的歌词和后面的歌词分别发送给其他对应的标签。这样就实现了动态效果。


4.总结

虽然代码很少,但是完成这个还是用了很长时间实现,反复修改,反复崩溃,没实现前,觉得这个功能,要是能实现多好,实现了后又觉得自己写的太简单了,而且效果有一点僵硬,并没有人家QQ 酷狗啊什么,歌词是慢慢往上滑,我这个是直接显示,后面会研究研究怎么滑动显示,让人看见更加平滑。学习就是这样,来回不断重复,对待问题的看法,逻辑的推理,思维的跳跃,从不会到实现,再到不满足再实现。可执行文件不是你的财富,修改过程中的经验才是,我是花狗,一名苟且偷生的大专生,我们下篇见。


  • 100
    点赞
  • 229
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 42
    评论
### 回答1: 音乐播放器频谱设计是一种十分吸引眼球的显示方式,可以通过色彩、动态和形状等多种因素展示音乐的节奏与情感。基于Qt音乐播放器频谱设计,可以让我们更加方便地实现这种设计效果。 首先,Qt是一种跨平台的C++应用程序开发框架,提供了很多高效的图形组件、界面布局和事件处理等功能。通过使用Qt提供的基础组件,我们可以轻松实现音乐播放器的界面设计和控制。例如,通过QMediaPlayer组件,我们可以加载音乐文件,并且控制音乐的播放、暂停、停止等操作。同时,通过QAudioInput/QAudioOutput组件,我们也可以实现音乐频谱的采集和展示等功能。 基于Qt音乐播放器频谱设计需要具备如下的功能: 1. 频谱显示:在界面上展示出音乐的频谱,包括音量大小和声波形状,可以按照不同的颜色和形状进行设置,以增强视觉效果。 2. 歌词显示:在频谱下方展示歌词歌词随着歌曲播放自动滚动,进行实时的同步显示。 3. 频率调整:通过调整频率,可改变音乐频率或人声音高音低,以满足不同的听众需求。 4. 播放列表:可以展示音乐播放列表,并支持导入、删除、重命名等操作。 总体来说,基于Qt音乐播放器频谱设计是一种富有创意与互动性的设计方式,可以为我们提供更加丰富的音乐体验。随着技术的不断创新和突破,相信QT音乐频谱播放器会在不久的将来发展为一种成熟的音频处理工具。 ### 回答2: Qt是一款跨平台的应用程序框架,具备高度的灵活性和可定制性。基于Qt开发的音乐播放器可以实现高效、稳定及用户友好的音乐播放和管理。频谱设计是音乐播放器的必备功能之一,可以通过图形界面将音乐频谱呈现出来。 在设计频谱时,需要先获取音频数据。可以通过Qt的多媒体模块,调用音频输入设备获取音频数据,然后进行FFT变换得到频域信息,从而计算出频谱。通过Qt的绘图工具,可以将频谱以曲线或柱状图的形式呈现在播放器窗口上。 为了提高用户体验,可在频谱的基础上加入歌词显示功能。歌词文件可以用LRC格式保存,然后在播放器中读取并解析。利用Qt的图形界面设计工具,可以将歌词滚动、静态或卡拉OK等多种形式呈现在播放器窗口上,帮助用户更好地理解和感受音乐。 同时,为了让用户能够更好地控制播放器与频谱,可以设计一个交互式的界面,包括可调节的音量、循环播放、随机播放、快进退等功能。这些功能和界面可以通过Qt的信号槽机制实现,使得音乐播放器更加易用和具有吸引力。 总之,基于Qt音乐播放器频谱设计不仅需要具备高效、稳定性,同时需要用户友好、交互式等特点,以能够满足用户多样化的需求和期望。 ### 回答3: 基于Qt音乐播放器频谱设计是一个非常有趣和有挑战性的项目。这个项目主要需要用到Qt的图形库和音频处理库来实现音乐播放器、频谱分析和歌词显示功能。 首先,我们需要实现一个基本的音乐播放器,能够读取并播放各种音频文件,并且具有基本的控制功能,例如播放、暂停、停止、快进、倒退等等。这部分的实现比较简单,只需要调用Qt提供的音频处理库即可。 接下来,我们需要实现频谱分析功能。这部分需要使用FFT算法对音频数据进行频谱分析,然后将结果显示为频谱图。我们可以使用Qt提供的绘图工具来实现频谱图的绘制。 最后,我们需要实现歌词显示功能。这部分的实现比较复杂,需要将歌词文件读入内存并进行解析,然后根据歌曲进度定位到相应的歌词位置,并将其显示出来。这一部分需要使用Qt提供的文本处理库来解析歌词,同时需要使用多线程来确保整个程序的流畅性。 综上所述,基于Qt音乐播放器频谱设计带歌词显示功能是一个非常有挑战性和有趣的项目,需要具备一定的Qt编程技能和音频处理知识。通过这个项目的实践,不仅可以提高编程能力,还可以深入了解音频的原理和应用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 42
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

花狗Fdog

你的鼓励将是我创作的最大动力~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值