在用ListBox显示歌词的时候,可以显示多行,同时可以控制每一行显示的样式等等。控制显示样式是在它的DrawItem事件中来控制的。首先要先将ListBox的DrawMode属性设置为OwnerDrawVariable或OwnerDrawFixed。ListBox有个ItemHeight属性,在DrawMode设置为Normal时,这个属性是不可设置的,并且其值是根据当前字体进行计算获得的。只有当DrawMode设置为OwnerDrawVariable或OwnerDrawFixed时,设置ItemHeight才生效。DrawMode属性中有三个成员变量:

属性值 说明
Normal 组件里面的所有元素都是操作系统来绘制,并且所有组件大小相同。(默认值)
OwnerDrawFixed 组件里面的所有元素都是由我们手动绘制,并且大小相同
OwnerDrawVariable 组件里面的所有元素都是由我们手动绘制,大小可以不相同

DrawItem事件中,的参数e里面有许多的属性值,包括获取前景色和背景色,获取当前绘制行(焦点行)从0开始的索引,获取字体、画笔、范围、状态等等很多的内容。这次试用的是索引,字体,画笔和范围。

在定时器中先为ListBox要显示的歌词进行赋值,代码如下:

<span style="font-size:18px;">private void timer1_Tick(object sender, EventArgs e)
{
//也几行可以写在定时器外面
int totalHeight = listShowSongLrc.Height;
//设置显示的每一项的高度
int height = listShowSongLrc.ItemHeight = 30;
//需要显示多少行歌词
int num = totalHeight / height;
//显示歌词为奇数行,方便获取当前歌词行
num = num%2==1?num:num-1; if (songLrc != null)
{
#region MyRegion
for (int i = 0; i < songLrc.Length - 1; i++)
{
//歌曲当前位置
string currenPosition = axMediaPlayer.Ctlcontrols.currentPositionString; //歌曲唱的过程中,显示当前歌词的前后各num/2行
if (CheckTime(currenPosition, songLrc[i]) && CheckTime(songLrc[i + 1], currenPosition))
{
//清除只能够放在if里面,放在if外面基本上看不到歌词
listShowSongLrc.Items.Clear(); for (int x = (i - num / 2); x <= (i + num / 2); x++)
{
listShowSongLrc.Items.Add(x < 0 || x >= songLrc.Length ? "" : GetSongLrc(x));
}
}
//歌曲唱完以后,后面显示为空
if (CheckTime(currenPosition, songLrc[songLrc.Length - 1]))
{
listShowSongLrc.Items.Clear();
for (int x = -num / 2; x <= num / 2; x++)
{
listShowSongLrc.Items.Add(x <= 0 ? GetSongLrc(songLrc.Length - 1 + x) : "");
}
}
//歌词还没有开始显示的时候,中间显示歌名,后面显示前几行歌词
if (CheckTime(songLrc[0], currenPosition))
{
listShowSongLrc.Items.Clear();
for (int x = -num / 2; x <= num / 2; x++)
{
listShowSongLrc.Items.Add(x > 0 ? GetSongLrc(x) : (x < 0 ? "" : listShowSong.Items[listSong.IndexOf(axMediaPlayer.URL)]));
}
}
//让每一项获得焦点,调用歌词绘制事件DrawItem
for (int j = 0; j < listShowSongLrc.Items.Count; j++)
{
listShowSongLrc.SelectedIndex = j;
}
}
#endregion
}
else
{
listShowSongLrc.Items.Clear();
for (int x = 0; x < num; x++)
{
listShowSongLrc.Items.Add(x != num / 2 ? "" : "---未 找 到 歌 词---");
}
}
}</span>

上面涉及到两个方法,比较当前时间和歌词的时间大小方法。歌词中,是按照时间来进行一个排序的,每一行歌词前面是时间,然后用’|‘和歌词分割,可以百度下一首歌词(.lrc)来看看

<span style="font-size:18px;">        private bool CheckTime(string str1, string str2)
{
return string.CompareOrdinal(str1, 0, str2, 0, str2.Length) > 0;
}</span>

