如何控制windows的音频属性 ( 积分: 100 )

  • 主题发起人 主题发起人 xxjli
  • 开始时间 开始时间
X

xxjli

Unregistered / Unconfirmed
GUEST, unregistred user!
我在做局域网网电话,现已基本调试通过,能够实现语音聊天。用的是ACM控件。
想实现以下目的
程序运行后, 调节系统的录音源为声卡,然后播放一段声音传给对方听,接着再讲系统的录音源调节为麦克风,就可以开始语音通话了。
现在问题是,怎样用delphi调节系统的录音源??
多谢!!
 
我在做局域网网电话,现已基本调试通过,能够实现语音聊天。用的是ACM控件。
想实现以下目的
程序运行后, 调节系统的录音源为声卡,然后播放一段声音传给对方听,接着再讲系统的录音源调节为麦克风,就可以开始语音通话了。
现在问题是,怎样用delphi调节系统的录音源??
多谢!!
 
给你一个单元,可以控制系统的录音源

unit AMixer;

interface

uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms,
MMSystem;

(*
* TAudioMixer v1.4 (FREEWARE component)
* ----------------
* Released 2 Jul 2000
*
* This component can cache data from audio mixer. It has direct support for
* getting/setting volume of any control (It can set also set state of that
* "Selected"
CheckBox in standard Windows Volume Control program). You can
* better use other features of mixer, but that's more difficult than volume
* setting and you must know something about audio mixer.
*
* The mixer has following structure (as it is in this component) :
*
* Destinations (destinations should be for example: Playback, Recording and Voice commands)
* |
* |--Destination[0] (if you want to get volume of this call GeVolume (<ThisDestinationNum>,-1,...))
* | | (=0) ----
* | |--Data:TMixerLine
* | |--Controls (controls of the line, ex: Master volume, master mute)
* | | |
* | | |--Control[0]
* | | |--Control[1]
* | | |--Control[..]
* | |
* | |--Connections (ex: Wave, MIDI, CD Audio, Line-In,...)
* | |
* | |--Connection[0] (GetVolume (<ThisDestinationNum>,<ThisConnectionNumb>,...))
* | | | (=0) (=0)
* | | |--Data:TMixerLine
* | | |--Controls (here can be volume and mute)
* | | |
* | | |--Control[0]
* | | |--Control[1]
* | | |--Control[..]
* | |
* | |--Connection[1]
* | |--Connection[..]
* |
* |--Destination[1]
* |--Destination[..]
*
*
* There are many types of controls - checkbox, list, slider,... they are
* described in Windows help. Common ones are volume slider, mute checkbox or
* volume meter.
*
* This component is universal, so you can work with all controls through it,
* but this is difficult. You can simply get/set volume level by procedures
* GetVolume or SetVolume (description is near their declaration;
use - see
* example program).
*
*
* What's New
* ----------
* 1.4 (2 Jun 2000)
* - added some examples
* - great improvements by Fabrice Fouqet :
* - added properties FDriverVersion, FManufacturer, FProductId,
* FNumberOfLine and FProductName
* - added functions GetMute/SetMute and GetPeak
* - changed function GetVolume (also returns if selected control
* is stereo or mono)
* 1.16 (20 May 1999)
* - made compatible with Delphi 2.0 (this wasdo
ne by Tom Lisjac)
* 1.15 (17 May 1999)
* - corrected "Windows couldn't be shutdo
wn while TAudioMixer is being used"
problem
* (many thanks to Jean Carlo Solis Ubago, who fixed this)
* - added example showing how to set bass / treble
* 1.12 (24 Mar 1999)
* - corrected setting "selected"
state
* 1.11 (4 Jan 1999)
* - now it supports also MIXERCONTROL_CONTROLTYPE_MUX flag
* (I got SB Live! for Christmas (:-)) and my component didn't work
* properly, this corrects that problem)
* 1.1 (16 Nov 1998)
* - made compatible with Delphi 4
* - corrected memory leaks (by Ishida Wataru)
* - some another minor changes (by Ishida Wataru)
* - added another example
* - added AMixer.dcr
* 1.0 (18 Aug 1998)
* - initial version
*
*
* You can use this component freely in your programs. But if youdo
so, please
* send me an e-mail. I would like to know if it is useful.
*
* (C) Vit Kovalcik
*
* e-mail: vkovalcik@iname.com
* WWW: http://www.fi.muni.cz/~xkovalc
* (if it will not work try http://www.geocities.com/SiliconValley/Hills/1335)
*)

