[源码下载]

重新想象 Windows 8.1 Store Apps (87) - TTS: Speak Text, Speak SSML

作者:webabcd

介绍
重新想象 Windows 8.1 Store Apps 之 TTS(Text To Speech)

  • Speak Text
  • Speak SSML

示例
1、演示如何通过 TTS 朗读一段文本,以及如何将其保存为音频文件
SpeakText.xaml

<Page
x:Class="Windows81.TTS.SpeakText"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:Windows81.TTS"
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"> <!--用于播放音频-->
<MediaElement Name="mediaElement" AutoPlay="False" /> <TextBlock Name="lblMsg" FontSize="14.667" Margin="0 10 0 0" /> <!--需要被 TTS 的文本-->
<TextBlock Name="lblText" FontSize="14.667" Margin="0 10 0 0" Text="TTS 是 Text To Speech 的缩写,即“从文本到语音”,是人机对话的一部分,让机器能够说话。" /> <!--开始 TTS-->
<Button Name="btnSpeakText" Content="Speek Text" Click="btnSpeakText_Click" Margin="0 10 0 0" /> <!--保存 TTS 转换出的音频文件-->
<Button Name="btnSaveAudio" Content="Save Audio" Click="btnSaveAudio_Click" Margin="0 10 0 0" /> </StackPanel>
</Grid>
</Page>

SpeakText.xaml.cs

/*
* 演示如何通过 TTS 朗读一段文本,以及如何将其保存为音频文件
*
* 注:Windows Phone 中的“TTS, 语音识别, 语音命令”请参见:http://www.cnblogs.com/webabcd/archive/2014/01/02/3501356.html
*/ using System;
using System.Collections.Generic;
using Windows.Media.SpeechSynthesis;
using Windows.Storage;
using Windows.Storage.Pickers;
using Windows.Storage.Streams;
using Windows.UI.Popups;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Media; namespace Windows81.TTS
{
public sealed partial class SpeakText : Page
{
// TTS 引擎
private SpeechSynthesizer _synthesizer = new SpeechSynthesizer(); public SpeakText()
{
this.InitializeComponent(); lblMsg.Text = "系统支持的语音有:";
// TTS 引擎所支持的全部语音信息
var voices = SpeechSynthesizer.AllVoices;
foreach (VoiceInformation voice in voices)
{
lblMsg.Text += voice.DisplayName + ", ";
} lblMsg.Text += Environment.NewLine;
// 获取或设置当前 TTS 引擎所使用的语音
lblMsg.Text += "当前 TTS 使用的语音是:" + _synthesizer.Voice.DisplayName; // SpeechSynthesizer 实现了 IDisposable 接口
// _synthesizer.Dispose(); this.mediaElement.CurrentStateChanged += mediaElement_CurrentStateChanged;
} void mediaElement_CurrentStateChanged(object sender, RoutedEventArgs e)
{
if (mediaElement.CurrentState == MediaElementState.Paused)
{
btnSpeakText.IsEnabled = true;
}
} // 播放 TTS 音频
private async void btnSpeakText_Click(object sender, RoutedEventArgs e)
{
// TTS 引擎生成的音频流
SpeechSynthesisStream synthesisStream; try
{
// 通过 TTS 引擎将字符串转换成音频流
synthesisStream = await _synthesizer.SynthesizeTextToStreamAsync(lblText.Text);
}
catch (Exception ex)
{
lblMsg.Text = ex.ToString(); synthesisStream = null;
btnSpeakText.IsEnabled = true;
} // 播放音频流
mediaElement.AutoPlay = true;
mediaElement.SetSource(synthesisStream, synthesisStream.ContentType);
mediaElement.Play();
} // 保存 TTS 音频
private async void btnSaveAudio_Click(object sender, RoutedEventArgs e)
{
FileSavePicker savePicker = new FileSavePicker();
savePicker.DefaultFileExtension = ".wav";
savePicker.FileTypeChoices.Add("Audio file", new List<string>() { ".wav" }); StorageFile file = await savePicker.PickSaveFileAsync();
if (file != null)
{
btnSaveAudio.IsEnabled = false; SpeechSynthesisStream synthesisStream;
try
{
synthesisStream = await _synthesizer.SynthesizeTextToStreamAsync(lblText.Text);
}
catch (Exception ex)
{
lblMsg.Text = ex.ToString(); synthesisStream = null;
btnSaveAudio.IsEnabled = true;
} // 保存音频数据到文件
Windows.Storage.Streams.Buffer buffer = new Windows.Storage.Streams.Buffer();
IRandomAccessStream writeStream = (IRandomAccessStream)await file.OpenAsync(FileAccessMode.ReadWrite);
IOutputStream outputStream = writeStream.GetOutputStreamAt();
DataWriter dataWriter = new DataWriter(outputStream); while (synthesisStream.Position < synthesisStream.Size)
{
await synthesisStream.ReadAsync(buffer, , InputStreamOptions.None);
dataWriter.WriteBuffer(buffer);
} dataWriter.StoreAsync().AsTask().Wait();
outputStream.FlushAsync().AsTask().Wait(); btnSaveAudio.IsEnabled = true;
}
}
}
}