获取当前时间的歌词方法,歌词是存放在一个数组中,传递过来当前要获取的行数,进行歌词的分割,然后返回

<span style="font-size:18px;"> private string GetSongLrc(int i)
{
return songLrc[i].Split(new char[] { '|' }, StringSplitOptions.RemoveEmptyEntries)[1];
}</span>

接下来便是ListBox中歌词的绘制,是通过画图的方式来实现的(GDI+)

<span style="font-size:18px;">private void listShowSongLrc_DrawItem(object sender, DrawItemEventArgs e)
{
//获取当前绘制的行的索引
int index = e.Index;
Graphics g = e.Graphics;
//得到每一项的绘制区域大小
Rectangle bound = e.Bounds;
//得到当前项的文本内容
string text = listShowSongLrc.Items[index].ToString(); //判断当前选择的项是正在唱的歌词,也就是中间一行歌词
if (index == listShowSongLrc.Items.Count / 2)
{//如果当前行为选中行。
//绘制选中时要显示的蓝色边框,实际不需要就注释掉了
// g.DrawRectangle(Pens.Blue, bound.Left, bound.Top, bound.Width - 1, bound.Height - 1);
//绘制边框后,里面的矩形框大小改变,故重新定义一个,如果没有绘制边框就不需要重新定义
Rectangle rect = new Rectangle(bound.Left - 1, bound.Top - 1,
bound.Width - 2, bound.Height - 2);
//绘制选中时要显示的蓝色背景。可以选中其它色,此处省略了背景绘制
// g.FillRectangle(Brushes.Blue, rect);
//定义一个字体,是用来绘制显示的当前歌词文本。
Font font = new System.Drawing.Font("微软雅黑", 18, FontStyle.Bold & FontStyle.Italic);
//绘制歌词,颜色为红色
TextRenderer.DrawText(g, text, font, rect, Color.Red,
TextFormatFlags.VerticalCenter);
}
else
{
//定义一个颜色为白色的画刷
using (Brush brush = new SolidBrush(Color.White))
{
g.FillRectangle(brush, bound);//绘制背景色。
}
//填充字体,字体的颜色为黑色
TextRenderer.DrawText(g, text, this.Font, bound, Color.Black,
TextFormatFlags.VerticalCenter | TextFormatFlags.Left);
}
}</span>

至此,整个歌词的自动绘制就基本上完成了,显示效果除了当前行是用红色一次性显示完成意外,其余的滚动和酷狗音乐中歌词写真基本上是一样的。