{Note:
When no mixer is present TAudioMixer.Destinations will be nil.
If you then
check Destinations.Count it will raise exception,
so be sure to check TAudioMixer.MixerCount first.}

type
TAudioMixer=class;

TPListFreeItemNotify=procedure (Pntr:Pointer) of object;
TMixerChange=procedure (Sender:TObject;MixerH:HMixer;ID:Integer) of object;
{MixerH is handle of mixer, which sent this message.
ID is ID of changed item (line or control).}

TPointerList=class(TObject)
private
FOnFreeItem:TPListFreeItemNotify;
Items:Tlist;
protected
function GetPointer (Ind:Integer):Pointer;
function GetCount :integer;
public
constructor Create;
destructor Destroy;
override;
procedure Clear;
procedure Add (Pntr:Pointer);
property Count:Integer read GetCount;
property Pointer[Ind:Integer]:Pointer read GetPointer;
default;
property OnFreeItem:TPListFreeItemNotify read FOnFreeItem write FOnFreeItem;
end;


TMixerControls=class(TObject)
private
heap:pointer;
FControls:TPointerList;
protected
function GetControl (Ind:Integer):PMixerControl;
function GetCount:Integer;
public
constructor Create (AMixer:TAudioMixer;
AData:TMixerLine);
destructor Destroy;
override;
property Control[Ind:Integer]:PMixerControl read GetControl;
default;
property Count:Integer read GetCount;
end;


TMixerConnection=class(TObject)
private
XMixer:TAudioMixer;
FData:TMixerLine;
FControls:TMixerControls;
public
constructor Create (AMixer:TAudioMixer;
AData:TMixerLine);
destructor Destroy;
override;
property Controls:TMixerControls read FControls;
property Data:TMixerLine read FData;
end;


TMixerConnections=class(TObject)
private
XMixer:TAudioMixer;
FConnections:TPointerList;
protected
proceduredo
FreeItem (Pntr:Pointer);
function GetConnection (Ind:Integer):TMixerConnection;
function GetCount:Integer;
public
constructor Create (AMixer:TAudioMixer;
AData:TMixerLine);
destructor Destroy;
override;
property Connection[Ind:Integer]:TMixerConnection read GetConnection;
default;
property Count:Integer read GetCount;
end;


TMixerDestination=class(TObject)
private
XMixer:TAudioMixer;
FData:TMixerLine;
FControls:TMixerControls;
FConnections:TMixerConnections;
public
constructor Create (AMixer:TAudioMixer;
AData:TMixerLine);
destructor Destroy;
override;
property Connections:TMixerConnections read FConnections;
property Controls:TMixerControls read FControls;
property Data:TMixerLine read FData;
end;


TMixerDestinations=class(TObject)
private
FDestinations:TPointerList;
protected
function GetDestination (Ind:Integer):TMixerDestination;
proceduredo
FreeItem (Pntr:Pointer);
function GetCount:Integer;
public
constructor Create (AMixer:TAudioMixer);
destructor Destroy;
override;
property Count:Integer read GetCount;
property Destination[Ind:Integer]:TMixerDestination read GetDestination;
default;
end;


TAudioMixer = class(TComponent)
private
XWndHandle:HWnd;

FDestinations:TMixerDestinations;
FMixersCount:Integer;
FMixerHandle:HMixer;
FMixerId:Integer;
FMixerCaps:TMixerCaps;
FDriverVersion: MMVERSION;
FManufacturer: String;
FProductId: Word;
FNumberOfLine: Integer;
FProductName: String;
FOnLineChange:TMixerChange;
FOnControlChange:TMixerChange;
protected
procedure SetMixerId (Value:Integer);
procedure MixerCallBack (var Msg:TMessage);
procedure CloseMixer;
published
constructor Create (AOwner:TComponent);
override;
destructor Destroy;
override;
property DriverVersion: MMVERSION read FDriverVersion;
property ProductId: WORD read FProductId;
property NumberOfLine: Integer read FNumberOfLine;
property Manufacturer: string read FManufacturer;
property ProductName: string read FProductName;
property MixerId:Integer read FMixerId write SetMixerId;
{Opened mixer - value must be in range 0..MixersCount-1
If no mixer is opened this value is -1}
property OnLineChange:TMixerChange read FOnLineChange write FOnLineChange;
property OnControlChange:TMixerChange read FOnControlChange write FOnControlChange;
public
function GetVolume (ADestination, AConnection:Integer;
var LeftVol, RightVol, Mute:Integer;
var Stereo, VolDisabled, MuteDisabled:Boolean):Boolean;
{This function return volume of selected Destination and Connection.
ADestination must be from range 0..Destinations.Count-1
AConnection must be in range 0..Destinations[ADestination].Connections.Count-1
If you want to read master volume of some Destination, you have to
set AConnection to -1.
If LeftVol, RightVol or Mute is not supported by queried connection,
it's return value will be -1.

LeftVol and RightVol are in range 0..65536

If Mute is non-zero then
the connection is silent.
If specified line is recording source then
Mute specifies if programs will
record from this connection (it is copy of "Select"
Checkbox in
standard Windows Volume Control program)
Stereo is true, then
this control is stereo.
VolDisabled or MuteDisabled is True when you cannot apply settings to this
control (but can read it).

Return value of the function is True if no error has occured,
otherwise it returns False.}
function SetVolume (ADestination, AConnection:Integer;
LeftVol, RightVol, Mute:Integer):Boolean;
{This function sets volume.
If you set RightVol to -1 and connection is stereo then
LeftVol will be
copied to RightVol.
If LeftVol or Mute is -1 then
this value will not be set.

Return value is True if ADestination and AConnection are correct, otherwise False.}

