zoukankan      html  css  js  c++  java
  • Silverlight 学习笔记——用MediaElement制作在线音乐播放器

      Silverlight的一个很大的特性就是针对多媒体内容,包括音频和视频的支持。在Silverlight中,多媒体内容通过窗口MediaElement来呈现。让我们通过实现一个在线音乐播放器,来学习MediaElement对于音频的支持特性。
    
      首先通过Blend Sketch来打个草稿,规划一下最终要实现的界面和功能:
    
     
    
    播放器功能:
    大进度条,用于追踪歌曲的播放进度,并且可以通过拖放来定位到歌曲的不同位置; 
    Start按钮,开始播放;按钮变成Pause,用于暂停;(实现时,改为了Play按钮); 
    Stop按钮,停止播放; 
    Mute按钮,静音; 
    Volume和Balance两个滑条分别控制音量和左右平衡。
      MediaElement类中提供的方法、属性和事件,可以为我们实现以上全部的功能:
    
    方法:
    
      Play
    
      从媒体当前位置开始播放。这个方法可以实现Start按钮播放部分的功能;
    
      Pause
    
      将媒体暂停在当前位置。当用户按下播放以后,播放按钮就会变成暂停按钮,当用户点击暂停按钮就,调用此方法将正在播放的歌曲停住。
    
      Stop
    
      将媒体停止并且重置至起始位置。正好对应了Stop按钮的功能。
    
    属性:
    
      AutoPlay,bool类型
    
      设置或者获取添加完Source时,是否自动播放。由于我们期望用户点击Start按钮时才进行播放,因此,这处需要将其设置为false。
    
      Balance,double类型
    
      设置或者获取扬声器平衡。数值区间:-1~1。用于绑定Balance滑块。
    
      IsMute,bool类型
    
      设置或者获取是否静音,我们通过Mute按钮设置其值。
    
      Position,TimeSpan类型。
    
      通过时间设置或者获取播放中媒体当前的位置。歌曲进度可以通过获取MeidaElement.Position属性中的TotalSeconds来实现,这里有一个小技巧,设置时,可以通过TimeSpan.FromSeconds方法来进行。由于Position是TimeSpan类型的,而滑动条的Value是double值,因此,在绑定的时候,需要引入一个转换器Converter。这将在后面的代码中有所体现。
    
      Source,System.Uri类型
    
      提供媒体源。指向要播放的音乐文件或者播放列表。
    
      Volume,double类型
    
      设置或者获取音量。用于绑定到Volume滑块。
    
    事件:
    
      MediaFailed
    
      设置Source失败时触发。为了简化,这里使用了一个固定的文件,因此,也没有考虑载入失败的问题。这个事件也没有被用到。另外,如果Source指向一个无效的路径,默认情况下,Silverlight会触发这个事件但是不会抛出异常。因此,如果载入文件失败时需要抛出异常,我们需要手动的在MediaFailed事件处理程序中创建并抛出异常。
    
      MediaEnded
    
      媒体播放完成时触发。当一首歌曲播放完成时,需要将界面复原,例如,将Pause按钮还原为Start按钮。
    
    实战
      好,理论知识准备完毕,让我们开始制作。首先,绘制一个5行3列的Grid。
    
       
    
      为了固定播放器的大小,只有最后一行和最后一列没有锁定宽和高。第四行添加了一个方向为水平的StackPanel,并且添加了各个按钮。界面完成以后,用Visual Studio打开项目,双击Play按钮,添加事件处理程序:
    
     
    
    view sourceprint?1 private void btnStart_Click(object sender, RoutedEventArgs e)<BR>        {<BR>            Button self = sender as Button;<BR>            if (self.Content.ToString().ToUpper() == "PLAY")<BR>            {<BR>                sldProgress.Maximum = mdeMusic.NaturalDuration.TimeSpan.TotalSeconds;<BR>                self.Content = "Pause";<BR>                mdeMusic.Play();<BR>            }<BR>            else<BR>            {<BR>                mdeMusic.Pause();<BR>                self.Content = "Play";<BR>            }<BR>        } 
    
     
    
      这里,我们通过判断按钮的Content是否为“Play”来判断当前按钮的功能。当为“Play”时,我们为大的进度滑块设置最大值——整个音乐的全部秒数,然后把按钮上的文字改成“Pause”,并开始播放音乐。而当按钮上的文字不为“Play”时,调用Pause方法将播放暂停,并且再次把按钮上的文字改成“Play”。
    
      Stop按钮的功能很直接,把“Play”按钮上的字改为“Play”,并且调用MediaElement的Stop方法将播放停止。
    
      Mute按钮其实相当于一个CheckBox,或者一个乒乓键,因此,每次点击,我们只需要将逻辑值置返即可:
    
     
    
    view sourceprint?1 private void btnMute_Click(object sender, RoutedEventArgs e)<BR>        {<BR>            mdeMusic.IsMuted = !mdeMusic.IsMuted;<BR>        } 
    
      Volume的滑块只需要设置其最大值为1并且绑定即可:
    
     
    
    view sourceprint?1 <Slider x:Name="sldVolumn" Style="{StaticResource SliderPlayers}" Value="{Binding Volume, ElementName=mdeMusic, Mode=TwoWay, UpdateSourceTrigger=Default}" Maximum="1" LargeChange="0.1" SmallChange="0.01" /> 
    
     
    
      Blalance滑块依样画瓢,不再赘述。
    
      最后,我们要把Position绑定到进度滑块上。做个简单的分析,这里,MediaElement的Position相当于模型,而Slider的Value即是控件上需要绑定的值。当从模型向控件绑定时,我们需要从TimeSpan中提取出总的秒数,提供给滑快的Value。而从控件到模型时,又需要把Value这个double类型的数值,设置成Position当前的秒数。因此,正向时,需要把TimeSpan转换成double,反向时则反过来。由此,便可以实现IValueConverter接口了:
    
     
    
    view sourceprint?1 public class ProgressConverter : IValueConverter<BR>    {<BR>        public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)<BR>        {<BR>            return ((TimeSpan)value).TotalSeconds;<BR>        }<BR><BR>        public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)<BR>        {<BR>            return TimeSpan.FromSeconds((double)value);<BR>        }<BR>    } 
    
     
    
      其中,Convert实现正向转换;ConvertBack则实现逆向的转换。有了转换器,我们便可以把MediaElement的Position和进度滑块绑定起来了:
    
     
    
    view sourceprint?1 <Slider x:Name="sldProgress" Grid.Column="1" Grid.Row="2" Minimum="0" Value="{Binding ElementName=mdeMusic, Path=Position, Mode=TwoWay, Converter={StaticResource PrgConverter}}" LargeChange="10" /> 
    
     
    
      让我们运行项目,效果如下:
    
     
    
      一曲放完,我们发现,Play按钮没有返回到原来的状态,仍然是Pause,这显然不是我们想要的效果,因此,我们还需要做些手脚。首先,在MainPage的构造函数中添加MediaElement的MediaEnded事件的处理程序:
    
     
    
    view sourceprint?1 public MainPage()<BR>        {<BR>            // Required to initialize variables<BR>            InitializeComponent();<BR>            tbMusicName.Text = "Guo Qiao";<BR>            mdeMusic.MediaEnded += new RoutedEventHandler(mdeMusic_MediaEnded);<BR>        } 
    
     
    
      然后,在事件处理程序中,将播放按钮复原:
    
     
    
    view sourceprint?1 void mdeMusic_MediaEnded(object sender, RoutedEventArgs e)<BR>        {<BR>            btnStart.Content = "Play";<BR>            sldProgress.Value = 0;<BR>        } 
    
     
    
     
    写在最后:
      Silverlight通过MediaElement类为多媒体支持,通过其内置的方法、属性和事件,基本上可以完成一个在线音乐播放器的所有功能。可惜Silverlight不能访问本地文件,否则,做一个Web版的本地播放器也是一件很Cool的事情:-)。
    
    Little knowledge is dangerous.
    

    Silverlight 学习笔记——用MediaElement制作在线音乐播放器 - Evil 域 - 博客园

  • 相关阅读:
    破周三,前不着村后不着店的,只好学pandas了,你该这么学,No.9
    周一不睡觉,也要把pandas groupy 肝完,你该这么学,No.8
    大周末的不休息,继续学习pandas吧,pandas你该这么学,No.7
    链接
    音乐
    术语
    新闻
    我的文章分类
    我的代码规则
    Jenkins 访问特别慢,且不消耗服务器资源
  • 原文地址:https://www.cnblogs.com/zjutsxj/p/1743385.html
Copyright © 2011-2022 走看看