Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
OpenHarmony
Docs
提交
fe14d589
D
Docs
项目概览
OpenHarmony
/
Docs
1 年多 前同步成功
通知
159
Star
292
Fork
28
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
D
Docs
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
未验证
提交
fe14d589
编写于
10月 17, 2022
作者:
O
openharmony_ci
提交者:
Gitee
10月 17, 2022
浏览文件
操作
浏览文件
下载
差异文件
!10659 媒体子系统Audio优化
Merge pull request !10659 from zengyawen/master
上级
f6744eae
473c8912
变更
7
隐藏空白更改
内联
并排
Showing
7 changed file
with
329 addition
and
281 deletion
+329
-281
zh-cn/application-dev/media/audio-capturer.md
zh-cn/application-dev/media/audio-capturer.md
+68
-57
zh-cn/application-dev/media/audio-interruptmode.md
zh-cn/application-dev/media/audio-interruptmode.md
+22
-25
zh-cn/application-dev/media/audio-renderer.md
zh-cn/application-dev/media/audio-renderer.md
+183
-182
zh-cn/application-dev/media/audio-stream-manager.md
zh-cn/application-dev/media/audio-stream-manager.md
+11
-8
zh-cn/application-dev/media/opensles-capture.md
zh-cn/application-dev/media/opensles-capture.md
+3
-3
zh-cn/application-dev/media/opensles-playback.md
zh-cn/application-dev/media/opensles-playback.md
+4
-4
zh-cn/application-dev/reference/apis/js-apis-audio.md
zh-cn/application-dev/reference/apis/js-apis-audio.md
+38
-2
未找到文件。
zh-cn/application-dev/media/audio-capturer.md
浏览文件 @
fe14d589
# 音频采集开发指导
##
场景介绍
##
简介
AudioCapturer提供了用于获取原始音频文件的方法。开发者可以通过本指导了解应用如何通过AudioCapturer
采集音频
。
AudioCapturer提供了用于获取原始音频文件的方法。开发者可以通过本指导了解应用如何通过AudioCapturer
接口的调用实现音频数据的采集
。
### 状态检查
-
**状态检查**
:在进行应用开发的过程中,建议开发者通过on('stateChange')方法订阅AudioCapturer的状态变更。因为针对AudioCapturer的某些操作,仅在音频采集器在固定状态时才能执行。如果应用在音频采集器处于错误状态时执行操作,系统可能会抛出异常或生成其他未定义的行为。
在进行应用开发的过程中,建议开发者通过on('stateChange')方法订阅AudioCapturer的状态变更。因为针对AudioCapturer的某些操作,仅在音频采集器在固定状态时才能执行。如果应用在音频采集器处于错误状态时执行操作,系统可能会抛出异常或生成其他未定义的行为。
## 运作机制
详细API含义可参考:
[
音频管理API文档AudioCapturer
](
../reference/apis/js-apis-audio.md#audiocapturer8
)
该模块提供了音频采集模块的状态变化示意图
**图1**
音频采集状态变化示意图
![
audio-capturer-state
](
figures/audio-capturer-state.png
)
**PREPARED状态:**
通过调用create()方法进入到该状态。
<br>
**RUNNING状态:**
正在进行音频数据播放,可以在prepared状态通过调用start()方法进入此状态,也可以在stopped状态通过调用start()方法进入此状态。
<br>
**STOPPED状态:**
在running状态可以通过stop()方法停止音频数据的播放。
<br>
**RELEASED状态:**
在prepared和stop状态,用户均可通过release()方法释放掉所有占用的硬件和软件资源,并且不会再进入到其他的任何一种状态了。
<br>
## 约束与限制
**图1**
音频采集状态机
开发者在进行音频数据采集功能开发前,需要先对所开发的应用配置麦克风权限(ohos.permission.MICROPHONE),权限配置相关内容可参考:
[
访问控制授权申请指导
](
../security/accesstoken-guidelines.md
)
![](
figures/audio-capturer-state.png
)
## 开发指导
## 开发步骤
详细API含义可参考:
[
音频管理API文档AudioCapturer
](
../reference/apis/js-apis-audio.md#audiocapturer8
)
1.
使用createAudioCapturer()创建一个AudioCapturer实例。
在audioCapturerOptions中设置音频采集器的相关参数。该实例可用于音频采集、控制和获取采集状态,以及注册通知回调。
```
js
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioCapturerInfo
=
{
source
:
audio
.
SourceType
.
SOURCE_TYPE_MIC
,
capturerFlags
:
1
}
var
audioCapturerOptions
=
{
streamInfo
:
audioStreamInfo
,
capturerInfo
:
audioCapturerInfo
}
let
audioCapturer
=
await
audio
.
createAudioCapturer
(
audioCapturerOptions
);
var
state
=
audioRenderer
.
state
;
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioCapturerInfo
=
{
source
:
audio
.
SourceType
.
SOURCE_TYPE_MIC
,
capturerFlags
:
1
}
var
audioCapturerOptions
=
{
streamInfo
:
audioStreamInfo
,
capturerInfo
:
audioCapturerInfo
}
let
audioCapturer
=
await
audio
.
createAudioCapturer
(
audioCapturerOptions
);
var
state
=
audioRenderer
.
state
;
```
2.
(可选)使用on('stateChange')订阅音频采集器状态更改。
如果应用需要在采集器状态更新时进行一些操作,可以订阅该事件。更多事件请参考
[
API参考文档
](
../reference/apis/js-apis-audio.md
)
。
```
js
audioCapturer
.
on
(
'
stateChange
'
,(
state
)
=>
{
console
.
info
(
'
AudioCapturerLog: Changed State to :
'
+
state
)
switch
(
state
)
{
case
audio
.
AudioState
.
STATE_PREPARED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------PREPARED--------------
'
);
console
.
info
(
'
Audio State is : Prepared
'
);
break
;
case
audio
.
AudioState
.
STATE_RUNNING
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------RUNNING--------------
'
);
console
.
info
(
'
Audio State is : Running
'
);
break
;
case
audio
.
AudioState
.
STATE_STOPPED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------STOPPED--------------
'
);
console
.
info
(
'
Audio State is : stopped
'
);
break
;
case
audio
.
AudioState
.
STATE_RELEASED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------RELEASED--------------
'
);
console
.
info
(
'
Audio State is : released
'
);
break
;
default
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------INVALID--------------
'
);
console
.
info
(
'
Audio State is : invalid
'
);
break
;
}
});
```
如果应用需要在采集器状态更新时进行一些操作,可以订阅该事件。更多事件请参考
[
API参考文档
](
../reference/apis/js-apis-audio.md
)
。
```
js
audioCapturer
.
on
(
'
stateChange
'
,(
state
)
=>
{
console
.
info
(
'
AudioCapturerLog: Changed State to :
'
+
state
)
switch
(
state
)
{
case
audio
.
AudioState
.
STATE_PREPARED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------PREPARED--------------
'
);
console
.
info
(
'
Audio State is : Prepared
'
);
break
;
case
audio
.
AudioState
.
STATE_RUNNING
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------RUNNING--------------
'
);
console
.
info
(
'
Audio State is : Running
'
);
break
;
case
audio
.
AudioState
.
STATE_STOPPED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------STOPPED--------------
'
);
console
.
info
(
'
Audio State is : stopped
'
);
break
;
case
audio
.
AudioState
.
STATE_RELEASED
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------RELEASED--------------
'
);
console
.
info
(
'
Audio State is : released
'
);
break
;
default
:
console
.
info
(
'
--------CHANGE IN AUDIO STATE----------INVALID--------------
'
);
console
.
info
(
'
Audio State is : invalid
'
);
break
;
}
});
```
3.
调用start()方法来启动/恢复采集任务。
...
...
@@ -131,7 +142,7 @@ AudioCapturer提供了用于获取原始音频文件的方法。开发者可以
6.
采集完成后,调用stop方法,停止录制。
```
```
js
await
audioCapturer
.
stop
();
if
(
audioCapturer
.
state
==
audio
.
AudioState
.
STATE_STOPPED
)
{
console
.
info
(
'
AudioRecLog: Capturer stopped
'
);
...
...
zh-cn/application-dev/media/audio-interruptmode.md
浏览文件 @
fe14d589
# 音频焦点模式开发指导
##
场景介绍
##
简介
音频焦点模式指的是应用内,允许对多个声音的播放进行控制。
<br>
音频应用可以在AudioRenderer下设置独立焦点模式、共享焦点模式。
<br>
当设置在共享的模式下,多个音频共用一个会话ID;独立焦点模式下,每一个音频拥有单独会话ID。
### 异步操作
-
**异步操作**
:为保证UI线程不被阻塞,大部分AudioRenderer调用都是异步的。对于每个API均提供了callback函数和Promise函数,以下示例均采用Promise函数。
为保证UI线程不被阻塞,大部分AudioRenderer调用都是异步的。对于每个API均提供了callback函数和Promise函数,以下示例均采用Promise函数。
## 开发步骤
## 开发指导
详细API含义可参考:
[
音频管理API文档AudioRenderer
](
../reference/apis/js-apis-audio.md#audiorenderer8
)
1.
使用createAudioRenderer()创建一个AudioRenderer实例。
<br>
在audioRendererOptions中设置相关参数。
<br>
该实例可用于音频渲染、控制和获取采集状态,以及注册通知回调。
<br>
...
...
@@ -21,25 +18,25 @@
```
js
import
audio
from
'
@ohos.multimedia.audio
'
;
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioRendererInfo
=
{
content
:
audio
.
ContentType
.
CONTENT_TYPE_SPEECH
,
usage
:
audio
.
StreamUsage
.
STREAM_USAGE_VOICE_COMMUNICATION
,
rendererFlags
:
1
}
var
audioRendererOptions
=
{
streamInfo
:
audioStreamInfo
,
rendererInfo
:
audioRendererInfo
}
let
audioRenderer
=
await
audio
.
createAudioRenderer
(
audioRendererOptions
);
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioRendererInfo
=
{
content
:
audio
.
ContentType
.
CONTENT_TYPE_SPEECH
,
usage
:
audio
.
StreamUsage
.
STREAM_USAGE_VOICE_COMMUNICATION
,
rendererFlags
:
1
}
var
audioRendererOptions
=
{
streamInfo
:
audioStreamInfo
,
rendererInfo
:
audioRendererInfo
}
let
audioRenderer
=
await
audio
.
createAudioRenderer
(
audioRendererOptions
);
```
2.
设置焦点模式。
...
...
zh-cn/application-dev/media/audio-renderer.md
浏览文件 @
fe14d589
# 音频渲染开发指导
##
场景介绍
##
简介
AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可以通过本指导,了解如何在输出设备中播放音频文件并管理播放任务。
AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可以通过本指导,了解如何在输出设备中播放音频文件并管理播放任务。同时,AudioRenderer支持音频中断的功能。
开发者在调用AudioRenderer提供的各个接口时,需要理解以下名词:
同时,AudioRenderer支持音频中断的功能。
-
**音频中断**
:当优先级较高的音频流需要播放时,AudioRenderer会中断优先级较低的流。例如,当用户在收听音乐时有来电,则优先级较低音乐播放将被暂停。
-
**状态检查**
:在进行应用开发的过程中,建议开发者通过on('stateChange')方法订阅AudioRenderer的状态变更。因为针对AudioRenderer的某些操作,仅在音频播放器在固定状态时才能执行。如果应用在音频播放器处于错误状态时执行操作,系统可能会抛出异常或生成其他未定义的行为。
-
**异步操作**
:为保证UI线程不被阻塞,大部分AudioRenderer调用都是异步的。对于每个API均提供了callback函数和Promise函数,以下示例均采用Promise函数,更多方式可参考
[
音频管理API文档AudioRenderer
](
../reference/apis/js-apis-audio.md#audiorenderer8
)
。
##
# 音频中断
##
运作机制
当优先级较高的音频流需要播放时,AudioRenderer会中断优先级较低的流。例如,当用户在收听音乐时有来电,则优先级较低音乐播放将被暂停,具体可参考
[
开发步骤
](
#开发步骤
)
。
该模块提供了音频渲染模块的状态变化示意
### 状态检查
**图1**
音频渲染状态示意图
在进行应用开发的过程中,建议开发者通过on('stateChange')方法订阅AudioRenderer的状态变更。因为针对AudioRenderer的某些操作,仅在音频播放器在固定状态时才能执行。如果应用在音频播放器处于错误状态时执行操作,系统可能会抛出异常或生成其他未定义的行为。
![
audio-renderer-state
](
figures/audio-renderer-state.png
)
**图1**
音频渲染状态机
**PREPARED状态:**
通过调用create()方法进入到该状态。
<br>
**RUNNING状态:**
正在进行音频数据播放,可以在prepared状态通过调用start()方法进入此状态,也可以在pause状态和stopped状态通过调用start()方法进入此状态。
<br>
**PAUSED状态:**
在running状态可以通过pause()方法暂停音频数据的播放,暂停播放之后可以通过调用start()方法继续音频数据播放。
<br>
**STOPPED状态:**
在paused状态可以通过调用stop()方法停止音频数据的播放,在running状态可以通过stop()方法停止音频数据的播放。
<br>
**RELEASED状态:**
在prepared、paused、stop等状态,用户均可通过release()方法释放掉所有占用的硬件和软件资源,并且不会再进入到其他的任何一种状态了。
<br>
![](
figures/audio-renderer-state.png
)
### 异步操作
为保证UI线程不被阻塞,大部分AudioRenderer调用都是异步的。对于每个API均提供了callback函数和Promise函数,以下示例均采用Promise函数,更多方式可参考
[
音频管理API文档AudioRenderer
](
../reference/apis/js-apis-audio.md#audiorenderer8
)
。
## 开发步骤
## 开发指导
1.
使用createAudioRenderer()创建一个AudioRenderer实例。
在audioCapturerOptions中设置相关参数。该实例可用于音频渲染、控制和获取采集状态,以及注册通知回调。
```
js
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioRendererInfo
=
{
content
:
audio
.
ContentType
.
CONTENT_TYPE_SPEECH
,
usage
:
audio
.
StreamUsage
.
STREAM_USAGE_VOICE_COMMUNICATION
,
rendererFlags
:
1
}
var
audioRendererOptions
=
{
streamInfo
:
audioStreamInfo
,
rendererInfo
:
audioRendererInfo
}
let
audioRenderer
=
await
audio
.
createAudioRenderer
(
audioRendererOptions
);
var
audioStreamInfo
=
{
samplingRate
:
audio
.
AudioSamplingRate
.
SAMPLE_RATE_44100
,
channels
:
audio
.
AudioChannel
.
CHANNEL_1
,
sampleFormat
:
audio
.
AudioSampleFormat
.
SAMPLE_FORMAT_S16LE
,
encodingType
:
audio
.
AudioEncodingType
.
ENCODING_TYPE_RAW
}
var
audioRendererInfo
=
{
content
:
audio
.
ContentType
.
CONTENT_TYPE_SPEECH
,
usage
:
audio
.
StreamUsage
.
STREAM_USAGE_VOICE_COMMUNICATION
,
rendererFlags
:
1
}
var
audioRendererOptions
=
{
streamInfo
:
audioStreamInfo
,
rendererInfo
:
audioRendererInfo
}
let
audioRenderer
=
await
audio
.
createAudioRenderer
(
audioRendererOptions
);
```
2.
使用on('interrupt')方法订阅音频中断事件。
...
...
@@ -58,49 +59,49 @@ AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可
在音频中断的情况下,应用可能会碰到音频数据写入失败的问题。所以建议不感知、不处理中断的应用在写入音频数据前,使用audioRenderer.state检查播放器状态。而订阅音频中断事件,可以获取到更多详细信息,具体可参考
[
InterruptEvent
](
../reference/apis/js-apis-audio.md#interruptevent9
)
。
```
js
audioRenderer
.
on
(
'
interrupt
'
,
(
interruptEvent
)
=>
{
console
.
info
(
'
InterruptEvent Received
'
);
console
.
info
(
'
InterruptType:
'
+
interruptEvent
.
eventType
);
console
.
info
(
'
InterruptForceType:
'
+
interruptEvent
.
forceType
);
console
.
info
(
'
AInterruptHint:
'
+
interruptEvent
.
hintType
);
if
(
interruptEvent
.
forceType
==
audio
.
InterruptForceType
.
INTERRUPT_FORCE
)
{
switch
(
interruptEvent
.
hintType
)
{
// Force Pause: Action was taken by framework.
// Halt the write calls to avoid data loss.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_PAUSE
:
isPlay
=
false
;
break
;
// Force Stop: Action was taken by framework.
// Halt the write calls to avoid data loss.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_STOP
:
isPlay
=
false
;
break
;
// Force Duck: Action was taken by framework,
// just notifying the app that volume has been reduced.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_DUCK
:
break
;
// Force Unduck: Action was taken by framework,
// just notifying the app that volume has been restored.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_UNDUCK
:
break
;
}
}
else
if
(
interruptEvent
.
forceType
==
audio
.
InterruptForceType
.
INTERRUPT_SHARE
)
{
switch
(
interruptEvent
.
hintType
)
{
// Share Resume: Action is to be taken by App.
// Resume the force paused stream if required.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_RESUME
:
startRenderer
();
break
;
// Share Pause: Stream has been interrupted,
// It can choose to pause or play concurrently.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_PAUSE
:
isPlay
=
false
;
pauseRenderer
();
break
;
}
}
});
audioRenderer
.
on
(
'
interrupt
'
,
(
interruptEvent
)
=>
{
console
.
info
(
'
InterruptEvent Received
'
);
console
.
info
(
'
InterruptType:
'
+
interruptEvent
.
eventType
);
console
.
info
(
'
InterruptForceType:
'
+
interruptEvent
.
forceType
);
console
.
info
(
'
AInterruptHint:
'
+
interruptEvent
.
hintType
);
if
(
interruptEvent
.
forceType
==
audio
.
InterruptForceType
.
INTERRUPT_FORCE
)
{
switch
(
interruptEvent
.
hintType
)
{
// Force Pause: Action was taken by framework.
// Halt the write calls to avoid data loss.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_PAUSE
:
isPlay
=
false
;
break
;
// Force Stop: Action was taken by framework.
// Halt the write calls to avoid data loss.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_STOP
:
isPlay
=
false
;
break
;
// Force Duck: Action was taken by framework,
// just notifying the app that volume has been reduced.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_DUCK
:
break
;
// Force Unduck: Action was taken by framework,
// just notifying the app that volume has been restored.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_UNDUCK
:
break
;
}
}
else
if
(
interruptEvent
.
forceType
==
audio
.
InterruptForceType
.
INTERRUPT_SHARE
)
{
switch
(
interruptEvent
.
hintType
)
{
// Share Resume: Action is to be taken by App.
// Resume the force paused stream if required.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_RESUME
:
startRenderer
();
break
;
// Share Pause: Stream has been interrupted,
// It can choose to pause or play concurrently.
case
audio
.
InterruptHint
.
INTERRUPT_HINT_PAUSE
:
isPlay
=
false
;
pauseRenderer
();
break
;
}
}
});
```
3.
调用start()方法来启动/恢复播放任务。
...
...
@@ -108,24 +109,24 @@ AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可
启动完成后,渲染器状态将变更为STATE_RUNNING,然后应用可以开始读取缓冲区。
```
js
async
function
startRenderer
()
{
var
state
=
audioRenderer
.
state
;
// state should be prepared, paused or stopped.
if
(
state
!=
audio
.
AudioState
.
STATE_PREPARED
||
state
!=
audio
.
AudioState
.
STATE_PAUSED
||
state
!=
audio
.
AudioState
.
STATE_STOPPED
)
{
console
.
info
(
'
Renderer is not in a correct state to start
'
);
return
;
}
await
audioRenderer
.
start
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
info
(
'
Renderer started
'
);
}
else
{
console
.
error
(
'
Renderer start failed
'
);
}
}
async
function
startRenderer
()
{
var
state
=
audioRenderer
.
state
;
// state should be prepared, paused or stopped.
if
(
state
!=
audio
.
AudioState
.
STATE_PREPARED
||
state
!=
audio
.
AudioState
.
STATE_PAUSED
||
state
!=
audio
.
AudioState
.
STATE_STOPPED
)
{
console
.
info
(
'
Renderer is not in a correct state to start
'
);
return
;
}
await
audioRenderer
.
start
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
info
(
'
Renderer started
'
);
}
else
{
console
.
error
(
'
Renderer start failed
'
);
}
}
```
4.
调用write()方法向缓冲区写入数据。
...
...
@@ -133,83 +134,83 @@ AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可
将需要播放的音频数据读入缓冲区,重复调用write()方法写入。
```
js
async
function
writeBuffer
(
buf
)
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
error
(
'
Renderer is not running, do not write
'
);
isPlay
=
false
;
return
;
}
let
writtenbytes
=
await
audioRenderer
.
write
(
buf
);
console
.
info
(
'
Actual written bytes:
'
+
writtenbytes
);
if
(
writtenbytes
<
0
)
{
console
.
error
(
'
Write buffer failed. check the state of renderer
'
);
}
}
// Reasonable minimum buffer size for renderer. However, the renderer can accept other read sizes as well.
const
bufferSize
=
await
audioRenderer
.
getBufferSize
();
const
path
=
'
/data/file_example_WAV_2MG.wav
'
;
let
ss
=
fileio
.
createStreamSync
(
path
,
'
r
'
);
const
totalSize
=
2146166
;
// file_example_WAV_2MG.wav
let
rlen
=
0
;
let
discardHeader
=
new
ArrayBuffer
(
44
);
ss
.
readSync
(
discardHeader
);
rlen
+=
44
;
async
function
writeBuffer
(
buf
)
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
error
(
'
Renderer is not running, do not write
'
);
isPlay
=
false
;
return
;
}
let
writtenbytes
=
await
audioRenderer
.
write
(
buf
);
console
.
info
(
'
Actual written bytes:
'
+
writtenbytes
);
if
(
writtenbytes
<
0
)
{
console
.
error
(
'
Write buffer failed. check the state of renderer
'
);
}
}
var
id
=
setInterval
(()
=>
{
if
(
isPlay
||
isRelease
)
{
if
(
rlen
>=
totalSize
||
isRelease
)
{
ss
.
closeSync
();
stopRenderer
();
clearInterval
(
id
);
}
let
buf
=
new
ArrayBuffer
(
bufferSize
);
rlen
+=
ss
.
readSync
(
buf
);
console
.
info
(
'
Total bytes read from file:
'
+
rlen
);
writeBuffer
(
buf
);
}
else
{
console
.
info
(
'
check after next interval
'
);
}
}
,
30
);
// interval to be set based on audio file format
// Reasonable minimum buffer size for renderer. However, the renderer can accept other read sizes as well.
const
bufferSize
=
await
audioRenderer
.
getBufferSize
();
const
path
=
'
/data/file_example_WAV_2MG.wav
'
;
let
ss
=
fileio
.
createStreamSync
(
path
,
'
r
'
);
const
totalSize
=
2146166
;
// file_example_WAV_2MG.wav
let
rlen
=
0
;
let
discardHeader
=
new
ArrayBuffer
(
44
);
ss
.
readSync
(
discardHeader
);
rlen
+=
44
;
var
id
=
setInterval
(()
=>
{
if
(
isPlay
||
isRelease
)
{
if
(
rlen
>=
totalSize
||
isRelease
)
{
ss
.
closeSync
();
stopRenderer
();
clearInterval
(
id
);
}
let
buf
=
new
ArrayBuffer
(
bufferSize
);
rlen
+=
ss
.
readSync
(
buf
);
console
.
info
(
'
Total bytes read from file:
'
+
rlen
);
writeBuffer
(
buf
);
}
else
{
console
.
info
(
'
check after next interval
'
);
}
}
,
30
);
// interval to be set based on audio file format
```
5.
(可选)调用pause()方法或stop()方法暂停/停止渲染音频数据。
```
js
async
function
pauseRenderer
()
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
info
(
'
Renderer is not running
'
);
return
;
}
await
audioRenderer
.
pause
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_PAUSED
)
{
console
.
info
(
'
Renderer paused
'
);
}
else
{
console
.
error
(
'
Renderer pause failed
'
);
}
}
async
function
stopRenderer
()
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
||
state
!=
audio
.
AudioState
.
STATE_PAUSED
)
{
console
.
info
(
'
Renderer is not running or paused
'
);
return
;
}
await
audioRenderer
.
stop
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_STOPPED
)
{
console
.
info
(
'
Renderer stopped
'
);
}
else
{
console
.
error
(
'
Renderer stop failed
'
);
}
async
function
pauseRenderer
()
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
)
{
console
.
info
(
'
Renderer is not running
'
);
return
;
}
await
audioRenderer
.
pause
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_PAUSED
)
{
console
.
info
(
'
Renderer paused
'
);
}
else
{
console
.
error
(
'
Renderer pause failed
'
);
}
}
async
function
stopRenderer
()
{
var
state
=
audioRenderer
.
state
;
if
(
state
!=
audio
.
AudioState
.
STATE_RUNNING
||
state
!=
audio
.
AudioState
.
STATE_PAUSED
)
{
console
.
info
(
'
Renderer is not running or paused
'
);
return
;
}
await
audioRenderer
.
stop
();
state
=
audioRenderer
.
state
;
if
(
state
==
audio
.
AudioState
.
STATE_STOPPED
)
{
console
.
info
(
'
Renderer stopped
'
);
}
else
{
console
.
error
(
'
Renderer stop failed
'
);
}
}
```
...
...
@@ -218,20 +219,20 @@ AudioRenderer提供了渲染音频文件和控制播放的接口,开发者可
AudioRenderer会使用大量的系统资源,所以请确保完成相关任务后,进行资源释放。
```
js
async
function
releaseRenderer
()
{
if
(
state_
==
RELEASED
||
state_
==
NEW
)
{
console
.
info
(
'
Resourced already released
'
);
return
;
}
await
audioRenderer
.
release
();
state
=
audioRenderer
.
state
;
if
(
state
==
STATE_RELEASED
)
{
console
.
info
(
'
Renderer released
'
);
}
else
{
console
.
info
(
'
Renderer release failed
'
);
}
}
async
function
releaseRenderer
()
{
if
(
state_
==
RELEASED
||
state_
==
NEW
)
{
console
.
info
(
'
Resourced already released
'
);
return
;
}
await
audioRenderer
.
release
();
state
=
audioRenderer
.
state
;
if
(
state
==
STATE_RELEASED
)
{
console
.
info
(
'
Renderer released
'
);
}
else
{
console
.
info
(
'
Renderer release failed
'
);
}
}
```
\ No newline at end of file
zh-cn/application-dev/media/audio-stream-manager.md
浏览文件 @
fe14d589
# 音频流管理开发指导
##
场景介绍
##
简介
AudioStreamManager提供了音频流管理的方法。开发者可以通过本指导了解应用如何通过AudioStreamManager管理音频流。
##
# 工作流程
##
运作机制
在进行应用开发的过程中,开发者需要使用getStreamManager()创建一个AudioStreamManager实例,进而通过该实例管理音频流。开发者可通过调用on('audioRendererChange')、on('audioCapturerChange')监听音频播放应用和音频录制应用,在应用状态变化、设备变化、音频属性变化时获得通知。同时可通过off('audioRendererChange')、off('audioCapturerChange')取消相关事件的监听。与此同时,开发者可以通过调用(可选)使用getCurrentAudioRendererInfoArray()获取当前音频播放应用的音频流唯一ID、音频播放客户端的UID、音频状态等信息,同理可调用getCurrentAudioCapturerInfoArray()获取音频录制应用的信息。其具体调用关系可参考音频流管理调用关系图。
详细API含义可参考:
[
音频管理API文档AudioStreamManager
](
../reference/apis/js-apis-audio.md#audiostreammanager9
)
该模块提供了音频流管理调用关系图
**图1**
音频流管理调用关系图
![](
figures/zh-ch_image_audio_stream_manager.png
)
![
zh-ch_image_audio_stream_manager
](
figures/zh-ch_image_audio_stream_manager.png
)
**说明**
:在进行应用开发的过程中,开发者需要使用getStreamManager()创建一个AudioStreamManager实例,进而通过该实例管理音频流。开发者可通过调用on('audioRendererChange')、on('audioCapturerChange')监听音频播放应用和音频录制应用,在应用状态变化、设备变化、音频属性变化时获得通知。同时可通过off('audioRendererChange')、off('audioCapturerChange')取消相关事件的监听。与此同时,开发者可以通过调用(可选)使用getCurrentAudioRendererInfoArray()获取当前音频播放应用的音频流唯一ID、音频播放客户端的UID、音频状态等信息,同理可调用getCurrentAudioCapturerInfoArray()获取音频录制应用的信息。
## 开发步骤
## 开发指导
详细API含义可参考:
[
音频管理API文档AudioStreamManager
](
../reference/apis/js-apis-audio.md#audiostreammanager9
)
1.
创建AudioStreamManager实例。
在使用AudioStreamManager的API前,需要使用getStreamManager()创建一个AudioStreamManager实例。
```
js
var
audioStreamManager
=
audio
.
getStreamManager
();
var
audioManager
=
audio
.
getAudioManager
();
var
audioStreamManager
=
audioManager
.
getStreamManager
();
```
2.
(可选)使用on('audioRendererChange')监听音频渲染器更改事件。
...
...
zh-cn/application-dev/media/opensles-capture.md
浏览文件 @
fe14d589
# OpenSL ES音频录制开发指导
##
场景介绍
##
简介
开发者可以通过本文
了解到在
**OpenHarmony**
如何使用
**OpenSL ES**
进行录音相关操作;当前仅实现了部分
[
**OpenSL ES**接口
](
https://gitee.com/openharmony/third_party_opensles/blob/master/api/1.0.1/OpenSLES.h
)
,未实现接口调用
后会返回
**SL_RESULT_FEATURE_UNSUPPORTED**
。
开发者可以通过本文
档了解在
**OpenHarmony**
中如何使用
**OpenSL ES**
进行录音相关操作;当前仅实现了部分
[
**OpenSL ES**接口
](
https://gitee.com/openharmony/third_party_opensles/blob/master/api/1.0.1/OpenSLES.h
)
,因此调用未实现接口
后会返回
**SL_RESULT_FEATURE_UNSUPPORTED**
。
## 开发
步骤
## 开发
指导
以下步骤描述了在
**OpenHarmony**
如何使用
**OpenSL ES**
开发音频录音功能:
...
...
zh-cn/application-dev/media/opensles-playback.md
浏览文件 @
fe14d589
# OpenSL ES音频播放开发指导
##
场景介绍
##
简介
开发者可以通过本文
了解到在
**OpenHarmony**
如何使用
**OpenSL ES**
进行音频播放相关操作;当前仅实现了部分
[
**OpenSL ES**接口
](
https://gitee.com/openharmony/third_party_opensles/blob/master/api/1.0.1/OpenSLES.h
)
,未实现接口调用
后会返回
**SL_RESULT_FEATURE_UNSUPPORTED**
开发者可以通过本文
档了解在
**OpenHarmony**
中如何使用
**OpenSL ES**
进行音频播放相关操作;当前仅实现了部分
[
**OpenSL ES**接口
](
https://gitee.com/openharmony/third_party_opensles/blob/master/api/1.0.1/OpenSLES.h
)
,因此调用未实现接口
后会返回
**SL_RESULT_FEATURE_UNSUPPORTED**
## 开发
步骤
## 开发
指导
以下步骤描述了在
**OpenHarmony**
如何使用
**OpenSL ES**
开发音频播放功能:
...
...
@@ -58,7 +58,7 @@
5.
获取接口
**SL_IID_OH_BUFFERQUEUE**
的
**bufferQueueItf**
实例
```
```
c++
SLOHBufferQueueItf
bufferQueueItf
;
(
*
pcmPlayerObject
)
->
GetInterface
(
pcmPlayerObject
,
SL_IID_OH_BUFFERQUEUE
,
&
bufferQueueItf
);
```
...
...
zh-cn/application-dev/reference/apis/js-apis-audio.md
浏览文件 @
fe14d589
# 音频管理
音频管理提供管理音频的一些基础能力,包括对音频音量、音频设备的管理,以及对音频数据的采集和渲染等。
音频管理提供管理音频的一些基础能力,包括对音频音量、音频设备的管理,以及对音频数据的采集和渲染等。
该模块提供以下音频相关的常用功能:
...
...
@@ -687,6 +687,16 @@ async function createTonePlayer(){
| volumeGroupId
<sup>
9+
</sup>
| number | 是 | 音量组id。可用于getGroupManager入参 |
| networkId
<sup>
9+
</sup>
| string | 是 | 网络id。 |
## MicStateChangeEvent<sup>9+</sup>
麦克风状态变化时,应用接收的事件。
**系统能力:**
以下各项对应的系统能力均为SystemCapability.Multimedia.Audio.Device
| 名称 | 类型 | 必填 | 说明 |
| ---------- | ----------------------------------- | ---- | -------------------------------------------------------- |
| mute | boolean | 是 | 回调返回系统麦克风静音状态,true为静音,false为非静音。 |
## ConnectType<sup>9+</sup>
枚举,设备连接类型。
...
...
@@ -3263,6 +3273,32 @@ async function getRoutingManager(){
}
```
### on('micStateChange')<sup>9+</sup>
on(type: 'micStateChange', callback: Callback
<
MicStateChangeEvent
>
): void
监听系统麦克风状态更改事件
目前此订阅接口在单进程多AudioManager实例的使用场景下,仅最后一个实例的订阅生效,其他实例的订阅会被覆盖(即使最后一个实例没有进行订阅),因此推荐使用单一AudioManager实例进行开发。
**系统能力:**
SystemCapability.Multimedia.Audio.Device
**参数:**
| 参数名 | 类型 | 必填 | 说明 |
| -------- | -------------------------------------- | ---- | ------------------------------------------------------------ |
| type | string | 是 | 事件回调类型,支持的事件为:'micStateChange'(系统麦克风状态变化事件,检测到系统麦克风状态改变时,触发该事件)。 |
| callback | Callback
<
[
MicStateChangeEvent
](
#micstatechangeevent9
)
>
| 是 | 回调方法,返回变更后的麦克风状态。 |
**示例:**
```
js
var
audioManager
=
audio
.
getAudioManager
();
audioManager
.
getRoutingManager
.
on
(
'
micStateChange
'
,
(
micStateChange
)
=>
{
console
.
info
(
`Current microphone status is:
${
micStateChange
.
mute
}
`
);
});
```
### selectInputDevice<sup>9+</sup>
selectInputDevice(inputAudioDevices: AudioDeviceDescriptors): Promise
<
void
>
...
...
@@ -5253,7 +5289,7 @@ load(type: ToneType, callback: AsyncCallback<void>): void
| 参数名 | 类型 | 必填 | 说明 |
| :--------------| :-------------------------- | :-----| :------------------------------ |
| type | ToneType(#tonetype9) | 是 | 配置的音调类型。 |
| callback | AsyncCallback<void
\>
| 是 | 使用callback方式异步返回
缓冲区
。 |
| callback | AsyncCallback<void
\>
| 是 | 使用callback方式异步返回
结果
。 |
**示例:**
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录