2、演示如何通过 TTS 朗读 SSML 协议文档,以及如何将其保存为音频文件
SpeakSSML.xaml

<Page
x:Class="Windows81.TTS.SpeakSSML"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:Windows81.TTS"
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"> <!--用于播放音频-->
<MediaElement Name="mediaElement" AutoPlay="False" /> <TextBlock Name="lblMsg" FontSize="14.667" Margin="0 10 0 0" /> <!--开始 TTS-->
<Button Name="btnSpeakSSML" Content="Speek SSML" Click="btnSpeakSSML_Click" Margin="0 10 0 0" /> <!--保存 TTS 转换出的音频文件-->
<Button Name="btnSaveAudio" Content="Save Audio" Click="btnSaveAudio_Click" Margin="0 10 0 0" /> </StackPanel>
</Grid>
</Page>

SpeakSSML.xaml.cs

/*
* 演示如何通过 TTS 朗读 SSML 协议文档,以及如何将其保存为音频文件
*
* 注:Windows Phone 中的“TTS, 语音识别, 语音命令”请参见:http://www.cnblogs.com/webabcd/archive/2014/01/02/3501356.html
*
*
* 另外:
* 1、SSML - Speech Synthesis Markup Language
* 2、微软关于 ssml 的说明:http://msdn.microsoft.com/en-us/library/hh361578
* 3、W3C 关于 ssml 的说明:http://www.w3.org/TR/speech-synthesis/
*/ using System;
using System.Collections.Generic;
using Windows.Media.SpeechSynthesis;
using Windows.Storage;
using Windows.Storage.Pickers;
using Windows.Storage.Streams;
using Windows.UI.Popups;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Media; namespace Windows81.TTS
{
public sealed partial class SpeakSSML : Page
{
// TTS 引擎
private SpeechSynthesizer _synthesizer = new SpeechSynthesizer(); public SpeakSSML()
{
this.InitializeComponent(); lblMsg.Text = "系统支持的语音有:";
// TTS 引擎所支持的全部语音信息
var voices = SpeechSynthesizer.AllVoices;
foreach (VoiceInformation voice in voices)
{
lblMsg.Text += voice.DisplayName + ", ";
} lblMsg.Text += Environment.NewLine;
// 获取或设置当前 TTS 引擎所使用的语音
lblMsg.Text += "当前 TTS 使用的语音是:" + _synthesizer.Voice.DisplayName; // SpeechSynthesizer 实现了 IDisposable 接口
// _synthesizer.Dispose(); this.mediaElement.CurrentStateChanged += mediaElement_CurrentStateChanged;
} void mediaElement_CurrentStateChanged(object sender, RoutedEventArgs e)
{
if (mediaElement.CurrentState == MediaElementState.Paused)
{
btnSpeakSSML.IsEnabled = true;
}
} // 播放 TTS 音频(SSML)
private async void btnSpeakSSML_Click(object sender, RoutedEventArgs e)
{
// TTS 引擎生成的音频流
SpeechSynthesisStream synthesisStream; try
{
// 构造一个 SSML 协议文档
string ssml = "<speak version=\"1.0\" xmlns=\"http://www.w3.org/2001/10/synthesis\" xml:lang=\"zh-CN\">"; // 中文
ssml += "<voice gender=\"male\">"; // 男声
ssml += "<prosody rate=\"-50%\">"; // 语速放慢 50%
ssml += "TTS 是 Text To Speech 的缩写,即“从文本到语音”,是人机对话的一部分,让机器能够说话。";
ssml += "</prosody>";
ssml += "</voice>";
ssml += "</speak>"; // 通过 TTS 引擎将 SSML 协议文档转换成音频流
synthesisStream = await _synthesizer.SynthesizeSsmlToStreamAsync(ssml);
}
catch (Exception ex)
{
lblMsg.Text = ex.ToString(); synthesisStream = null;
btnSpeakSSML.IsEnabled = true;
} // 播放音频流
mediaElement.AutoPlay = true;
mediaElement.SetSource(synthesisStream, synthesisStream.ContentType);
mediaElement.Play();
} // 保存 TTS 音频(SSML)
private async void btnSaveAudio_Click(object sender, RoutedEventArgs e)
{
FileSavePicker savePicker = new FileSavePicker();
savePicker.DefaultFileExtension = ".wav";
savePicker.FileTypeChoices.Add("Audio file", new List<string>() { ".wav" }); StorageFile file = await savePicker.PickSaveFileAsync();
if (file != null)
{
btnSaveAudio.IsEnabled = false; SpeechSynthesisStream synthesisStream;
try
{
// 构造一个 SSML 协议文档
string ssml = "<speak version=\"1.0\" xmlns=\"http://www.w3.org/2001/10/synthesis\" xml:lang=\"zh-CN\">"; // 中文
ssml += "<voice gender=\"male\">"; // 男声
ssml += "<prosody rate=\"-50%\">"; // 语速放慢 50%
ssml += "TTS 是 Text To Speech 的缩写,即“从文本到语音”,是人机对话的一部分,让机器能够说话。";
ssml += "</prosody>";
ssml += "</voice>";
ssml += "</speak>"; // 通过 TTS 引擎将 SSML 协议文档转换成音频流
synthesisStream = await _synthesizer.SynthesizeSsmlToStreamAsync(ssml);
}
catch (Exception ex)
{
lblMsg.Text = ex.ToString(); synthesisStream = null;
btnSaveAudio.IsEnabled = true;
} // 保存音频数据到文件
Windows.Storage.Streams.Buffer buffer = new Windows.Storage.Streams.Buffer();
IRandomAccessStream writeStream = (IRandomAccessStream)await file.OpenAsync(FileAccessMode.ReadWrite);
IOutputStream outputStream = writeStream.GetOutputStreamAt();
DataWriter dataWriter = new DataWriter(outputStream); while (synthesisStream.Position < synthesisStream.Size)
{
await synthesisStream.ReadAsync(buffer, , InputStreamOptions.None);
dataWriter.WriteBuffer(buffer);
} dataWriter.StoreAsync().AsTask().Wait();
outputStream.FlushAsync().AsTask().Wait(); btnSaveAudio.IsEnabled = true;
}
}
}
}

