重新想象 Windows 8 Store Apps (65) - 后台任务: 音乐的后台播放和控制
作者:webabcd
介绍
重新想象 Windows 8 Store Apps 之 后台任务
- 音乐的后台播放和控制
示例
用于保存每首音乐的相关信息的对象
BackgroundTask/SongModel.cs
/*
* 用于保存每首音乐的相关信息
*/ using System;
using System.Threading.Tasks;
using Windows.Storage; namespace XamlDemo.BackgroundTask
{
public class SongModel
{
/// <summary>
/// 音乐文件
/// </summary>
public StorageFile File; /// <summary>
/// 艺术家
/// </summary>
public string Artist; /// <summary>
/// 音乐名称
/// </summary>
public string Title; public SongModel(StorageFile file)
{
File = file;
} /// <summary>
/// 加载音乐的相关属性
/// </summary>
public async Task LoadMusicPropertiesAsync()
{
var properties = await this.File.Properties.GetMusicPropertiesAsync(); Artist = properties.Artist;
Title = properties.Title;
}
}
}
演示如何通过 MediaControl 实现音乐的后台播放和控制
BackgroundTask/MediaControlDemo.xaml
<Page
x:Class="XamlDemo.BackgroundTask.MediaControlDemo"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:XamlDemo.BackgroundTask"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"> <Grid Background="Transparent">
<StackPanel Margin="120 0 0 0"> <TextBlock Name="lblMsg" FontSize="14.667" /> <Button Name="btnSelectFiles" Content="Select Files" Click="btnSelectFiles_Click" Margin="0 10 0 0" />
<Button Name="btnPlay" Content="Play" Click="btnPlay_Click" Margin="0 10 0 0" /> <!--
为了使音乐可以后台播放,需要将 MediaElement 的 AudioCategory 属性设置为 BackgroundCapableMedia
-->
<MediaElement Name="mediaElement" AudioCategory="BackgroundCapableMedia" AutoPlay="False"
MediaOpened="mediaElement_MediaOpened" MediaEnded="mediaElement_MediaEnded" CurrentStateChanged="mediaElement_CurrentStateChanged" /> </StackPanel>
</Grid> </Page>
BackgroundTask/MediaControlDemo.cs
/*
* 演示如何通过 MediaControl 实现音乐的后台播放和控制
*
* 注:
* 1、需要在 Package.appxmanifest 中增加后台任务声明,并勾选“音频”
* 2、需要将 MediaElement 的 AudioCategory 属性设置为 BackgroundCapableMedia
*
* 另:
* 按下平板上的音量键或者多媒体键盘上的相关按键,会弹出后台音频的控制框,通过 MediaControl 来监听用户在此音频控制框上的操作
*/ using System;
using System.Collections.Generic;
using System.Threading.Tasks;
using Windows.Media;
using Windows.Storage;
using Windows.Storage.Pickers;
using Windows.Storage.Streams;
using Windows.UI.Core;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Media; namespace XamlDemo.BackgroundTask
{
public sealed partial class MediaControlDemo : Page
{
private List<SongModel> _playList = new List<SongModel>(); // 后台需要播放的音乐列表
private int _currentSongIndex = ; // 当前播放的音乐 private bool _previousTrackPressedRegistered = false; // MediaControl 是否注册了 PreviousTrackPressed 事件
private bool _nextTrackPressedRegistered = false; // MediaControl 是否注册了 NextTrackPressed 事件 public MediaControlDemo()
{
this.InitializeComponent(); MediaControl.PlayPauseTogglePressed += MediaControl_PlayPauseTogglePressed; // 按下了 play/pause 键
MediaControl.PlayPressed += MediaControl_PlayPressed; // 按下了 play 键
MediaControl.PausePressed += MediaControl_PausePressed; // 按下了 pause 键
MediaControl.StopPressed += MediaControl_StopPressed; // 按下了 stop 键 MediaControl.PreviousTrackPressed += MediaControl_PreviousTrackPressed; // 按下了“上一首”键
MediaControl.NextTrackPressed += MediaControl_NextTrackPressed; // 按下了“下一首”键
MediaControl.RewindPressed += MediaControl_RewindPressed; // 按下了“快退”键
MediaControl.FastForwardPressed += MediaControl_FastForwardPressed; // 按下了“快进”键 MediaControl.SoundLevelChanged += MediaControl_SoundLevelChanged; // 音量级别发生了改变 // MediaControl.RecordPressed += MediaControl_RecordPressed;
// MediaControl.ChannelDownPressed += MediaControl_ChannelDownPressed;
// MediaControl.ChannelUpPressed += MediaControl_ChannelUpPressed; // MediaControl.ArtistName; // 当前播放的音乐的艺术家
// MediaControl.TrackName; // 当前播放的音乐的名称
// MediaControl.AlbumArt; // 当前播放的音乐的专辑封面的路径
// MediaControl.IsPlaying; // 当前音乐是否正在播放
// MediaControl.SoundLevel; // 当前播放的音乐的音量级别(Muted, Low, Full) _previousTrackPressedRegistered = true;
_nextTrackPressedRegistered = true;
} // 按下了音频控制框上的 play/pause 键
async void MediaControl_PlayPauseTogglePressed(object sender, object e)
{
if (MediaControl.IsPlaying)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.Pause();
});
await OutputMessage("Play/Pause Pressed - Pause");
}
else
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.Play();
});
await OutputMessage("Play/Pause Pressed - Play");
}
} // 按下了音频控制框上的 play 键
async void MediaControl_PlayPressed(object sender, object e)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.Play();
});
await OutputMessage("Play Pressed");
} // 按下了音频控制框上的 pause 键
async void MediaControl_PausePressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal, () =>
{
mediaElement.Pause();
});
await OutputMessage("Pause Pressed");
} // 按下了音频控制框上的 stop 键
async void MediaControl_StopPressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal, () =>
{
mediaElement.Stop();
});
await OutputMessage("Stop Pressed");
} // 按下了音频控制框上的“上一首”键
async void MediaControl_PreviousTrackPressed(object sender, object e)
{
await OutputMessage("Previous Track Pressed"); if (_currentSongIndex > )
{
if (_currentSongIndex == (_playList.Count - ))
{
if (!_nextTrackPressedRegistered)
{
MediaControl.NextTrackPressed += MediaControl_NextTrackPressed;
_nextTrackPressedRegistered = true;
}
} _currentSongIndex--; if (_currentSongIndex == )
{
MediaControl.PreviousTrackPressed -= MediaControl_PreviousTrackPressed;
_nextTrackPressedRegistered = false;
} await SetCurrentPlayingAsync(_currentSongIndex);
} } // 按下了音频控制框上的“下一首”键
async void MediaControl_NextTrackPressed(object sender, object e)
{
await OutputMessage("Next Track Pressed"); if (_currentSongIndex < (_playList.Count - ))
{
_currentSongIndex++;
await SetCurrentPlayingAsync(_currentSongIndex); if (_currentSongIndex > )
{
if (!_previousTrackPressedRegistered)
{
MediaControl.PreviousTrackPressed += MediaControl_PreviousTrackPressed;
_previousTrackPressedRegistered = true;
} } if (_currentSongIndex == (_playList.Count - ))
{
if (_nextTrackPressedRegistered)
{
MediaControl.NextTrackPressed -= MediaControl_NextTrackPressed;
_nextTrackPressedRegistered = false;
}
}
}
} // 按下了音频控制框上的“快退”键,即长按“上一首”键
async void MediaControl_RewindPressed(object sender, object e)
{
await OutputMessage("Rewind Pressed");
} // 按下了音频控制框上的“快进”键,即长按“下一首”键
async void MediaControl_FastForwardPressed(object sender, object e)
{
await OutputMessage("Fast Forward Pressed");
} // 音量级别发生变化时(Muted, Low, Full)
async void MediaControl_SoundLevelChanged(object sender, object e)
{
await OutputMessage("Sound Level Changed");
} // 创建一个需要在后台播放的音乐列表
private async void btnSelectFiles_Click(object sender, RoutedEventArgs e)
{
FileOpenPicker openPicker = new FileOpenPicker();
openPicker.ViewMode = PickerViewMode.List;
openPicker.SuggestedStartLocation = PickerLocationId.MusicLibrary;
openPicker.FileTypeFilter.Add(".mp3");
openPicker.FileTypeFilter.Add(".wma");
IReadOnlyList<StorageFile> files = await openPicker.PickMultipleFilesAsync();
if (files.Count > )
{
await CreatePlaylist(files);
await SetCurrentPlayingAsync(_currentSongIndex);
}
} // 开始播放
private async void btnPlay_Click(object sender, RoutedEventArgs e)
{
if (MediaControl.IsPlaying)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.Pause();
});
await OutputMessage("Play/Pause Pressed - Pause");
}
else
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.Play();
});
await OutputMessage("Play/Pause Pressed - Play");
}
} // MediaElement 打开文件后,如果需要播放则立刻播放
private void mediaElement_MediaOpened(object sender, RoutedEventArgs e)
{
if (MediaControl.IsPlaying)
{
mediaElement.Play();
}
} // 当前音乐播放完后,转到下一首
private async void mediaElement_MediaEnded(object sender, RoutedEventArgs e)
{
if (_currentSongIndex < _playList.Count - )
{
_currentSongIndex++; await SetCurrentPlayingAsync(_currentSongIndex); if (MediaControl.IsPlaying)
{
mediaElement.Play();
}
}
} // 根据前台的操作,设置 MediaControl 的 IsPlaying 属性
private void mediaElement_CurrentStateChanged(object sender, RoutedEventArgs e)
{
if (mediaElement.CurrentState == MediaElementState.Playing)
{
MediaControl.IsPlaying = true;
btnPlay.Content = "Pause";
}
else
{
MediaControl.IsPlaying = false;
btnPlay.Content = "Play";
}
} // 创建音乐列表
private async Task CreatePlaylist(IReadOnlyList<StorageFile> files)
{
if (_previousTrackPressedRegistered)
{
MediaControl.PreviousTrackPressed -= MediaControl_PreviousTrackPressed;
_previousTrackPressedRegistered = false;
}
if (_nextTrackPressedRegistered)
{
MediaControl.NextTrackPressed -= MediaControl_NextTrackPressed;
_nextTrackPressedRegistered = false;
} _playList.Clear();
_currentSongIndex = ; if (files.Count > )
{
if (files.Count > )
{
MediaControl.NextTrackPressed += MediaControl_NextTrackPressed;
_nextTrackPressedRegistered = true;
} foreach (StorageFile file in files)
{
SongModel song = new SongModel(file);
await song.LoadMusicPropertiesAsync();
_playList.Add(song);
}
}
} // 指定当前需要播放的音乐
private async Task SetCurrentPlayingAsync(int playListIndex)
{
string errorMessage = null;
IRandomAccessStream stream = null; try
{
stream = await _playList[playListIndex].File.OpenAsync(Windows.Storage.FileAccessMode.Read);
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
mediaElement.SetSource(stream, _playList[playListIndex].File.ContentType);
});
}
catch (Exception e)
{
errorMessage = e.Message;
} if (errorMessage != null)
{
await OutputMessage(errorMessage);
} MediaControl.ArtistName = _playList[playListIndex].Artist;
MediaControl.TrackName = _playList[playListIndex].Title;
} // 输出信息
private async Task OutputMessage(string message)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
lblMsg.Text = message + ": " + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
});
}
}
}
OK
[源码下载]
重新想象 Windows 8 Store Apps (65) - 后台任务: 音乐的后台播放和控制的更多相关文章
- 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务
[源码下载] 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 后 ...
- 重新想象 Windows 8 Store Apps (66) - 后台任务: 下载和上传
[源码下载] 重新想象 Windows 8 Store Apps (66) - 后台任务: 下载和上传 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 后台任务 后台 ...
- 重新想象 Windows 8 Store Apps (67) - 后台任务: 推送通知
[源码下载] 重新想象 Windows 8 Store Apps (67) - 后台任务: 推送通知 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 后台任务 推送通 ...
- 重新想象 Windows 8 Store Apps (68) - 后台任务: 控制通道(ControlChannel)
[源码下载] 重新想象 Windows 8 Store Apps (68) - 后台任务: 控制通道(ControlChannel) 作者:webabcd 介绍重新想象 Windows 8 Store ...
- 重新想象 Windows 8 Store Apps 系列文章索引
[源码下载][重新想象 Windows 8.1 Store Apps 系列文章] 重新想象 Windows 8 Store Apps 系列文章索引 作者:webabcd 1.重新想象 Windows ...
- 重新想象 Windows 8 Store Apps (3) - 控件之内容控件: ToolTip, Frame, AppBar, ContentControl, ContentPresenter; 容器控件: Border, Viewbox, Popup
原文:重新想象 Windows 8 Store Apps (3) - 控件之内容控件: ToolTip, Frame, AppBar, ContentControl, ContentPresenter ...
- 重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationContext, CoreDispatcher, ThreadLocal, ThreadStaticAttribute
[源码下载] 重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationCont ...
- 重新想象 Windows 8 Store Apps (59) - 锁屏
[源码下载] 重新想象 Windows 8 Store Apps (59) - 锁屏 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 锁屏 登录锁屏,获取当前程序的锁 ...
- 重新想象 Windows 8 Store Apps (61) - 通信: http, oauth
[源码下载] 重新想象 Windows 8 Store Apps (61) - 通信: http, oauth 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 通信 ...
随机推荐
- Kafka 分区备份实战
1.概述 在 Kafka 集群中,我们可以对每个 Topic 进行一个或是多个分区,并为该 Topic 指定备份数.这部分元数据信息都是存放在 Zookeeper 上,我们可以使用 zkCli 客户端 ...
- Hadoop - Kylin On OLAP
1.概述 Apache Kylin是一个开源的分布式分析引擎,提供SQL接口并且用于OLAP业务于Hadoop的大数据集上,该项目由eBay贡献于Apache. 2.What is Kylin 在使用 ...
- git删除远程文件夹或文件的方法
由于本地修改了文件夹大全名大小写的原因,同步到git上并不区分大小写,造成了一些文件同步不了,所以要先把git远程库上文件夹删除掉,然后再重新同步 如下,我把src里的全部移除,但是本地文件还保留. ...
- centos 7 中 tomcat 安装
一. 安装 jdk [root@iZm5ei4m ~]# /usr/local/tomcat/bin/startup.sh 一.安装 jdk 二.设置环境变量 #################### ...
- kail2在虚拟机上的安装
首先先要安装虚拟机,打开安装包,下一步 选择典型 选择要安装到的目录,点下一步 4 输入密钥,下一步(密钥网上有很多我这边就例举一个,没用的话就自己找.我这个密钥是VM11 ...
- repo andrid
http://www.cnblogs.com/bluestorm/p/4419135.html
- npm 发布包
每个包都必须包含package.json配置文件 生成package.page文件 1.到项目目录下执行npm init根据提示输入即可 最后记得要yes 完成项目后就是要发布到npm了 首先需要有n ...
- 创建 iPhone/iOS8 弹出菜单(窗口)
基本步骤 添加视图:主视图与弹出视图 关联视图 配置弹出视图 编码实现:弹出菜单样式及控制器委托 override func prepareForSegue(segue: UIStoryboardSe ...
- U3D中IOS平台泛型方法尽少使用
U3D的IOS最小运行库use micro mscorlib是不包含泛型反射方法的,如FieldType.GetGenericArguments方法.所以尽量少用List而直接使用array. 另外请 ...
- BI项目需求分析书-模板
目录 目录 .............................................................................................. ...