51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

独家,MP3音频淡入淡出播放和转换的JS实现


封面占位图

一、需求背景

希望特效音可以有淡入淡出效果,同时能够在视频合成的时候体现在音轨中。

二、播放淡入淡出

先讲下偏浅层应用的技术实现,那就是播放淡入淡出。

1. 使用howlerjs实现

howlerjs这个项目(https://github.com/goldfire/howler.js)我多次提到过,凡是音频播放的需求(除了那种简单的点击播放),无需任何迟疑,也不用考虑其他,就是要howlerjs。

两三万Star的项目,品质保证。

下面演示下如何实现。

假设页面上有个按钮,点击按钮播放某音频,HTML示意:

<button id="button">点击播放</button>

则下面几行JavaScript代码可以实现音频播放淡入,结束淡出的效果。

const url = './htmlbook.wav';
const sound = new Howl({
src: [url]
});
// 点击按钮,音频淡入淡出播放
button.onclick = function () {
sound.play();
sound.fade(0, 1, 1000);
setTimeout(() => {
sound.fade(1, 0, 1000);
}, sound.duration() * 1000 - 1500);
}

根据我查找的资料,howlerjs并未提供内置的结尾淡出方法,只有一个fade()方法,因此,结束时候声音淡出使用了定时器。而之所以定时器offset的时间是1500ms而不是淡出的1000ms时间,是因为案例使用的htmlbook.wav末尾有一段声音是静音,为了让淡出效果明显,才如此处理的,不代表通用场景。

眼见为实,您可以狠狠地点击这里:使用Howler.js实现音频播放淡入淡出demo

点击下图所示的按钮,就可以听到WAV音频声音fade播放的效果了。

音频播放demo示意

2. 使用原生的GainNode实现

JS Web Audio API提供了原生的音量动态函数变化方法。包括:

  • linearRampToValueAtTime() 音量线性变化
    线性音量变化示意图
  • exponentialRampToValueAtTime() 音量指数变化
    音量指数变化示意图
  • setValueCurveAtTime() 音量曲线变化
    音量曲线变化示意图
  • setTargetAtTime() 指数级音量变化,无限接近。
    指数接近变化示意图
  • setValueAtTime() 即时改变音量。

下面,我们使用GainNode对象和linearRampToValueAtTime()方法示意如何实现音频播放首尾淡入淡出效果,和上面按钮同样的HTML代码,就一个按钮,然后JS代码如下所示。

const url = './htmlbook.wav';
// 点击按钮,音频淡入淡出播放
button.onclick = function () {
// 请求数据
fetch(url).then(res => res.arrayBuffer()).then(buffer => {
const audioContext = new AudioContext();
audioContext.decodeAudioData(buffer, function (audioBuffer) {
const source = audioContext.createBufferSource();
const gainNode = audioContext.createGain();
// 起始音量静音
gainNode.gain.value = 0;
gainNode.connect(audioContext.destination);
// buffer数据设置
source.buffer = audioBuffer;
source.connect(gainNode);
source.start();
`        gainNode.gain.linearRampToValueAtTime(1.0, audioContext.currentTime + 1);
    // 时长计算
    const duration = audioBuffer.duration;
setTimeout(() =&amp;amp;gt; {
    gainNode.gain.linearRampToValueAtTime(0.01, audioContext.currentTime + 1);
}, duration * 1000 - 1200);

});

}); ` }


此时,点击按钮,就可以听到声音淡入淡出播放的效果了。例如,点击下面这个按钮:

点击播放


如果没有效果,多半是在第三方网站,狠击这里访问实例:使用GainNode实现音频淡入淡出播放demo

补充说明

根据自己和官方demo测试,exponentialRampToValueAtTime()函数执行的时候,声音是突然变化的,不知道是不是我设备的问题。

三、音频底层淡入淡出转换

如果我们希望对音频进行转换,也就是直接把原始音频转换成淡入淡出的音频,就需要处理音频的AudioBuffer数据。

这个就相对深入些了,原理什么的大家应该都不关心,不啰嗦,3,2,1,直接上代码,见:

const sliceAudio = function (buffer, start, end, fadeIn = 0, fadeOut = 0) {
  const sampleRate = buffer.sampleRate;
  const audioContext = new AudioContext();
const length = Math.round((end - start) * sampleRate);
const offset = Math.round(start * sampleRate);
const newBuffer = audioContext.createBuffer(buffer.numberOfChannels, length, sampleRate);
for (let channel = 0; channel < buffer.numberOfChannels; channel++) {
const inputData = buffer.getChannelData(channel);
const outputData = newBuffer.getChannelData(channel);
`for (let i = 0; i &lt; length; i++) {
  outputData[i] = inputData[offset + i];

// Apply fade in if (i &lt; fadeIn * sampleRate) { outputData[i] *= i / (fadeIn * sampleRate); }

// Apply fade out if (i &gt; length - fadeOut * sampleRate) { outputData[i] *= (length - i) / (fadeOut * sampleRate); } } ` } return newBuffer; }


其中:

buffer : 需要转换的AudioBuffer数据

start : 音频复制起始时间

end : 音频复制结束时间,如果结束时间就是音频时长,此参数可以设置为buffer.duration

fadeIn : 开头淡入的时长,单位是秒

fadeOut : 结束淡出的时长,单位是秒

具体如何应用呢?可以参见下面的演示页面。

您可以狠狠地点击这里:MP3/Wav AudioBuffer转换成淡入淡出音频demo

点击"转换"按钮,下面的音频播放器播放的就是被淡入淡出处理后的音频,点击现在我们可以得到这个新的WAV音频文件。

淡入淡出转换截图示意

主要应用代码示意(从点击按钮开始):

const url = './htmlbook.wav';
// 点击按钮,音频淡入淡出播放
button.onclick = function () {
// 请求数据
fetch(url).then(res => res.arrayBuffer()).then(buffer => {
const audioContext = new AudioContext();
audioContext.decodeAudioData(buffer, (audioBuffer) => {
const convertBuffer = sliceAudio(audioBuffer, 0, audioBuffer.duration, 1, 1.5);
// 转blob WAV
const wavBuffer = audioBufferToWav(convertBuffer);
const wavBlob = new Blob([wavBuffer], { type: 'audio/wav' });
const urlBlob = URL.createObjectURL(wavBlob);
`  // create download link and append to Dom
  const downloadLink = document.createElement('a');
  downloadLink.href = urlBlob;
  downloadLink.setAttribute('download', 'htmlbook-fade.wav');
  downloadLink.textContent = '下载';

// 按钮禁用 this.disabled = true; // 试听支持 audio.src = urlBlob; // 下载支持 audio.after(downloadLink); }); ` }); }


更完整的处理代码,参见上面的演示页面的源代码(演示页面省略了AudioBuffer转Wav blob数据的代码)。

四、来了来了,本周的碎碎念

周三的时候,我的新书已经是京东日榜第1了,看来口碑发酵了,感谢大家的正版支持。

日榜第1

CSS世界精讲视频继续更新,关于新书介绍的文字版,在自己博客和公众号也都发布了。

视频下面都是有抽奖活动的。

目前累计送出去14本《HTML并不简单》,3本《CSS新世界》,还包了400~500的红包,前期活动成本都1000块钱出去了。

关键还要看后期如何了,只是国内盗版盛行,好忧虑啊。

另外,不少人提供了宝贵的勘误,特别感谢,重印的时候会调整。

上周是平凡的一周,小朋友去了浦东图书馆,游了泳,家里领导周六还听了莫文蔚的演唱会,我呢继续抽空钓鱼。

钓的都是廉价鱼塘,80元还有88元的,果然难钓的一塌糊涂,前者就4条鲤鱼,后者4条半斤鲫鱼和两条太阳鱼。

没办法,作为探钓博主,不能挑地方,好与不好都要去。

下半周天很热,是不是可以找个凉快的地方避暑呢?

?

(本篇完)

赞(2)
未经允许不得转载:工具盒子 » 独家,MP3音频淡入淡出播放和转换的JS实现