OK
[源码下载]

重新想象 Windows 8.1 Store Apps (87) - TTS: Speak Text, Speak SSML的更多相关文章

  1. 重新想象 Windows 8.1 Store Apps 系列文章索引

    [源码下载] [重新想象 Windows 8 Store Apps 系列文章] 重新想象 Windows 8.1 Store Apps 系列文章索引 作者:webabcd 1.重新想象 Windows ...

  2. 重新想象 Windows 8.1 Store Apps (81) - 控件增强: WebView 之加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Contract 分享 WebView 中的内容, 为 WebView 截图

    [源码下载] 重新想象 Windows 8.1 Store Apps (81) - 控件增强: WebView 之加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Co ...

  3. 重新想象 Windows 8.1 Store Apps (72) - 新增控件: AppBar, CommandBar

    [源码下载] 重新想象 Windows 8.1 Store Apps (72) - 新增控件: AppBar, CommandBar 作者:webabcd 介绍重新想象 Windows 8.1 Sto ...

  4. 重新想象 Windows 8.1 Store Apps (73) - 新增控件: DatePicker, TimePicker

    [源码下载] 重新想象 Windows 8.1 Store Apps (73) - 新增控件: DatePicker, TimePicker 作者:webabcd 介绍重新想象 Windows 8.1 ...

  5. 重新想象 Windows 8.1 Store Apps (74) - 新增控件: Flyout, MenuFlyout, SettingsFlyout

    [源码下载] 重新想象 Windows 8.1 Store Apps (74) - 新增控件: Flyout, MenuFlyout, SettingsFlyout 作者:webabcd 介绍重新想象 ...

  6. 重新想象 Windows 8.1 Store Apps (75) - 新增控件: Hub, Hyperlink

    [源码下载] 重新想象 Windows 8.1 Store Apps (75) - 新增控件: Hub, Hyperlink 作者:webabcd 介绍重新想象 Windows 8.1 Store A ...

  7. 重新想象 Windows 8.1 Store Apps (76) - 新增控件: SearchBox

    [源码下载] 重新想象 Windows 8.1 Store Apps (76) - 新增控件: SearchBox 作者:webabcd 介绍重新想象 Windows 8.1 Store Apps 之 ...

  8. 重新想象 Windows 8.1 Store Apps (77) - 控件增强: 文本类控件的增强, 部分控件增加了 Header 属性和 HeaderTemplate 属性, 部分控件增加了 PlaceholderText 属性

    [源码下载] 重新想象 Windows 8.1 Store Apps (77) - 控件增强: 文本类控件的增强, 部分控件增加了 Header 属性和 HeaderTemplate 属性, 部分控件 ...

  9. 重新想象 Windows 8.1 Store Apps (78) - 控件增强: ScrollViewer, FlipView, Popup

    [源码下载] 重新想象 Windows 8.1 Store Apps (78) - 控件增强: ScrollViewer, FlipView, Popup 作者:webabcd 介绍重新想象 Wind ...

