新时尚Windows8开发(26):后台播放Musics

2024-03-20 01:08

本文主要是介绍新时尚Windows8开发(26):后台播放Musics,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

要说这Windows Store应用像啥,像网页,而且还全屏的,所以,这和传统桌面应用就有着明显的差异,在同一时刻就只有一个应用在前台运行,对,和手机一样。

故今天我们的话题,本质上是讨论媒体播放的,不过,如果不顺便考虑一下后台播放,感觉有点……你想,当你启动一个播放音乐的应用,正听着很美的音乐,突然我想看看我微博上有啥更新,回到“开始”屏幕,启动一下微博应用。我靠!音乐停了,你想这多没意思啊。

所以啊,当我们希望开发一个播放音乐的应用时,支持后台播放那可是头等大事。

 

好了,牛皮就不吹了,开始说正题。

你说在应用程序中如何播放多媒体,这个应该各位都会,当然我这里指的是XAML中的,HTML5本身就支持多媒体播放。是的,MediaElement,想起来了吧,这位老哥们儿,你不可能不认识D,在玩WPF/Silverlight/Windows Phone等开发的时候,你一定和她亲密接触过的。这控件用起来也很简单,给她一个播放源就可以,就好像一位演技很好的MM,你只需为她简单妆扮,上了舞台就发挥自如。

<MediaElement Source="abcd.mp3" />

可能有时候会设置一下音量什么的,或者把AutoPlay设置为True,这使得这个控件在设置源后便马上播放。

 

这个,我相信大家都会,这里就不多重复了,不然又有人说“这和Win8有毛关系,这不是WPF中的XAML吗?”是的,XAML就是好,跨项目应用;完美的集成性就是好,学会一个知识,可以迅速迁移到其他知识;统一性就是好,专心做程序,不必须要冲突等问题浪费青春。


那么,如何实现后台播放呢,我们来做一做实验吧。如何新建项目我就省略了,我相信这操作连鸵鸟都会了。

1、打开清单文件编辑器,切换到“功能”选项卡,选上“音乐库”,我们待会儿要用。

 

2、切换到“声明”选项卡,在下拉列表中选择“后台任务”,并点击“添加”按钮;添加后,在右侧窗格中,勾选“音频”,并填上入口点,默认就是App。

 

3、打开主页的XAML文件,我们来完成一下UI布局。

<Page
x:Class="BackgroundAudioExample.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:BackgroundAudioExample"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d">
<Grid Background="{StaticResource ApplicationPageBackgroundThemeBrush}">
<StackPanel Margin="28">
<TextBlock Margin="2,1,2,7" TextWrapping="Wrap">
<Span FontSize="24" FontWeight="Bold">实现后台播放,应该要具备以下条件:</Span>
<LineBreak/>
<Run FontSize="20">在清单文件中,加入后台任务的声明</Run><LineBreak/>
<Run FontSize="20">你的PC或笔记本的键盘最好具有多媒体控制键,这样方便操作</Run>
<LineBreak/>
<Run FontSize="20" Foreground="Blue">MediaElement的AudioCategory属性设置为BackgroundCapableMedia或Communications</Run>
<LineBreak/>
<Run FontSize="20">注册MediaControl的控制事件</Run>
</TextBlock>
<StackPanel Orientation="Horizontal" Margin="3,4,3,0">
<Button Content="选择音频文件" Click="onSelectFile"/>
<Button x:Name="btnPlayorPause" Content="播放" Click="onPlayOrPause" Margin="12,0,0,0"/>
<Button Content="停止" Click="onStop" Margin="8,0,0,0"/>
<TextBlock x:Name="tbMessage" Margin="30,0,0,0"/>
</StackPanel>
</StackPanel>
<MediaElement x:Name="myMediaElement" Width="0" Height="0" Opacity="0" AudioCategory="BackgroundCapableMedia" AutoPlay="False"
CurrentStateChanged="myMediaElement_CurrentStateChanged_1"
MediaFailed="myMediaElement_MediaFailed_1"
MediaEnded="myMediaElement_MediaEnded_1"
MediaOpened="myMediaElement_MediaOpened_1"/>
</Grid>
</Page>

 

