在 Delphi 下使用 DirectSound (4): 设置音量、相位、播放频率和播放位置


通过 IDirectSoundBuffer 的 SetVolume、SetPan、SetFrequency、SetCurrentPosition 方法可以简单进行这些设置.
同时 IDirectSoundBuffer 也有对应的 GetVolume、GetPan、GetFrequency、GetCurrentPosition 方法.

关键的一点是如果能让缓冲区接受音量、相位和频率的设置, 必须在建立缓冲区时指定相应的标志.

下面的常量说明了它们的取值范围:
DSBVOLUME_MAX = 0;      //音量最大值, 保持在控制面板设置的音量
DSBVOLUME_MIN = -10000; //音量最小值

DSBPAN_LEFT   = -10000; //左声道
DSBPAN_CENTER = 0;      //均衡
DSBPAN_RIGHT  = 10000;  //右声道

DSBFREQUENCY_ORIGINAL = 0;      //使用默认
DSBFREQUENCY_MIN      = 100;    //频率最小值
DSBFREQUENCY_MAX      = 200000; //频率最大值, 在 DirectSound 9.0 之下的版本, 此值是 100000

在上一个例子中, 最占篇幅的就是那两个函数; 为了更方便使用, 把它们做在了一个 TReadWaveFile 类里:
{实现 TReadWaveFile 类的单元}
unit ReadWaveFile;

interface

uses Windows, Classes, SysUtils, MMSystem;

type
TReadWaveFile = class
private
  FFileHandle: HMMIO;
  FFormat: TWaveFormatEx;
  FSize: DWORD;
public
  constructor Create;
  destructor Destroy; override;
  function Open(FileName: string): Boolean;            //打开文件并读取信息
  function Read(pDest: Pointer; Size: DWORD): Boolean; //读出波形数据
  property Format: TWaveFormatEx read FFormat;         //读出格式数据
  property Size: DWORD read FSize;                     //读出波形数据的大小
end;

implementation

{ TReadWaveFile }

constructor TReadWaveFile.Create;
begin
  inherited;

end;

destructor TReadWaveFile.Destroy;
begin
  if FFileHandle > 0 then mmioClose(FFileHandle, 0);
  inherited;
end;

function TReadWaveFile.Open(FileName: string): Boolean;
var
  ckiRIFF,ckiFmt,ckiData: TMMCKInfo;
begin
  Result := False;
  if not FileExists(FileName) then Exit;

  FFileHandle := mmioOpen(PChar(FileName), nil, MMIO_READ);
  if FFileHandle = 0 then Exit;
  ZeroMemory(@ckiRIFF, SizeOf(TMMCKInfo));
  mmioDescend(FFileHandle, @ckiRIFF, nil, MMIO_FINDRIFF);
  if (ckiRIFF.ckid <> FOURCC_RIFF) or (ckiRIFF.fccType <> mmioStringToFOURCC('WAVE',0)) then Exit;

  ZeroMemory(@FFormat, SizeOf(TWaveFormatEx));
  ZeroMemory(@ckiFmt, SizeOf(TMMCKInfo));
  ckiFmt.ckid := mmioStringToFOURCC('fmt', 0);

  ZeroMemory(@ckiData, SizeOf(TMMCKInfo));
  ckiData.ckid := mmioStringToFOURCC('data', 0);

  if (mmioDescend(FFileHandle, @ckiFmt, @ckiRIFF, MMIO_FINDCHUNK) = MMSYSERR_NOERROR) then
    mmioRead(FFileHandle, @FFormat, SizeOf(TWaveFormatEx));

  mmioAscend(FFileHandle, @ckiFmt, 0);

  if (mmioDescend(FFileHandle, @ckiData, @ckiRIFF, MMIO_FINDCHUNK) = MMSYSERR_NOERROR) then
    FSize := ckiData.cksize;

  Result := FFormat.wFormatTag = WAVE_FORMAT_PCM;
end;

function TReadWaveFile.Read(pDest: Pointer; Size: DWORD): Boolean;
begin
  Result := mmioRead(FFileHandle, pDest, Size) = Size;
end;

end.

测试程序用到了四个 Button 和三个 TrackBar 还有它们的默认事件:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, ComCtrls;

