使用 Web Audio API 和 createMediaElement 方法,您可以使用类型化数组从音频播放中获取频率数据。<audio>
元素,只要源 URL 是本地的(不是流式传输的),它就可以在大多数浏览器中工作。参见代码笔:http://codepen.io/soulwire/pen/Dscga
实际代码:
var audioCtx = new (window.AudioContext || window.webkitAudioContext)();
var audioElement = new Audio('http://crossorigin.me/http://87.230.103.9:80/top100station.mp3'); // example stream
audioElement.crossOrigin = 'anonymous';
audioElement.type = 'audio/mpeg';
var analyser = audioCtx.createAnalyser();
audioElement.addEventListener('canplay', function() {
var audioSrc = audioCtx.createMediaElementSource(audioElement);
// Bind our analyser to the media element source.
audioSrc.connect(analyser);
audioSrc.connect(audioCtx.destination);
});
var frequencyData = new Uint8Array(20);
var svgHeight = ($window.innerHeight / 2) - 20;
var svgWidth = $window.innerWidth - 20;
var barPadding = '2';
function createSvg(parent, height, width) {
return d3.select(parent).append('svg').attr('height', height).attr('width', width);
}
var svg = createSvg('.visualizer', svgHeight, svgWidth);
// Create our initial D3 chart.
svg.selectAll('rect')
.data(frequencyData)
.enter()
.append('rect')
.attr('x', function (d, i) {
return i * (svgWidth / frequencyData.length);
})
.attr('width', svgWidth / frequencyData.length - barPadding);
// Continuously loop and update chart with frequency data.
function renderChart() {
requestAnimationFrame(renderChart);
// Copy frequency data to frequencyData array.
analyser.getByteFrequencyData(frequencyData);
console.log(frequencyData);
// Update d3 chart with new data.
svg.selectAll('rect')
.data(frequencyData)
.attr('y', function(d) {
return svgHeight - d;
})
.attr('height', function(d) {
return d;
})
.style('opacity', function(d) {
return d / 255;
})
.attr('fill', function() {
return 'rgb(255, 255, 255)';
});
}
// Run the loop
renderChart();
Where .visualizer
是一个空的<div>
我正在使用 Ionic/Angular 为广播电台开发一个混合应用程序,音频流通过 Icecast (http://dir.xiph.org/)并且我遇到了以下问题:本地 mp3 的分析和可视化没有问题,但是如果您使用流 URL,analysisr.getByteFrequencyData 在 iOS Safari 中全为零,但播放正常。
回顾一下:
我知道早期版本的 Safari 中有一个错误,其中 createMediaElementSource() 会失败,但如果情况仍然如此,那么它就无法在本地文件上工作?
有任何想法吗?