function GetPeak(ADestination, AConnection:Integer;
var LeftPeak, RightPeak:Integer):Boolean;
function GetMute(ADestination, AConnection:Integer;
var Mute:Boolean):Boolean;
function SetMute(ADestination, AConnection:Integer;
Mute:Boolean):Boolean;

property Destinations:TMixerDestinations read FDestinations;
{Ind must be in range 0..DestinationsCount-1}
property MixerCaps:TMixerCaps read FMixerCaps;
property MixerCount:Integer read FMixersCount;
{Number of mixers present in system;
mostly 1}
property MixerHandle:HMixer read FMixerHandle;
{Handle of opened mixer}
end;


procedure Register;

implementation

{------------}
{TPointerList}
{------------}

constructor TPointerList.Create;
begin

Items := TList.Create;
end;


destructor TPointerList.Destroy;
begin

Clear;
Items.Free;
end;


procedure TPointerList.Add (Pntr:Pointer);
begin

Items.Add (Pntr);
end;


function TPointerList.GetPointer (Ind:Integer):Pointer;
begin

Result := Items[Ind];
end;


procedure TPointerList.Clear;
var I:Integer;
begin

for I := 0 to Items.Count-1do
begin

If Assigned (FOnFreeItem) then

FOnFreeItem (Items)
end;

Items.Clear;
end;


function TPointerList.GetCount:Integer;
begin

Result := Items.Count;
end;


{--------------}
{TMixerControls}
{--------------}

constructor TMixerControls.Create (AMixer:TAudioMixer;
AData:TMixerLine);
var MLC:TMixerLineControls;
A,B:Integer;
P:PMixerControl;
begin

FControls := TPointerList.Create;
GetMem (P, SizeOf(TMixerControl)*AData.cControls);
heap := P;
MLC.cbStruct := SizeOf(MLC);
MLC.dwLineID := AData.dwLineID;
MLC.cbmxctrl := SizeOf(TMixerControl);
MLC.cControls := AData.cControls;
MLC.pamxctrl := P;
A := MixerGetLineControls(AMixer.MixerHandle, @MLC, MIXER_GETLINECONTROLSF_ALL);
If A = MMSYSERR_NOERROR then

begin

For B := 0 to AData.cControls-1do

begin

FControls.Add (P);
P := PMixerControl (DWORD(P) + sizeof (TMixerControl));
end;

end;

end;


destructor TMixerControls.Destroy;
begin

FControls.free;
freemem(heap);
inherited;
end;


function TMixerControls.GetControl (Ind:Integer):PMixerControl;
begin

Result := FControls.Pointer[Ind];
end;


function TMixerControls.GetCount:Integer;
begin

Result := FControls.Count;
end;


{----------------}
{TMixerConnection}
{----------------}

constructor TMixerConnection.Create (AMixer:TAudioMixer;
AData:TMixerLine);
begin

FData := AData;
XMixer := AMixer;
FControls := TMixerControls.Create (AMixer, AData);
end;


destructor TMixerConnection.Destroy;
begin

FControls.Free;
inherited;
end;


{-----------------}
{TMixerConnections}
{-----------------}

constructor TMixerConnections.Create (AMixer:TAudioMixer;
AData:TMixerLine);
var A,B:Integer;
ML:TMixerLine;
begin

