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

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

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

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

  1. <span style="font-size:18px;">private void timer1_Tick(object sender, EventArgs e)
  2. {
  3. //也几行可以写在定时器外面
  4. int totalHeight = listShowSongLrc.Height;
  5. //设置显示的每一项的高度
  6. int height = listShowSongLrc.ItemHeight = 30;
  7. //需要显示多少行歌词
  8. int num = totalHeight / height;
  9. //显示歌词为奇数行,方便获取当前歌词行
  10. num = num%2==1?num:num-1;
  11.  
  12. if (songLrc != null)
  13. {
  14. #region MyRegion
  15. for (int i = 0; i < songLrc.Length - 1; i++)
  16. {
  17. //歌曲当前位置
  18. string currenPosition = axMediaPlayer.Ctlcontrols.currentPositionString;
  19.  
  20. //歌曲唱的过程中,显示当前歌词的前后各num/2行
  21. if (CheckTime(currenPosition, songLrc[i]) && CheckTime(songLrc[i + 1], currenPosition))
  22. {
  23. //清除只能够放在if里面,放在if外面基本上看不到歌词
  24. listShowSongLrc.Items.Clear();
  25.  
  26. for (int x = (i - num / 2); x <= (i + num / 2); x++)
  27. {
  28. listShowSongLrc.Items.Add(x < 0 || x >= songLrc.Length ? "" : GetSongLrc(x));
  29. }
  30. }
  31. //歌曲唱完以后,后面显示为空
  32. if (CheckTime(currenPosition, songLrc[songLrc.Length - 1]))
  33. {
  34. listShowSongLrc.Items.Clear();
  35. for (int x = -num / 2; x <= num / 2; x++)
  36. {
  37. listShowSongLrc.Items.Add(x <= 0 ? GetSongLrc(songLrc.Length - 1 + x) : "");
  38. }
  39. }
  40. //歌词还没有开始显示的时候,中间显示歌名,后面显示前几行歌词
  41. if (CheckTime(songLrc[0], currenPosition))
  42. {
  43. listShowSongLrc.Items.Clear();
  44. for (int x = -num / 2; x <= num / 2; x++)
  45. {
  46. listShowSongLrc.Items.Add(x > 0 ? GetSongLrc(x) : (x < 0 ? "" : listShowSong.Items[listSong.IndexOf(axMediaPlayer.URL)]));
  47. }
  48. }
  49. //让每一项获得焦点,调用歌词绘制事件DrawItem
  50. for (int j = 0; j < listShowSongLrc.Items.Count; j++)
  51. {
  52. listShowSongLrc.SelectedIndex = j;
  53. }
  54. }
  55. #endregion
  56. }
  57. else
  58. {
  59. listShowSongLrc.Items.Clear();
  60. for (int x = 0; x < num; x++)
  61. {
  62. listShowSongLrc.Items.Add(x != num / 2 ? "" : "---未 找 到 歌 词---");
  63. }
  64. }
  65. }</span>

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

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

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

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

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

  1. <span style="font-size:18px;">private void listShowSongLrc_DrawItem(object sender, DrawItemEventArgs e)
  2. {
  3. //获取当前绘制的行的索引
  4. int index = e.Index;
  5. Graphics g = e.Graphics;
  6. //得到每一项的绘制区域大小
  7. Rectangle bound = e.Bounds;
  8. //得到当前项的文本内容
  9. string text = listShowSongLrc.Items[index].ToString();
  10.  
  11. //判断当前选择的项是正在唱的歌词,也就是中间一行歌词
  12. if (index == listShowSongLrc.Items.Count / 2)
  13. {//如果当前行为选中行。
  14. //绘制选中时要显示的蓝色边框,实际不需要就注释掉了
  15. // g.DrawRectangle(Pens.Blue, bound.Left, bound.Top, bound.Width - 1, bound.Height - 1);
  16. //绘制边框后,里面的矩形框大小改变,故重新定义一个,如果没有绘制边框就不需要重新定义
  17. Rectangle rect = new Rectangle(bound.Left - 1, bound.Top - 1,
  18. bound.Width - 2, bound.Height - 2);
  19. //绘制选中时要显示的蓝色背景。可以选中其它色,此处省略了背景绘制
  20. // g.FillRectangle(Brushes.Blue, rect);
  21. //定义一个字体,是用来绘制显示的当前歌词文本。
  22. Font font = new System.Drawing.Font("微软雅黑", 18, FontStyle.Bold & FontStyle.Italic);
  23. //绘制歌词,颜色为红色
  24. TextRenderer.DrawText(g, text, font, rect, Color.Red,
  25. TextFormatFlags.VerticalCenter);
  26. }
  27. else
  28. {
  29. //定义一个颜色为白色的画刷
  30. using (Brush brush = new SolidBrush(Color.White))
  31. {
  32. g.FillRectangle(brush, bound);//绘制背景色。
  33. }
  34. //填充字体,字体的颜色为黑色
  35. TextRenderer.DrawText(g, text, this.Font, bound, Color.Black,
  36. TextFormatFlags.VerticalCenter | TextFormatFlags.Left);
  37. }
  38. }</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. ASP.NET中Button控件的CommandName和CommandArgument属性用法

    在Repeater中的使用: <asp:Repeater ID="rptActionList" runat="server" OnItemCommand= ...

  2. VS2008使用技巧及快捷键大全

    VS2008技巧,非常实用,非常提高效率. 1,Visual Studio 2008自带的1000多个 Windows 系统使用的各种图标.光标和动画文件在Visual Studio 2008的安装目 ...

  3. IOS中UIKit——UIButton的背景图像无法正常显示的原因

    主要是在将图像引入项目中,没有选择Destination:Copy items if needed一项. 没有选择Destination:Copy items if needed一项,图像只能是以链接 ...

  4. ADO .NET 链接 增删改查

    ADO.NET: 数据访问技术 就是将C#和MSSQL连接起来的一个纽带 可以通过ADO.NET将内存中的临时数据写入到数据库中也可以将数据库中的数据提取到内存中供程序调用 所有数据访问技术的基础 连 ...

  5. 使用maven, myeclipse工具构建spring mvc项目

    一.使用myeclipse 创建一个新的 maven项目. (ps:1.在filter过滤的时候输入 webapp 选择"maven-archetype-webapp". 2.在m ...

  6. 【转】在delphi中实现控件的拖拽

    提示:可以添加一个布尔来控制可否拖动的状态,这里提供所有都能拖动的方法. procedure TForm1.FormMouseDown(Sender: TObject; Button: TMouseB ...

  7. Hadoop伪分布式搭建CentOS

    所需软件及版本: jdk-7u80-linux-x64.tar.gz hadoop-2.6.0.tar.gz 1.安装JDK Hadoop 在需在JDK下运行,注意JDK最好使用Oracle的否则可能 ...

  8. shell脚本调用spark-sql

    为了更方便的查询并产生报表, 需要使用shell脚本调用spark-sql spark/bin/spark-sql --master spark://host:7077 -f ${SQL_FILE} ...

  9. properties文件

    properties文件也叫资源文件,以键值对的形式存放文本内容.一个properties对象代表一个资源文件 步骤:1.生成properties对象2.生成InputStream/Reader来读取 ...

  10. 【转】IT领域技能图谱