type
  TForm1 = class(TForm)
    Button1: TButton;     //打开并播放
    Button2: TButton;     //反复播放
    Button3: TButton;     //暂停
    Button4: TButton;     //从头播放
    TrackBar1: TTrackBar; //用于音量调节
    TrackBar2: TTrackBar; //用于相位调节
    TrackBar3: TTrackBar; //用于频率调节
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
    procedure Button3Click(Sender: TObject);
    procedure Button4Click(Sender: TObject);
    procedure TrackBar1Change(Sender: TObject);
    procedure TrackBar2Change(Sender: TObject);
    procedure TrackBar3Change(Sender: TObject);
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

uses DirectSound, ReadWaveFile; // ReadWaveFile 是 TReadWaveFile 类所在的单元

var
  myDSound: IDirectSound8;
  buf: IDirectSoundBuffer;

procedure TForm1.FormCreate(Sender: TObject);
begin
  TrackBar1.Min := DSBVOLUME_MIN;
  TrackBar1.Max := DSBVOLUME_MAX;

  TrackBar2.Min := DSBPAN_LEFT;
  TrackBar2.Max := DSBPAN_RIGHT;

  TrackBar3.Min := 100;
  TrackBar3.Max := 100000;

  Button1.Caption := '打开并播放';
  Button2.Caption := '反复播放';
  Button3.Caption := '暂停';
  Button4.Caption := '从头播放';
  System.ReportMemoryLeaksOnShutdown := true;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  bufDesc: TDSBufferDesc;
  p1: Pointer;
  n1: DWORD;
  wavPath: string;
  wav: TReadWaveFile; //
begin
  buf := nil;
  myDSound := nil;
  with TOpenDialog.Create(nil) do begin
    Filter := 'Wave File(*.wav)|*.wav';
    if Execute then wavPath := FileName;
    Free;
  end;

  wav := TReadWaveFile.Create;
  if not wav.Open(wavPath) then
  begin
    ShowMessage('只能是 PCM 格式的 WAVE 文件');
    wav.Free;
    Exit;
  end;

  DirectSoundCreate8(nil, myDSound, nil);
  myDSound.SetCooperativeLevel(Self.Handle, DSSCL_NORMAL);

  ZeroMemory(@bufDesc, SizeOf(TDSBufferDesc));
  bufDesc.dwSize := SizeOf(TDSBufferDesc);
  {指定缓冲区允许音量、相位和频率调节}
  bufDesc.dwFlags := DSBCAPS_STATIC or DSBCAPS_CTRLVOLUME or DSBCAPS_CTRLPAN or DSBCAPS_CTRLFREQUENCY;
  bufDesc.dwBufferBytes := wav.Size;
  bufDesc.lpwfxFormat := @wav.Format;

  myDSound.CreateSoundBuffer(bufDesc, buf, nil);
  buf.Lock(0, 0, @p1, @n1, nil, nil, DSBLOCK_ENTIREBUFFER);
  wav.Read(p1, n1);

  buf.Unlock(p1, n1, nil, 0);
  buf.Play(0, 0, 0);

  TrackBar1.Position := 0;
  TrackBar2.Position := 0;
  TrackBar3.Position := wav.Format.nSamplesPerSec;
  wav.Free;
end;

procedure TForm1.Button2Click(Sender: TObject);
begin
  if buf <> nil then buf.Play(0, 0, DSBPLAY_LOOPING);
end;

procedure TForm1.Button3Click(Sender: TObject);
begin
  if buf <> nil then buf.Stop;
end;

{从头播放}
procedure TForm1.Button4Click(Sender: TObject);
begin
  if buf = nil then Exit;
  buf.Stop;
  buf.SetCurrentPosition(0);
  buf.Play(0, 0, 0);
end;

{音量调节}
procedure TForm1.TrackBar1Change(Sender: TObject);
begin
  if buf <> nil then buf.SetVolume(TTrackBar(Sender).Position);
end;

{相位调节}
procedure TForm1.TrackBar2Change(Sender: TObject);
begin
  if buf <> nil then buf.SetPan(TTrackBar(Sender).Position);
end;

{频率调节}
procedure TForm1.TrackBar3Change(Sender: TObject);
begin
  if buf <> nil then buf.SetFrequency(TTrackBar(Sender).Position);
end;

procedure TForm1.FormDestroy(Sender: TObject);
begin
  buf := nil;
  myDSound := nil;
end;

end.

你可能感兴趣的:(在 Delphi 下使用 DirectSound (4): 设置音量、相位、播放频率和播放位置)