XMixer := AMixer;
FConnections := TPointerList.Create;
FConnections.OnFreeItem :=do
freeitem;
ML.cbStruct := SizeOf(TMixerLine);
ML.dwDestination := AData.dwDestination;
For A := 0 to AData.cConnections-1do

begin

ML.dwSource := A;
B := MixerGetLineInfo (AMixer.MixerHandle, @ML, MIXER_GETLINEINFOF_SOURCE);
If B = MMSYSERR_NOERROR then

FConnections.Add (Pointer(TMixerConnection.Create (XMixer, ML)));
end;

end;


destructor TMixerConnections.Destroy;
begin

FConnections.Free;
inherited;
end;


procedure TMixerConnections.DoFreeItem (Pntr:Pointer);
begin

TMixerConnection(Pntr).Free;
end;


function TMixerConnections.GetConnection (Ind:Integer):TMixerConnection;
begin

Result := FConnections.Pointer[Ind];
end;


function TMixerConnections.GetCount:Integer;
begin

Result := FConnections.Count;
end;


{-----------------}
{TMixerDestination}
{-----------------}

constructor TMixerDestination.Create (AMixer:TAudioMixer;
AData:TMixerLine);
begin

FData := AData;
XMixer := AMixer;
FConnections := TMixerConnections.Create (XMixer, FData);
FControls := TMixerControls.Create (XMixer, AData);
end;


destructor TMixerDestination.Destroy;
begin

Fcontrols.Free;
FConnections.Free;
inherited;
end;


{------------------}
{TMixerDestinations}
{------------------}

constructor TMixerDestinations.Create (AMixer:TAudioMixer);
var A,B:Integer;
ML:TMixerLine;
begin

FDestinations := TPointerList.Create;
FDestinations.OnFreeItem :=do
FreeItem;
For A := 0 to AMixer.MixerCaps.cDestinations-1do

begin

ML.cbStruct := SizeOf(TMixerLine);
ML.dwDestination := A;
B := MixerGetLineInfo (AMixer.MixerHandle, @ML, MIXER_GETLINEINFOF_DESTINATION);
If B = MMSYSERR_NOERROR then

FDestinations.Add (Pointer(TMixerDestination.Create (AMixer, ML)));
end;

end;


procedure TMixerDestinations.DoFreeItem (Pntr:Pointer);
begin

TMixerDestination(Pntr).Free;
end;


destructor TMixerDestinations.Destroy;
begin

FDestinations.Free;
inherited;
end;


function TMixerDestinations.GetDestination (Ind:Integer):TMixerDestination;
begin

Result := FDestinations.Pointer[Ind];
end;


function TMixerDestinations.GetCount:Integer;
begin

Result := FDestinations.Count;
end;


{-----------}
{TAudioMixer}
{-----------}

constructor TAudioMixer.Create (AOwner:TComponent);
begin

inherited Create (AOwner);
FDestinations := nil;
XWndHandle := AllocateHWnd (MixerCallBack);
FMixersCount := mixerGetNumDevs;
FMixerId := -1;
SetMixerId (0);
end;


destructor TAudioMixer.Destroy;
begin

CloseMixer;
if XWndHandle <>
0 then

DeAllocateHwnd (XWndHandle);
inherited;
end;


procedure TAudioMixer.CloseMixer;
begin

If FMixerId >= 0 then

begin

mixerClose (FMixerHandle);
FMixerId := -1;
end;

FDestinations.Free;
FDestinations := nil;
end;


procedure TAudioMixer.SetMixerId (Value:Integer);
begin

If FMixersCount = 0 then

Exit;
If Value >= FMixersCount then

Value := FMixersCount - 1;
CloseMixer;
If Value >= 0 then

begin

If mixerOpen (@FMixerHandle, Value, XWndHandle, 0, CALLBACK_WINDOW OR MIXER_OBJECTF_MIXER) = MMSYSERR_NOERROR then

begin

FMixerId := Value;
mixerGetDevCaps (MixerId, @FMixerCaps, SizeOf (TMixerCaps));

if FMixerCaps.wMid = MM_MICROSOFT then

FManufacturer := 'Microsoft'
else

FManufacturer := IntToStr(FMixerCaps.wMid) + ' = Unknown';
FDriverVersion := FMixerCaps.vDriverVersion;
FProductId := FMixerCaps.wPid;
FProductName := StrPas(FMixerCaps.szPName);
FNumberOfLine := FMixerCaps.cDestinations;

FDestinations := TMixerDestinations.Create (Self);
end;

end;

end;


procedure TAudioMixer.MixerCallBack (var Msg:TMessage);
begin

