使用Mozilla音频API创建哔声

时间:2011-09-10 19:57:36

标签: api audio mozilla beep

我正在研究使用mozilla音频API音源:

  function AudioDataDestination(sampleRate, readFn) {
    // Initialize the audio output.
    var audio = new Audio();
    audio.mozSetup(1, sampleRate);

    var currentWritePosition = 0;
    var prebufferSize = sampleRate / 2; // buffer 500ms
    var tail = null, tailPosition;

    // The function called with regular interval to populate 
    // the audio output buffer.
    setInterval(function() {
      var written;
      // Check if some data was not written in previous attempts.
      if(tail) {
        written = audio.mozWriteAudio(tail.subarray(tailPosition));
        currentWritePosition += written;
        tailPosition += written;
        if(tailPosition < tail.length) {
          // Not all the data was written, saving the tail...
          return; // ... and exit the function.
        }
        tail = null;
      }

      // Check if we need add some data to the audio output.
      var currentPosition = audio.mozCurrentSampleOffset();
      var available = currentPosition + prebufferSize - currentWritePosition;
      if(available > 0) {
        // Request some sound data from the callback function.
        var soundData = new Float32Array(available);
        readFn(soundData);

        // Writting the data.
        written = audio.mozWriteAudio(soundData);
        if(written < soundData.length) {
          // Not all the data was written, saving the tail.
          tail = soundData;
          tailPosition = written;
        }
        currentWritePosition += written;
      }
    }, 100);
  }

  // Control and generate the sound.

  var frequency = 0, currentSoundSample;
  var sampleRate = 44100;

  function requestSoundData(soundData) {
    if (!frequency) { 
      return; // no sound selected
    }

    var k = 2* Math.PI * frequency / sampleRate;
    for (var i=0, size=soundData.length; i<size; i++) {
      soundData[i] = Math.sin(k * currentSoundSample++);
    }        
  }

  var audioDestination = new AudioDataDestination(sampleRate, requestSoundData);

  function start() {
    currentSoundSample = 0;
    frequency = parseFloat(document.getElementById("freq").value);
  }

  function stop() {
    frequency = 0;
  }

我希望这个音调开启500ms然后关闭500ms然后重复直到按下停止按钮。我想我可以通过将样本号22,050设置为44,100到零来完成此操作。但是这种方法似乎不起作用。我认为这是因为重新填充缓冲区的功能每100ms发生一次,但现在已经超出了我的知识限制。非常感谢任何帮助。

1 个答案:

答案 0 :(得分:0)

实际上,它似乎对我来说很好。我改变了requestSoundData函数中的循环,如下所示:

for (var i=0, size=soundData.length; i<size; i++) {
  if (currentSoundSample % 44100 < 22050)
    soundData[i] = Math.sin(k * currentSoundSample);
  else
    soundData[i] = 0;
  currentSoundSample++;
}

将样品22,050到44,100设置为零,这似乎产生了您想要的效果。您可以在此处尝试代码:http://jsfiddle.net/95jCt/