4、切换代码视图,上面有几个事件要处理。

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Windows.Foundation;
using Windows.Foundation.Collections;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Controls.Primitives;
using Windows.UI.Xaml.Data;
using Windows.UI.Xaml.Input;
using Windows.UI.Xaml.Media;
using Windows.UI.Xaml.Navigation;
// 引入以下命名空间
using Windows.Media;
using Windows.Storage;
using Windows.Storage.Pickers;
using Windows.Storage.Streams;
namespace BackgroundAudioExample
{
public sealed partial class MainPage : Page
{
public MainPage()
{
this.InitializeComponent();
}
private string MusicTitle = string.Empty;//音乐标题
private string MusicArtist = string.Empty;//演唱者
private void myMediaElement_CurrentStateChanged_1(object sender, RoutedEventArgs e)
{
switch (this.myMediaElement.CurrentState)
{
case MediaElementState.Buffering:
tbMessage.Text = "加载中。";
break;
case MediaElementState.Closed:
tbMessage.Text = "已关闭。";
MediaControl.IsPlaying = false;
break;
case MediaElementState.Opening:
tbMessage.Text = "正在打开媒体。";
break;
case MediaElementState.Paused:
MediaControl.IsPlaying = false;
tbMessage.Text = "已暂停。";
btnPlayorPause.Content = "播放";
break;
case MediaElementState.Playing:
MediaControl.IsPlaying = true;
tbMessage.Text = "正在播放。";
btnPlayorPause.Content = "暂停";
break;
case MediaElementState.Stopped:
MediaControl.IsPlaying = false;
tbMessage.Text = "已停止";
btnPlayorPause.Content = "播放";
break;
default:
break;
}
}
private void myMediaElement_MediaFailed_1(object sender, ExceptionRoutedEventArgs e)
{
tbMessage.Text = "播放音频失败,请检查人品。";
}
private void myMediaElement_MediaEnded_1(object sender, RoutedEventArgs e)
{
// 播放完了,就解除事件绑定吧
MediaControl.PlayPressed -= MediaControl_PlayPressed;
MediaControl.PlayPauseTogglePressed -= MediaControl_PlayPauseTogglePressed;
MediaControl.PausePressed -= MediaControl_PausePressed;
MediaControl.StopPressed -= MediaControl_StopPressed;
}
private void myMediaElement_MediaOpened_1(object sender, RoutedEventArgs e)
{
// 注册播放控件事件
MediaControl.PlayPressed += MediaControl_PlayPressed;
MediaControl.PlayPauseTogglePressed += MediaControl_PlayPauseTogglePressed;
MediaControl.PausePressed += MediaControl_PausePressed;
MediaControl.StopPressed += MediaControl_StopPressed;
// 显示歌曲标题和歌手名字
MediaControl.TrackName = this.MusicTitle;
MediaControl.ArtistName = this.MusicArtist;
}
async void MediaControl_StopPressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal, () =>
{
this.myMediaElement.Stop();
});
}
async void MediaControl_PausePressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal,
() =>
{
this.myMediaElement.Pause();
});
}
async void MediaControl_PlayPauseTogglePressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal, () =>
{
onPlayOrPause(null, null);
});
}
async void MediaControl_PlayPressed(object sender, object e)
{
await Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.Normal, () =>
{
this.myMediaElement.Play();
});
}
private async void onSelectFile(object sender, RoutedEventArgs e)
{
FileOpenPicker picker = new FileOpenPicker();
picker.FileTypeFilter.Add(".mp3");
picker.FileTypeFilter.Add(".wma");
picker.SuggestedStartLocation = PickerLocationId.MusicLibrary;
StorageFile audioFile = await picker.PickSingleFileAsync();
if (audioFile != null)
{
// 获取音频文件的属性
var musickProperty = await audioFile.Properties.GetMusicPropertiesAsync();
this.MusicTitle = musickProperty.Title;
if (!string.IsNullOrEmpty(musickProperty.AlbumArtist))
{
this.MusicArtist = musickProperty.AlbumArtist;
}
else
{
this.MusicArtist = musickProperty.Artist;
}
// 设置播放源
var stream = await audioFile.OpenAsync(FileAccessMode.Read);
this.myMediaElement.SetSource(stream, audioFile.FileType);
}
}
private void onPlayOrPause(object sender, RoutedEventArgs e)
{
if (this.myMediaElement.CurrentState == MediaElementState.Playing)
{
this.myMediaElement.Pause();
}
else
{
this.myMediaElement.Play();
}
}
private void onStop(object sender, RoutedEventArgs e)
{
this.myMediaElement.Stop();
}
}
}