case Msg.Msg of
MM_MIXM_LINE_CHANGE:
If Assigned (OnLineChange) then

OnLineChange (Self, Msg.wParam, Msg.lParam);
MM_MIXM_CONTROL_CHANGE:
If Assigned (OnControlChange) then

OnControlChange (Self, Msg.wParam, Msg.lParam);
else

Msg.Result := DefWindowProc (XWndHandle, Msg.Msg, Msg.WParam, Msg.LParam);
end;

end;


function TAudioMixer.GetVolume (ADestination,AConnection:Integer;var LeftVol, RightVol, Mute:Integer;var Stereo, VolDisabled, MuteDisabled:Boolean):Boolean;
var MD:TMixerDestination;
MC:TMixerConnection;
Cntrls:TMixerControls;
MCD:TMixerControlDetails;
Cntrl:PMixerControl;
A,B:Integer;
ML:TMixerLine;
details:array [0..30] of Integer;
begin

Result := False;
Stereo := False;
MD := Destinations[ADestination];
If MD <>
nil then

begin

If AConnection = -1 then

begin

Cntrls := MD.Controls;
ML := MD.Data;
end
else

begin

MC := MD.Connections[AConnection];
If MC <>
nil then

begin

Cntrls := MC.Controls;
ML := MC.Data;
end
else

Cntrls := nil;
end;

If Cntrls <>
nil then

begin

A := 0;
Result := True;
LeftVol := -1;
RightVol := -1;
Mute := -1;
while ((LeftVol = -1) OR (Mute = -1)) AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If Cntrl <>
nil then

begin

If ((Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME) OR
(Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUTE)) AND
(Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE <>
MIXERCONTROL_CONTROLF_MULTIPLE)
then

begin

if (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUTE) then

MCD.cbStruct := SizeOf(TMixerControlDetails) //1
else

MCD.cbStruct := SizeOf(TMixerControlDetails);
MCD.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

MCD.cChannels := 1
else

MCD.cChannels := ML.cChannels;
MCD.cMultipleItems := 0;
MCD.cbDetails := SizeOf(Integer);
MCD.paDetails := @details;
B := mixerGetControlDetails (FMixerHandle, @MCD, MIXER_GETCONTROLDETAILSF_VALUE);
If B = MMSYSERR_NOERROR then

begin

If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME) AND (LeftVol = -1) then

begin

VolDisabled := Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_DISABLED >
0;
If not VolDisabled then

begin

LeftVol := details[0];
If MCD.cChannels >
1 then

begin

RightVol := Details[1];
Stereo := True;
end;

end;

end
else

If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUTE) AND (Mute = -1) then

begin

MuteDisabled := Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_DISABLED >
0;//2
If not MuteDisabled then

begin

If Details[0] <>
0 then

Mute := 1
else

Mute := 0;
end;

end;

end;

end;

end;

Inc (A);
end;

If Mute = -1 then

begin

If AConnection <>
-1 then

begin

Cntrls := MD.Controls;
ML := MD.Data;
If Cntrls <>
nil then

begin

A := 0;
while (Mute = -1) AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If (Cntrl.dwControlType AND MIXERCONTROL_CONTROLTYPE_MIXER = MIXERCONTROL_CONTROLTYPE_MIXER) OR
(Cntrl.dwControlType AND MIXERCONTROL_CONTROLTYPE_MUX = MIXERCONTROL_CONTROLTYPE_MUX) then

// Mux is similar to mixer, but only one line can be selected at a time
begin

MCD.cbStruct := SizeOf(TMixerControlDetails);
MCD.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

MCD.cChannels := 1
else

MCD.cChannels := ML.cChannels;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE = MIXERCONTROL_CONTROLF_MULTIPLE then

MCD.cMultipleItems := Cntrl.cMultipleItems
else

MCD.cMultipleItems := 0;
MCD.cbDetails := 4;
MCD.paDetails := @Details;
B := mixerGetControlDetails (FMixerHandle,@MCD,MIXER_GETCONTROLDETAILSF_VALUE);
If B = MMSYSERR_NOERROR then

Mute := Details[AConnection];
end;

Inc (A);
end;

end;

end;

end;

If LeftVol = -1 then

VoldIsabled := True;
If Mute = -1 then

MuteDisabled := True;
end;

end;

end;


function TAudioMixer.SetVolume (ADestination, AConnection:Integer;
LeftVol, RightVol, Mute:Integer):Boolean;
var MD:TMixerDestination;
MC:TMixerConnection;
Cntrls:TMixerControls;
MCD:TMixerControlDetails;
Cntrl:PMixerControl;
A,B:Integer;
ML:TMixerLine;
details:array [0..30] of Integer;
VolSet,MuteSet:Boolean;
begin

