从麦克风录制并在Windows Phone中播放

时间:2012-01-07 16:56:56

标签: windows-phone-7.1 windows-phone-7

我正在开发的应用程序的一部分允许用户使用麦克风录制任何内容。要从麦克风录制,我使用了Microphone Code sample by Microsoft中给出的代码。我只是对代码做了一些更改,将wav标头写入流中,这样我就可以将记录存储为隔离存储中的波形文件。

private void recordButton_Click(object sender, EventArgs e)
        {
            // Get audio data in 1/2 second chunks
            microphone.BufferDuration = TimeSpan.FromMilliseconds(500);

            // Allocate memory to hold the audio data
            buffer = new byte[microphone.GetSampleSizeInBytes(microphone.BufferDuration)];

            // Set the stream back to zero in case there is already something in it
            stream.SetLength(0);

            //Write wav header
            WriteWavHeader(microphone.SampleRate);

            // Start recording
            microphone.Start();

            SetButtonStates(false, false, true,false);
            UserHelp.Text = "record";
            StatusImage.Source = microphoneImage;
            StatusImage.Visibility = System.Windows.Visibility.Visible;
        }

        public void WriteWavHeader(int sampleRate)
        {
            const int bitsPerSample = 16;
            const int bytesPerSample = bitsPerSample / 8;
            var encoding = System.Text.Encoding.UTF8;
            // ChunkID Contains the letters "RIFF" in ASCII form (0x52494646 big-endian form).
            stream.Write(encoding.GetBytes("RIFF"), 0, 4);

            // NOTE this will be filled in later
            stream.Write(BitConverter.GetBytes(0), 0, 4);

            // Format Contains the letters "WAVE"(0x57415645 big-endian form).
            stream.Write(encoding.GetBytes("WAVE"), 0, 4);

            // Subchunk1ID Contains the letters "fmt " (0x666d7420 big-endian form).
            stream.Write(encoding.GetBytes("fmt "), 0, 4);

            // Subchunk1Size 16 for PCM.  This is the size of therest of the Subchunk which follows this number.
            stream.Write(BitConverter.GetBytes(16), 0, 4);

            // AudioFormat PCM = 1 (i.e. Linear quantization) Values other than 1 indicate some form of compression.
            stream.Write(BitConverter.GetBytes((short)1), 0, 2);

            // NumChannels Mono = 1, Stereo = 2, etc.
            stream.Write(BitConverter.GetBytes((short)1), 0, 2);

            // SampleRate 8000, 44100, etc.
            stream.Write(BitConverter.GetBytes(sampleRate), 0, 4);

            // ByteRate =  SampleRate * NumChannels * BitsPerSample/8
            stream.Write(BitConverter.GetBytes(sampleRate * bytesPerSample), 0, 4);

            // BlockAlign NumChannels * BitsPerSample/8 The number of bytes for one sample including all channels.
            stream.Write(BitConverter.GetBytes((short)(bytesPerSample)), 0, 2);

            // BitsPerSample    8 bits = 8, 16 bits = 16, etc.
            stream.Write(BitConverter.GetBytes((short)(bitsPerSample)), 0, 2);

            // Subchunk2ID Contains the letters "data" (0x64617461 big-endian form).
            stream.Write(encoding.GetBytes("data"), 0, 4);

            // NOTE to be filled in later
            stream.Write(BitConverter.GetBytes(0), 0, 4);
        }

        public void UpdateWavHeader()
        {
            //if (!stream.CanSeek) throw new Exception("Can't seek stream to update wav header");

            var oldPos = stream.Position;

            // ChunkSize  36 + SubChunk2Size
            stream.Seek(4, SeekOrigin.Begin);
            stream.Write(BitConverter.GetBytes((int)stream.Length - 8), 0, 4);

            // Subchunk2Size == NumSamples * NumChannels * BitsPerSample/8 This is the number of bytes in the data.
            stream.Seek(40, SeekOrigin.Begin);
            stream.Write(BitConverter.GetBytes((int)stream.Length - 44), 0, 4);

            stream.Seek(oldPos, SeekOrigin.Begin);
        }

        private void stopButton_Click(object sender, EventArgs e)
        {
            if (microphone.State == MicrophoneState.Started)
            {
                // In RECORD mode, user clicked the 
                // stop button to end recording
                microphone.Stop();
                UpdateWavHeader();
            }
            else if (soundInstance.State == SoundState.Playing)
            {
                // In PLAY mode, user clicked the 
                // stop button to end playing back
                soundInstance.Stop();
            }

            SetButtonStates(true, true, false,true);
            UserHelp.Text = "ready";
            StatusImage.Source = blankImage;
        }

此后,如果用户选择保存录制内容,我将其保存在隔离存储中。

应用程序中有各种页面,使用麦克风的页面就是其中之一。完成此页面的导航是从另一个页面完成的:

NavigationService.Navigate(new Uri("/Mic.xaml", UriKind.Relative));

现在,在Mic.xaml页面上,无论是否录制任何内容,如果用户点击硬件后退按钮,他都会返回旧页面。如果他再次导航到Mic.xaml页面然后尝试录制内容,则从此页面开始,录制的文件似乎无法正确录制。录音中似乎存在问题。

因此,当我们第一次导航到页面时,录制成功,但是当我们第二次导航或者导航时,它没有。

我正在模拟器上执行测试

1 个答案:

答案 0 :(得分:4)

Microsoft的示例不适用于多页方案,因为它们在离开页面时不会清理资源。

首先,将DispatcherTimer放在一个字段中:

    private DispatcherTimer dt;

然后,更改构造函数以使用此字段,而不是初始化新的dt:

    this.dt = new DispatcherTimer();

最后,覆盖OnNavigateFrom方法以检测用户何时离开页面并清理资源:

    protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
    {
        base.OnNavigatedFrom(e);

        dt.Stop();

        microphone.BufferReady -= this.microphone_BufferReady;
    }