新时尚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

相关文章

Python38个游戏开发库整理汇总

《Python38个游戏开发库整理汇总》文章介绍了多种Python游戏开发库,涵盖2D/3D游戏开发、多人游戏框架及视觉小说引擎,适合不同需求的开发者入门,强调跨平台支持与易用性,并鼓励读者交流反馈以... 目录PyGameCocos2dPySoyPyOgrepygletPanda3DBlenderFife

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

SpringBoot开发中十大常见陷阱深度解析与避坑指南

《SpringBoot开发中十大常见陷阱深度解析与避坑指南》在SpringBoot的开发过程中,即使是经验丰富的开发者也难免会遇到各种棘手的问题,本文将针对SpringBoot开发中十大常见的“坑... 目录引言一、配置总出错?是不是同时用了.properties和.yml?二、换个位置配置就失效?搞清楚加

Python如何将OpenCV摄像头视频流通过浏览器播放

《Python如何将OpenCV摄像头视频流通过浏览器播放》:本文主要介绍Python如何将OpenCV摄像头视频流通过浏览器播放的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完... 目录方法1:使用Flask + MJPEG流实现代码使用方法优点缺点方法2:使用WebSocket传输视

Python中对FFmpeg封装开发库FFmpy详解

《Python中对FFmpeg封装开发库FFmpy详解》:本文主要介绍Python中对FFmpeg封装开发库FFmpy,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、FFmpy简介与安装1.1 FFmpy概述1.2 安装方法二、FFmpy核心类与方法2.1 FF

基于Python开发Windows屏幕控制工具

《基于Python开发Windows屏幕控制工具》在数字化办公时代,屏幕管理已成为提升工作效率和保护眼睛健康的重要环节,本文将分享一个基于Python和PySide6开发的Windows屏幕控制工具,... 目录概述功能亮点界面展示实现步骤详解1. 环境准备2. 亮度控制模块3. 息屏功能实现4. 息屏时间