代码我们稍候再总结,现在我们迫不及待地想看一下,实验是否成功。果断运行吧。

在你的电脑上随便选一个MP3文件,然后点击播放,然后切换到“开始”屏幕,看看音乐是不是继续播放?如果是,那么实验就成功了。

 

如果你的笔记本键盘有多媒体控制键,在任务屏幕,按下“音量”控制键,就会在屏幕上看到如下图的控制按钮。

 

好了,后台播放大体情况就这样。

大家不妨测试一下,在本例中,如果你只是把MediaElement的AudioCategory属性设置为BackgroundCapableMedia,而不注册MediaControl类的事件处理,你会发现,是不能在后台播放的。

MediaControl类位于Windows.Media命名空间,要完成音频的后台播放和控制,少不了这个类。它是一个static类,所以你能想到它的所有成员都是静态的,它有N多个事件,如下面截图所示,大家不妨打开“对象浏览器”研究一下。

 

在我们本次实验中,只注册了4个事件,也是最基本的4个,大家可以试一下,注释掉其中一个,都不能后台播放,所以,这4个事件是必须的,其他的是可选。

比较好一个处理方案是在有MediaElement的页面的Loaded事件中注册这些事件处理,在UnLoaded中释放。不过本例有所不同,本例是在媒体打开时注册,媒体关闭时释放,因为本例只播放一个音频。所以没有“上一首”,“下一首”。

 

 

 

这篇关于新时尚Windows8开发(26):后台播放Musics的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/827805

相关文章

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

Python+PyQt5实现多屏幕协同播放功能

《Python+PyQt5实现多屏幕协同播放功能》在现代会议展示、数字广告、展览展示等场景中,多屏幕协同播放已成为刚需,下面我们就来看看如何利用Python和PyQt5开发一套功能强大的跨屏播控系统吧... 目录一、项目概述:突破传统播放限制二、核心技术解析2.1 多屏管理机制2.2 播放引擎设计2.3 专

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

使用Python实现文本转语音(TTS)并播放音频

《使用Python实现文本转语音(TTS)并播放音频》在开发涉及语音交互或需要语音提示的应用时,文本转语音(TTS)技术是一个非常实用的工具,下面我们来看看如何使用gTTS和playsound库将文本... 目录什么是 gTTS 和 playsound安装依赖库实现步骤 1. 导入库2. 定义文本和语言 3

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

基于Python开发批量提取Excel图片的小工具

《基于Python开发批量提取Excel图片的小工具》这篇文章主要为大家详细介绍了如何使用Python中的openpyxl库开发一个小工具,可以实现批量提取Excel图片,有需要的小伙伴可以参考一下... 目前有一个需求,就是批量读取当前目录下所有文件夹里的Excel文件,去获取出Excel文件中的图片,并

基于Python开发PDF转PNG的可视化工具

《基于Python开发PDF转PNG的可视化工具》在数字文档处理领域,PDF到图像格式的转换是常见需求,本文介绍如何利用Python的PyMuPDF库和Tkinter框架开发一个带图形界面的PDF转P... 目录一、引言二、功能特性三、技术架构1. 技术栈组成2. 系统架构javascript设计3.效果图

基于Python开发PDF转Doc格式小程序

《基于Python开发PDF转Doc格式小程序》这篇文章主要为大家详细介绍了如何基于Python开发PDF转Doc格式小程序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用python实现PDF转Doc格式小程序以下是一个使用Python实现PDF转DOC格式的GUI程序,采用T