首页 > 其他分享 >基于Qt的网络音乐播放器(五)实现歌词滚动显示

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

时间:2023-05-25 15:34:46浏览次数:50  
标签:播放器 Qt setText 歌词 iter label ui value




文章目录

  • 1.思路和效果图
  • 2.歌词的解析与存储
  • 3.onDurationChanged()
  • 4.总结



网络播放器系列:

  • qt 布局和样式表
  • 基于Qt的网络音乐播放器(一)添加音频文件,播放音乐,更新进度条
  • 基于Qt的网络音乐播放器(二)切换歌曲,调节音量,调节语速,暂停
  • 基于Qt的网络音乐播放器(三)通过酷狗音乐的api接口,返回json格式歌曲信息(播放地址,歌词,图片)
  • 基于Qt的网络音乐播放器(四)酷狗API接口获取歌曲的搜索列表和歌曲的播放
  • 项目已上传GitHub(更新中),点击获取

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显示。大体思路就是这样,然后具体实现的时候,还是有许多细节需要注意的,遇到再说,还有就是上面提到的函数等等,在前面的文章中已经建立,下面的代码是直接写实现,如果不知道在哪里写,可查看前面几篇文章。

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


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 酷狗啊什么,歌词是慢慢往上滑,我这个是直接显示,后面会研究研究怎么滑动显示,让人看见更加平滑。学习就是这样,来回不断重复,对待问题的看法,逻辑的推理,思维的跳跃,从不会到实现,再到不满足再实现。可执行文件不是你的财富,修改过程中的经验才是,我是花狗,一名苟且偷生的大专生,我们下篇见。



标签:播放器,Qt,setText,歌词,iter,label,ui,value
From: https://blog.51cto.com/u_14770531/6348298

相关文章

  • Qt读取qss文件失败或qss不生效解决方案
            最近在写qt加载样式表的博文,发现qss文件要么打开失败,要么加载成功,但是不生效,经过一番搜索也是算解决了这个问题。读取qss失败:读取文件的方式有两种,一种是绝对路径,一种是相对路径://绝对路径C:\\Users\\fdog\\Desktop\\sheet.qss//相对路径./lib/sheet.qss出现错......
  • Qt父窗口与子窗口数据交互(用拾色器举例)
    文章目录一.效果图二.实现1.在子窗口中声明信号2.在主窗口中声明并实现槽函数,并进行信号与槽的绑定3.在子窗口中发送信号(emit),并完成其控件的相应4.在子窗口的构造函数中传入需要的主窗口数据一.效果图二.实现首先我们创建主窗口和子窗口,并拖动控件,完成基本界面。这里的布局以及......
  • Qt正则表达式类QRegExp(附检验小程序)
           在许多场景中,我们需要验证用户输入的数据是否有效,或者是查找并修改文本,或者是提取指定数据,为此,相对于Qstring的一些函数,QT提供了一个更加强大的类——QRegExp,使用函数配合正则表达式来操作字符串,QRegExp可以进行下面的操作,并附带检验小程序,可在文末下载。一.正则表达......
  • 基于Qt的音乐播放器(三)通过酷狗音乐的api接口,返回json格式歌曲信息(播放地址,歌词,图片)
    2020博客之星年度总评选进行中:请为74号的狗子投上宝贵的一票!我的投票地址:点击为我投票文章目录前言1.获取歌曲搜索列表api接口2.获取单个歌曲详细信息包括歌词3.总结前言首先说明,本教程仅供个人学习,研究使用,禁止用于任何的商业和非法用途。(手动狗头)之所以要研究这个,是因为我想......
  • 【转载】vs设置qt应用程序logo
    1、下载一个ico,拷贝到主函数工程目录下 2、新建一个文件如logo.rc,内容如下:IDI_ICON1ICONDISCARDABLE“logo.ico”3、修改主函数工程文件Main.vcxproj在文件最后添加<ItemGroup><ResourceCompileInclude="logo.rc"/></ItemGroup> 4、生成工程后即可看到ex......
  • MQTT入门DEMO(Java语言)
    目录快速开始准备下载及安装第一次安装EMQX第一次运行EMQX客户端代码快速开始准备MQTT简介EMQX简介下载及安装第一次安装EMQX版本选择EMQX支持多种操作系统,请选择合适您的版本下载。下载地址:https://www.emqx.io/cn/downloads#broker在MicrosoftWindows下安装目前EMQX......
  • QTLtools 协变量文件说明(covariate)
    协变量格式如下所示:注意事项:缺失值用NA表示;接受定性和定量的协变量;定量协变量用数值型表示;定性协变量用非数值型表示,类似于上图的A,B,C。定量和定性的判断依据是有没有等级关系,比如年龄属于定量,比如地区属于定性。需要注意的是,只要是定性的变量,都不能用数值表示,不然会被当成......
  • MQTT实现(Java语言)
    下面是我们Java语言实现的MQTT服务的发布/订阅1、添加Maven依赖<dependency><groupId>org.eclipse.paho</groupId><artifactId>org.eclipse.paho.client.mqttv3</artifactId><version>1.1.1</version></dependency......
  • QT输出彩色log
    要输出彩色信息有点类似于html的语法,即在要输出的文字前加上一段颜色指令。指令格式如下\033[*m这里的*就是转义字符,例如我们要输出一段绿色的文字,则qDebug()<<"\033[32m"<<"Hello!";即在输出文字前,先输出一个颜色指令。注意这个指令对后续的输出都会生效,如果想关掉颜色......
  • Qt Creator按顺序编译多个子项目
    0.环境Qt5.3.2mingw482_321.创建子项目这个子项目类似于VisualStudio中解决方案一样的存在,用于管理多个其他子项目。点击文件->新建文件或项目->其他项目->子项目项目->Choose:之后下一步,选择需要的编译器,我这里只有mingw,若有msvc编译器也可选择msvc,不同的编译器生成的静......