我将您的问题解释为:
如何淡出 .wav 文件的音频SoundPlayer
而不是立即停止它?
简短的回答是,您无法SoundPlayer
单独完成。您需要使用winmm.dll
库中的几个 api。
如果您对 WPF 感到满意,那么在其 volume 属性上使用 aMediaElement
可能DoubleAnimation
是一种更好的方法。有关这方面的信息,请参阅此问题。
但是,如果您真的想使用 WinForms,以下是应该可以工作的实现。
关于注意事项Form1
:
- 两个按钮,分别命名为
startButton
和stopButton
- 一个叫
timer1
我用来淡出声音的计时器
这是使它工作的代码:
using System;
using System.Media;
using System.Runtime.InteropServices;
using System.Windows.Forms;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
[DllImport("winmm.dll", EntryPoint = "waveOutGetVolume")]
private static extern int WaveOutGetVolume(IntPtr hwo, out uint dwVolume);
[DllImport("winmm.dll", EntryPoint="waveOutSetVolume")]
private static extern int WaveOutSetVolume(IntPtr hwo, uint dwVolume);
private SoundPlayer player = new SoundPlayer();
// a crude delta time field
private float totalElapsedTime;
// tweak this value to determine how quickly you want the fade to happen
private const float Velocity = 0.001f;
public Form1()
{
this.InitializeComponent();
// i was using 100 milliseconds as my "frame rate"
this.timer1.Interval = 100;
this.stopButton.Enabled = false;
}
private void startButton_Click(object sender, EventArgs e)
{
// sets the audio device volume to the max.
// this is not the computer's volume so it won't
// blast out your ear drums by doing this unless you
// have the computer volume super high - which is not this
// code's fault
WaveOutSetVolume(IntPtr.Zero, uint.MaxValue);
this.startButton.Enabled = false;
this.stopButton.Enabled = true;
this.totalElapsedTime = 0f;
this.player.SoundLocation = @"Music File.wav";
this.player.Load();
this.player.Play();
}
private void stopButton_Click(object sender, EventArgs e)
{
// being the stop
this.timer1.Start();
this.stopButton.Enabled = false;
}
private void timer1_Tick(object sender, EventArgs e)
{
// amount to interpolate (value between 0 and 1 inclusive)
float amount = Math.Min(1f, this.totalElapsedTime * Velocity);
// the new channel volume after a lerp
float lerped = Lerp(ushort.MaxValue, 0, amount);
// each channel's volume is actually represented as a ushort
ushort channelVolume = (ushort)lerped;
// the new volume for all the channels
uint volume = (uint)channelVolume | ((uint)channelVolume << 16);
// sets the volume
WaveOutSetVolume(IntPtr.Zero, volume);
// checks if the interpolation is finished
if (amount >= 1f)
{
// stop the timer
this.timer1.Stop();
// stop the player
this.player.Stop();
// stop is complete so let user start again
this.startButton.Enabled = true;
}
// add the elapsed milliseconds (very crude delta time)
this.totalElapsedTime += this.timer1.Interval;
}
public static float Lerp(float value1, float value2, float amount)
{
// does a linear interpolation
return (value1 + ((value2 - value1) * amount));
}
}
}
希望这应该是不言自明的。我想大多数人会认为这是一种粗略的“游戏循环esqe”方法,但它似乎运作良好。这里需要注意的是,调整淡入淡出的速度是Velocity
恒定的。
代码设置为在 1 秒内淡出。timer1.Interval
通过查看和很容易弄清楚Velocity
。在 1000 毫秒(10 个计时器滴答声)之后,然后 1000 * 0.001 = 1 导致停止代码完成。
改变的唯一原因timer1.Interval
是让更多的“褪色”发生。当前设置在停止前会进行 10 次音量衰减。