Result := False;
MD := Destinations[ADestination];
If MD <>
nil then

begin

If AConnection = -1 then

begin

Cntrls := MD.Controls;
ML := MD.Data;
end
else

begin

MC := MD.Connections[AConnection];
If MC <>
nil then

begin

Cntrls := MC.Controls;
ML := MC.Data;
end
else

Cntrls := nil;
end;

If Cntrls <>
nil then

begin

A := 0;
VolSet := LeftVol = -1;
MuteSet := Mute = -1;
Result := True;
while (not VolSet OR not MuteSet) AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If Cntrl <>
nil then

begin

If ((Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME) OR
(Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUTE)) AND
(Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE <>
MIXERCONTROL_CONTROLF_MULTIPLE)
then

begin

MCD.cbStruct := SizeOf(TMixerControlDetails);
MCD.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

MCD.cChannels := 1
else

MCD.cChannels := ML.cChannels;
MCD.cMultipleItems := 0;
MCD.cbDetails := SizeOf(Integer);
MCD.paDetails := @Details;
If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME) AND not VolSet then

begin

Details[0] := LeftVol;
If RightVol = -1 then

Details[1] := LeftVol
else

Details[1] := RightVol;
VolSet := True;
end
else

If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUTE) AND not MuteSet then

begin

Details[0] := Mute;
MuteSet := True;
end;

mixerSetControlDetails (FMixerHandle, @MCD, MIXER_GETCONTROLDETAILSF_VALUE);
end;

end;

Inc (A);
end;

If not MuteSet then

begin

If AConnection <>
-1 then

begin

Cntrls := MD.Controls;
ML := MD.Data;
If Cntrls <>
nil then

begin

A := 0;
while not MuteSet AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MIXER) OR
(Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUX) then

begin

MCD.cbStruct := SizeOf(TMixerControlDetails);
MCD.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

MCD.cChannels := 1
else

MCD.cChannels := ML.cChannels;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE = MIXERCONTROL_CONTROLF_MULTIPLE then

MCD.cMultipleItems := Cntrl.cMultipleItems
else

MCD.cMultipleItems := 0;
MCD.cbDetails := 4;
MCD.paDetails := @Details;
MuteSet := True;
mixerGetControlDetails (FMixerHandle, @MCD, MIXER_GETCONTROLDETAILSF_VALUE);
if (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUX) then

For B := 0 to Cntrl.cMultipleItems-1do

Details := 0;
Details[AConnection] := Mute;
mixerSetControlDetails (FMixerHandle, @MCD, MIXER_GETCONTROLDETAILSF_VALUE);
end;

Inc (A);
end;

end;

end;

end;

end;

end;

end;


function TAudioMixer.GetMute(ADestination, AConnection: Integer;
var Mute: Boolean):Boolean;
var
MD : TMixerDestination;
MC : TMixerConnection;
mlcMixerLineControlsMute : TMIXERLINECONTROLS;
mcdMixerDataMute : TMIXERCONTROLDETAILS;
pmcMixerControlMute : PMIXERCONTROL;
pmcdsMixerDataUnsignedMute : PMIXERCONTROLDETAILSBOOLEAN;
mlMixerLine : TMixerLine;
Cntrl:PMixerControl;
Cntrls:TMixerControls;
ML:TMixerLine;
A:Integer;
details:array [0..30] of Integer;
begin

Mute := False;
Result := False;
MD := Destinations[ADestination];
if MD <>
nil then

begin

if AConnection = -1 then

mlMixerLine := MD.Data
else

begin

MC := MD.Connections[AConnection];
if MC <>
nil then

mlMixerLine := MC.Data
else

Exit;
end;


GetMem(pmcMixerControlMute, SizeOf(TMIXERCONTROL));
GetMem(pmcdsMixerDataUnsignedMute, SizeOf(TMIXERCONTROLDETAILSBOOLEAN));

with mlcMixerLineControlsMutedo

begin

cbStruct := SizeOf(TMIXERLINECONTROLS);
dwLineID := mlMixerLine.dwLineID;
dwControlType := MIXERCONTROL_CONTROLTYPE_MUTE;
cControls := 1;
cbmxctrl := SizeOf(TMIXERCONTROL);
pamxctrl := pmcMixerControlMute;
end;


if (mixerGetLineControls(FMixerHandle, @mlcMixerLineControlsMute, MIXER_GETLINECONTROLSF_ONEBYTYPE) = MMSYSERR_NOERROR) then