随机推荐

  1. .NET Core:面向未来的开源跨平台开发技术

    作为一种全新的开源和跨平台的开发平台,.NET Core 历经两年多的开发,终于在于2016年6月27日针对所有主流服务器和桌面操作系统发布 1.0 RTM 版本..NET Core 是一种通用开发平 ...

  2. Android学习之Handler消息传递机制

    Android只允许UI线程修改Activity里的UI组件.当Android程序第一次启动时,Android会同时启动一条主线程(Main Thread),主线程主要负责处理与UI相关的事件,如用户 ...

  3. Firefox终于返回到了Debian stable

    6月8日,firefox 45.2以安全修复包的名义回到了Debian oldstable (即wheezy),两天以后,Debian 8 jessie里面也有了(https://packages.d ...

  4. 实现仿知乎的开场动画,图片zoomin的效果,实现原理,没加动效

    知乎等应用的开场动画是:全屏显示一副图像,并以图像的中间为原点,实现放大(也就是zoomin)的动画,让等待的过程不再单调乏味. 最近不是很忙,因此想了下如何实现这种效果,方案是:采用调整imagev ...

  5. sruts2:单个文件上传,多个文件上传(属性驱动)

    文件上传功能在Struts2中得到了很好的封装,主要使用fileUpload上传组件. 1. 单个文件上传 1.1 创建上传单个文件的JSP页面.显示提交结果的JSP页面 uploadTest1.js ...

  6. Kafka - 消费接口分析

    1.概述 在 Kafka 中,官方对外提供了两种消费 API,一种是高等级消费 API,另一种是低等级的消费 API.在 <高级消费 API>一文中,介绍了其高级消费的 API 实现.今天 ...

  7. js/jquery 实时监听输入框值变化的完美方案:oninput & onpropertychange

    (1)     先说jquery, 使用 jQuery 库的话,只需要同时绑定 oninput 和 onpropertychange 两个事件就可以了,示例代码: $('#username').bin ...

  8. OceanBase RPC机制简要说明

    http://note.youdao.com/share/?id=d2163a7ba8ec1ec58e64683a961c5121&type=note RPC是OB一个比较重要的各个组件间进行 ...

  9. Hashing Trick

    本博客已经迁往http://www.kemaswill.com/, 博客园这边也会继续更新, 欢迎关注~ 在机器学习领域, kernel trick是一种非常有效的比较两个样本(对象)的方法. 给定两 ...

  10. APP-BOM-20516 错误处理一例

    昨天在处理一个工单异常时,需要将一个Released的工单改为Unreleased状态,程序报APP-BOM-20516错误,如下图.百度只搜到两条记录,均无用.Google能搜到的多一些,也无用.进 ...