C#_音乐播放器_用ListBox显示歌词的更多相关文章

  1. pygame系列_原创百度随心听音乐播放器_完整版

    程序名:PyMusic 解释:pygame+music 之前发布了自己写的小程序:百度随心听音乐播放器的一些效果图 你可以去到这里再次看看效果: pygame系列_百度随心听_完美的UI设计 这个程序 ...

  2. Lily_music 网页音乐播放器 -可搜索(附歌词联动播放效果解说)

    博客地址:https://ainyi.com/59 写在前面 这是我今年(2018)年初的小项目,当时也是手贱,不想用别的播放器,想着做一个自己的网页播放器,有个歌曲列表.可关键词搜索.歌词滚动播放的 ...

  3. linux笔记_20150417_ubuntu 常见问题_文件_音乐播放器

    最近在学习ubuntu的过程中,遇到了一些问题,就记下来了它的解决办法.以希望对你也有用. ),至少保证周围局域网内用户可以访问.至于配置文件,内容比较少,反正对我来讲能用就ok了~不知道会不会很弱 ...

  4. Android——音乐播放器完善——进度条显示当前播放进度,加可拖动进度条(未待解决完问题)

    效果: 问题:可拖动进度条随进度条移动时,会致使音乐卡顿(待解决) xml <?xml version="1.0" encoding="utf-8"?&g ...

  5. python 开发在线音乐播放器-简易版

    在线音乐播放器,使用python的Tkinter库做了一个界面,感觉这个库使用起来还是挺方便的,音乐的数据来自网易云音乐的一个接口,通过urllib.urlopen模块打开网址,使用Json模块进行数 ...

  6. 【.NET 与树莓派】用 MPD 制作数字音乐播放器

    树莓派的日常家居玩法多多,制作一台属于自己的数字音乐播放机是其中的一种.严格上说,树莓派是没有声卡的,其板载的 3.5 mm 音频孔实际是通过 PWM 来实现音频输出的(通过算法让PWM信号变成模拟信 ...

  7. 基于Qt Phonon模块实现音乐播放器

    这次使用Qt实现的是一个本地音乐播放器,可以播放下载在计算机本地的音乐,提供了添加歌曲,歌曲列表,清空列表的功能.默认歌曲列表循环播放.音乐播放的实现主要依赖的是Qt 的多媒体框架phonon.该音乐 ...

  8. 在线音乐播放器-----酷狗音乐api接口抓取

    首先身为一个在线音乐播放器,需要前端和数据库的搭配使用. 在数据库方面,我们没有办法制作,首先是版权问题,再加上数据量.所以我们需要借用其他网络播放器的数据库. 但是这些在线播放器,如百度,酷狗,酷我 ...

  9. 记一次酷狗音乐API的获取,感兴趣的可以自己封装开发自己的音乐播放器

    1.本教程仅供个人学习用,禁止用于任何的商业和非法用途,如涉及版权问题请联系笔者删除. 2.随笔系作者原创文档,转载请注明文档来源:http://www.cnblogs.com/apresunday/ ...

随机推荐

  1. sql批量修改插入数据

    1.批量修改 select 'update 读者库 set 单位代码='''+新单位代码+''' where 单位代码='''+单位代码+'''' from 读者单位 ,)<'L' and is ...

  2. MapReduce框架Partitioner分区方法

    前言:对于二次排序相信大家也是似懂非懂,我也是一样,对其中的很多方法都不理解诶,所有只有暂时放在一边,当你接触到其他的函数,你知道的越多时你对二次排序的理解也就更深入了,同时建议大家对wordcoun ...

  3. svn的使用!!!

    1.SVN:subversion(子级源代码版本控制管理软件) 2.SVN的作用 (1)避免开发同一项目不会出现代码覆盖. (2)同一文件可以创建许多不同的版本,并可以随时查看不同版本的内容. (3) ...

  4. 记录C++学习历程

    从今天开始学习C++,将学习中遇到的问题,以及解决方案记录在这个博客里. 函数 1.C++函数声明(原型) 函数原型跟函数的定义在返回值类型,函数名,参数上必须完全一致. 2.程序的内存区域:全局数据 ...

  5. css 串联选择器和后代选择器

    串联选择器:作用在同一个标签上 <div class=”a” id ="qq"><span>look at the color</span>&l ...

  6. 容易被忽略的事----sql语句中select语句的执行顺序

    关于Sql中Select语句的执行顺序,一直很少注意这个问题,对于关键字的使用也很随意,至于效率问题,因为表中的数据量都不是很大,所以也不是很在意. 今天在一次面试的时候自己见到了,感觉没一点的印象, ...

  7. linux下的声卡驱动架构

    1.linux下的声卡驱动架构主要分为OSS架构和ALSA架构. 2.OSS架构 OSS全称是Open Sound System,叫做开放式音频系统,这种早期的音频系统这种基于文件系统的访问方式,这意 ...

  8. jquery mobile最棘手的一个问题

    大多数jquery mobile开发的妹子们都碰到过这个问题: 如何调用loading效果   这里给出一段代码,赶紧练手吧. //显示loading function showLoading(){ ...

  9. [笔记]--Oracle修改SGA大小

    1.通过spfile修改 (1)使用SYS用户以SYSDBA身份登录系统 (2)查看修改前sga_max_size,sga_target大小 show parameter sga_max_size; ...

  10. C 网页压力测试器

    引言 <<独白>> 席慕蓉 节选一 把向你借来的笔还给你吧. 一切都发生在回首的刹那. 我的彻悟如果是缘自一种迷乱,那么,我的种种迷乱不也就只是因为一种彻悟? 在一回首间,才忽 ...