begin

with mcdMixerDataMutedo

begin

cbStruct := SizeOf(TMIXERCONTROLDETAILS);
dwControlID := pmcMixerControlMute^.dwControlID;
cChannels := 1;
cMultipleItems := 0;
cbDetails := SizeOf(TMIXERCONTROLDETAILSBOOLEAN);
paDetails := pmcdsMixerDataUnsignedMute;
end;


if mixerGetControlDetails(FMixerHandle, @mcdMixerDataMute, MIXER_GETCONTROLDETAILSF_VALUE) = MMSYSERR_NOERROR then

begin

Mute := pmcdsMixerDataUnsignedMute^.fValue = 1;
Result := True;
end;

end
else

begin

If (AConnection <>
-1) then

begin

Cntrls := MD.Controls;
ML := MD.Data;
If Cntrls <>
nil then

begin

A := 0;
while (Result = False) AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MIXER) OR
(Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUX) then

begin

mcdMixerDataMute.cbStruct := SizeOf(TMixerControlDetails);
mcdMixerDataMute.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

mcdMixerDataMute.cChannels := 1
else

mcdMixerDataMute.cChannels := ML.cChannels;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE = MIXERCONTROL_CONTROLF_MULTIPLE then

mcdMixerDataMute.cMultipleItems := Cntrl.cMultipleItems
else

mcdMixerDataMute.cMultipleItems := 0;
mcdMixerDataMute.cbDetails := 4;
mcdMixerDataMute.paDetails := @Details;
mixerGetControlDetails (FMixerHandle, @mcdMixerDataMute, MIXER_GETCONTROLDETAILSF_VALUE);
Result := True;
Mute := Details[AConnection] <>
0;
end;

Inc (A);
end;

end;

end;

end;


FreeMem(pmcdsMixerDataUnsignedMute);
FreeMem(pmcMixerControlMute);
end;

end;


function TAudioMixer.SetMute(ADestination, AConnection: Integer;
Mute: Boolean):Boolean;
var
MD : TMixerDestination;
MC : TMixerConnection;
mlcMixerLineControlsMute : TMIXERLINECONTROLS;
mcdMixerDataMute : TMIXERCONTROLDETAILS;
pmcMixerControlMute : PMIXERCONTROL;
pmcdsMixerDataUnsignedMute : PMIXERCONTROLDETAILSBOOLEAN;
mlMixerLine : TMixerLine;
Cntrl:PMixerControl;
Cntrls:TMixerControls;
ML:TMixerLine;
A,B:Integer;
details:array [0..30] of Integer;
begin

MD := Destinations[ADestination];
Result := False;
if MD <>
nil then

begin

if AConnection = -1 then

mlMixerLine := MD.Data
else

begin

MC := MD.Connections[AConnection];
if MC <>
nil then

mlMixerLine := MC.Data
else

Exit;
end;


GetMem(pmcMixerControlMute, SizeOf(TMIXERCONTROL));
GetMem(pmcdsMixerDataUnsignedMute, SizeOf(TMIXERCONTROLDETAILSBOOLEAN));

with mlcMixerLineControlsMutedo

begin

cbStruct := SizeOf(TMIXERLINECONTROLS);
dwLineID := mlMixerLine.dwLineID;
dwControlType := MIXERCONTROL_CONTROLTYPE_MUTE;
cControls := 0;
// 1
cbmxctrl := SizeOf(TMIXERCONTROL);
pamxctrl := pmcMixerControlMute;
end;


if (mixerGetLineControls(FMixerHandle, @mlcMixerLineControlsMute, MIXER_GETLINECONTROLSF_ONEBYTYPE) = MMSYSERR_NOERROR) then

begin

with mcdMixerDataMutedo

begin

cbStruct := SizeOf(TMixerControlDetails);
dwControlID := pmcMixerControlMute^.dwControlID;
cChannels := 1;
cMultipleItems := 0;
cbDetails := SizeOf(TMIXERCONTROLDETAILSBOOLEAN);
paDetails := pmcdsMixerDataUnsignedMute;
end;


if Mute then

pmcdsMixerDataUnsignedMute^.fValue := 1
else

pmcdsMixerDataUnsignedMute^.fValue := 0;

if (mixerSetControlDetails(FMixerHandle,@mcdMixerDataMute,MIXER_SETCONTROLDETAILSF_VALUE) = MMSYSERR_NOERROR) then

Result := True;
end
else

begin

If (AConnection <>
-1) then

begin

Cntrls := MD.Controls;
ML := MD.Data;
If Cntrls <>
nil then

begin

A := 0;
while (Result = False) AND (A <
Cntrls.Count)do

begin

Cntrl := Cntrls[A];
If (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MIXER) OR
(Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUX) then

begin

mcdMixerDataMute.cbStruct := SizeOf(TMixerControlDetails);
mcdMixerDataMute.dwControlID := Cntrl.dwControlID;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_UNIFORM >
0 then

mcdMixerDataMute.cChannels := 1
else

mcdMixerDataMute.cChannels := ML.cChannels;
If Cntrl.fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE = MIXERCONTROL_CONTROLF_MULTIPLE then

mcdMixerDataMute.cMultipleItems := Cntrl.cMultipleItems
else

mcdMixerDataMute.cMultipleItems := 0;
mcdMixerDataMute.cbDetails := 4;
mcdMixerDataMute.paDetails := @Details;
if (mixerGetControlDetails (FMixerHandle, @mcdMixerDataMute, MIXER_GETCONTROLDETAILSF_VALUE) <>
MMSYSERR_NOERROR) then

continue;
if (Cntrl.dwControlType = MIXERCONTROL_CONTROLTYPE_MUX) then

For B := 0 to Cntrl.cMultipleItems-1do

Details := 0;
If Mute then

Details[AConnection] := 1;
if (mixerSetControlDetails (FMixerHandle, @mcdMixerDataMute, MIXER_GETCONTROLDETAILSF_VALUE) = MMSYSERR_NOERROR) then

Result := True;
end;

Inc (A);
end;

end;

end;

end;


FreeMem(pmcdsMixerDataUnsignedMute);
FreeMem(pmcMixerControlMute);
end;

end;



function TAudioMixer.GetPeak (ADestination, AConnection:Integer;
var LeftPeak, RightPeak:Integer): Boolean;
var
MD : TMixerDestination;
MC : TMixerConnection;
mcdMixerDataPeak : TMIXERCONTROLDETAILS;
pmcMixerControlPeak : PMIXERCONTROL;
{ pmcdsMixerDataSignedPeak : PMIXERCONTROLDETAILSSIGNED;}
mlMixerLine : TMixerLine;
A:Integer;
Cntrls:TMixerControls;
Details:Array [1..10] of Integer;
begin

LeftPeak := 0;
RightPeak := 0;
Result := False;
MD := Destinations[ADestination];
if MD <>
nil then

begin

if AConnection = -1 then

begin

mlMixerLine := MD.Data;
Cntrls := MD.Controls;
end
else

begin

MC := MD.Connections[AConnection];
if MC <>
nil then

begin

mlMixerLine := MC.Data;
Cntrls := MC.Controls;
end
else

Exit;
end;


GetMem(pmcMixerControlPeak, SizeOf(TMIXERCONTROL));

A := 0;
while (A <
Cntrls.Count)do

begin

If Cntrls[A].dwControlType AND MIXERCONTROL_CT_CLASS_MASK = MIXERCONTROL_CT_CLASS_METER then

break;
Inc (A);

end;

If A = Cntrls.Count then

Exit;

with mcdMixerDataPeakdo

begin

cbStruct := SizeOf(TMIXERCONTROLDETAILS);
dwControlID := Cntrls[A].dwControlID;
cChannels := mlMixerLine.cChannels;
If Cntrls[A].fdwControl AND MIXERCONTROL_CONTROLF_MULTIPLE=MIXERCONTROL_CONTROLF_MULTIPLE then

cMultipleItems:=Cntrls[A].cMultipleItems
else

cMultipleItems:=0;
cbDetails := SizeOf(TMIXERCONTROLDETAILSSIGNED);
paDetails := @Details;
end;


if (mixerGetControlDetails(FMixerHandle,@mcdMixerDataPeak,MIXER_GETCONTROLDETAILSF_VALUE) = MMSYSERR_NOERROR) then

begin

LeftPeak := Details[1];
if mlMixerLine.cChannels = 2 then

RightPeak := Details[2]
else

RightPeak := LeftPeak;
Result := True;
end;


FreeMem(pmcMixerControlPeak);
end;

end;


procedure Register;
begin

RegisterComponents('Samples', [TAudioMixer]);
end;


end.
 
用MMTOOLS控件应该可以做到,里面有源码的。
 

Similar threads

S
回复
0
查看
3K
SUNSTONE的Delphi笔记
S
S
回复
0
查看
2K
SUNSTONE的Delphi笔记
S
I
回复
0
查看
736
import
I
后退
顶部