home *** CD-ROM | disk | FTP | other *** search
/ Sound, Music & MIDI Collection 2 / SMMVOL2.bin / PROG / MIDICOMP.ZIP / MIDIOUT.PAS < prev    next >
Encoding:
Pascal/Delphi Source File  |  1995-07-03  |  15.8 KB  |  487 lines

  1. { $Header:   F:/delphi/midi/vcs/midiout.pas   1.5   03 Jul 1995 01:51:06   DAVEC  $ }
  2.  
  3. unit MidiOut;
  4.  
  5.   MIDI Output component.
  6.  
  7.   Properties:
  8.       DeviceID:     Windows numeric device ID for the MIDI output device.
  9.     Between 0 and (midioutGetNumDevs-1), or MIDI_MAPPER (-1).
  10.     Special value MIDI_MAPPER specifies output to the Windows MIDI mapper
  11.     Read-only while device is open, exception if changed while open
  12.  
  13.     MIDIHandle:    The output handle to the MIDI device.
  14.     0 when device is not open
  15.     Read-only, runtime-only
  16.  
  17.     ProductName: Name of the output device product that corresponds to the
  18.     DeviceID property (e.g. 'MPU 401 out').
  19.     You can write to this while the device is closed to select a particular
  20.     output device by name (the DeviceID property will change to match).
  21.     Exception if this property is changed while the device is open.
  22.  
  23.     Numdevs: Number of MIDI output devices installed on the system. This
  24.     is the value returned by midiOutGetNumDevs. It's included for 
  25.     completeness.
  26.  
  27.     See the MIDIOUTCAPS entry in MMSYSTEM.HLP for descriptions of the
  28.     following properties:
  29.         DriverVersion
  30.         Technology
  31.         Voices
  32.         Notes
  33.         ChannelMask
  34.         Support
  35.  
  36.     Error: The error code for the last MMSYSTEM error. See the MMSYSERR_
  37.     entries in MMSYSTEM.INT for possible values.
  38.  
  39.   Methods:
  40.     Open: Open MIDI device specified by DeviceID property for output
  41.  
  42.     Close: Close device
  43.  
  44.     PutMidiEvent(Event:TMidiEvent): Output a note or sysex message to the
  45.     device. This method takes a TMidiEvent object and transmits it.
  46.     Notes:
  47.       1. If the object contains a sysex event the OnMidiOutput event will
  48.           be triggered when the sysex transmission is complete.
  49.       2. You can queue up multiple blocks of system exclusive data for
  50.           transmission by chucking them at this method; they will be
  51.          transmitted as quickly as the device can manage.
  52.       3. This method will not free the TMidiEvent object, the caller
  53.           must do that. Any sysex data in the TMidiEvent is copied before
  54.          transmission so you can free the TMidiEvent immediately after
  55.          calling PutMidiEvent, even if output has not yet finished.
  56.  
  57.     PutShort(MidiMessage: Byte; Data1: Byte; Data2: Byte): Output a short
  58.     MIDI message. Handy when you can't be bothered to build a TMidiEvent.
  59.     If the message you're sending doesn't use Data1 or Data2, set them to 0.
  60.  
  61.     PutLong(SysexPointer: Pointer; msgLength: Word): Output sysex data.
  62.         SysexPointer: Pointer to sysex data to send
  63.         msgLength: Length of sysex data.
  64.     This is handy when you don't have a TMidiEvent.
  65.  
  66.     SetVolume(Left: Word, Right: Word): Set the volume of the
  67.     left and right channels on the output device (only on internal devices?).
  68.     0xFFFF is maximum volume. If the device doesn't support separate
  69.     left/right volume control, the value of the Left parameter will be used.
  70.     Check the Support property to see whether the device supports volume
  71.     control. See also other notes on volume control under midiOutSetVolume()
  72.     in MMSYSTEM.HLP.
  73.  
  74.   Events:
  75.     OnMidiOutput: Procedure called when output of a system exclusive block
  76.     is completed.
  77.  
  78.   Notes:
  79.    I haven't implemented any methods for midiOutCachePatches and 
  80.   midiOutCacheDrumpatches, mainly 'cause I don't have any way of testing
  81.   them. Does anyone really use these?
  82. }
  83.  
  84. interface
  85.  
  86. uses
  87.   SysUtils, WinTypes, WinProcs, Messages, Classes, Graphics, Controls,
  88.   Forms, Dialogs, MMSystem, Circbuf, Midiin;
  89.  
  90. type
  91.     midioutputState = (mosOpen, mosClosed);
  92.     EmidioutputError = class(Exception);
  93.  
  94.     {-------------------------------------------------------------------}
  95.     { Derived from TWinControl because we need a window handle to receive
  96.     messages from the interrupt handler in the DLL }
  97.   TMidiOutput = class(TComponent)
  98.   private
  99.     Handle: THandle;                { Window handle used for callback notification }
  100.     FDeviceID: Integer;                { MIDI device ID }
  101.     FMIDIHandle: Hmidiout;        { Handle to output device }
  102.     FState: midioutputState;    { Current device state }
  103.     PCtlInfo: PMidiCtlInfo;    { Pointer to control info for DLL }
  104.  
  105.     FError: Word;    { Last MMSYSTEM error }
  106.  
  107.     { Stuff from midioutCAPS }
  108.     FDriverVersion: Version;    { Driver version from midioutGetDevCaps }
  109.     FProductName: string;         { product name }
  110.     FTechnology: Word;            { Type of MIDI output device }
  111.     FVoices: Word;                { Number of voices (internal synth) }
  112.     FNotes: Word;                { Number of notes (internal synth) }
  113.     FChannelMask: Word;            { Bit set for each MIDI channels that the 
  114.                                   device responds to (internal synth) }
  115.     FSupport: Longint;            { Technology supported (volume control,
  116.                                   patch caching etc. }
  117.     FNumdevs: Word;                { Number of MIDI output devices on system }
  118.  
  119.  
  120.     FOnMIDIOutput: TNotifyEvent;    { Sysex output finished }
  121.  
  122.   protected
  123.     procedure MidiOutput(var Message: TMessage); 
  124.     procedure SetDeviceID(DeviceID: Integer);
  125.     procedure SetProductName( NewProductName: String );
  126.     function midioutErrorString( WError: Word ): String;
  127.  
  128.   public
  129.     { Properties }
  130.     property MIDIHandle: Hmidiout read FMIDIHandle;
  131.     property DriverVersion: Version     { Driver version from midioutGetDevCaps }
  132.                 read FDriverVersion;
  133.     property Technology: Word            { Type of MIDI output device }
  134.                 read FTechnology;
  135.     property Voices: Word                { Number of voices (internal synth) }
  136.                 read FVoices;
  137.     property Notes: Word                { Number of notes (internal synth) }
  138.                 read FNotes;
  139.     property ChannelMask: Word            { Bit set for each MIDI channels that the }
  140.                 read FChannelMask;      { device responds to (internal synth) }
  141.     property Support: Longint            { Technology supported (volume control, }
  142.                 read FSupport;          { patch caching etc. }
  143.     property Error: Word read FError;
  144.     property Numdevs: Word read FNumdevs;
  145.  
  146.     { Methods }
  147.     function Open: Boolean;
  148.     function Close: Boolean;
  149.     procedure PutMidiEvent( theEvent: TMidiEvent);
  150.     procedure PutShort(MidiMessage: Byte; Data1: Byte; Data2: Byte);
  151.     procedure PutLong(SysexPointer: Pointer; msgLength: Word);
  152.  
  153.     procedure SetVolume( Left: Word; Right:Word );
  154.     constructor Create(AOwner:TComponent); override;
  155.     destructor Destroy; override;
  156.  
  157.    { Some functions to decode and classify incoming messages would be nice }
  158.  
  159.   published
  160.     { TODO: Property editor with dropdown list of product names }
  161.     property ProductName: String read FProductName write SetProductName;
  162.  
  163.     property DeviceID: Integer read FDeviceID write SetDeviceID default 0;
  164.     { TODO: midiOutGetVolume? Or two properties for Left and Right volume?
  165.       Is it worth it??
  166.             midiOutMessage?? Does anyone use this? }
  167.  
  168.     { Events }
  169.     property Onmidioutput: TNotifyEvent
  170.         read FOnmidioutput
  171.         write FOnmidioutput;
  172. end;
  173.  
  174. procedure Register;
  175.  
  176. {-------------------------------------------------------------------}
  177. implementation
  178.  
  179. { This is the callback procedure in the external DLL. 
  180.   It's used when midioutOpen is called by the Open method. 
  181.   There are special requirements and restrictions for this callback 
  182.   procedure (see midioutOpen in MMSYSTEM.HLP) so it's impractical to 
  183.   make it an object method }
  184. function midiHandler(
  185.       hmidiout: Hmidiout;
  186.       wMsg: Word;
  187.       dwInstance: Longint;
  188.       dwParam1: Longint;
  189.       dwParam2: Longint): Boolean; far; external 'DELPHMID';
  190.  
  191.  
  192. {-------------------------------------------------------------------}
  193. constructor Tmidioutput.Create(AOwner:TComponent);
  194. begin
  195.     inherited Create(AOwner);
  196.     FState := mosClosed;
  197.     FNumdevs := midiOutGetNumDevs;
  198.  
  199.     { Set defaults }
  200.     SetDeviceID(0);    { TODO: Exception if no MIDI devices installed? }
  201.  
  202.     { Create the window for callback notification }
  203.     if not (csDesigning in ComponentState) then
  204.         begin
  205.         Handle := AllocateHwnd(MidiOutput);
  206.         end;
  207.  
  208. end;
  209.  
  210. {-------------------------------------------------------------------}
  211. destructor Tmidioutput.Destroy;
  212. begin
  213.     if FState = mosOpen then
  214.         Close;
  215.     if (PCtlInfo <> Nil) then
  216.         GlobalSharedLockedFree( PCtlinfo^.hMem, PCtlInfo );
  217.     DeallocateHwnd(Handle);
  218.     inherited Destroy;
  219. end;
  220.  
  221. {-------------------------------------------------------------------}
  222. { Convert the numeric return code from an MMSYSTEM function to a string
  223.   using midioutGetErrorText. TODO: These errors aren't very helpful
  224.   (e.g. "an invalid parameter was passed to a system function") so
  225.   some proper error strings would be nice. }
  226. function Tmidioutput.midioutErrorString( WError: Word ): String;
  227. var
  228.     errorDesc: PChar;
  229. begin
  230.     try
  231.         errorDesc := StrAlloc(MAXERRORLENGTH);
  232.         if midioutGetErrorText(WError, errorDesc, MAXERRORLENGTH) = 0 then
  233.             result := StrPas(errorDesc)
  234.         else
  235.             result := 'Specified error number is out of range';
  236.     finally
  237.         StrDispose(errorDesc);
  238.     end;
  239. end;
  240.  
  241. {-------------------------------------------------------------------}
  242. { Set the output device ID and change the other properties to match }
  243. procedure Tmidioutput.SetDeviceID(DeviceID: Integer);
  244. var
  245.     midioutCaps: TmidioutCaps;
  246. begin
  247.     if FState = mosOpen then
  248.         raise EmidioutputError.Create('Change to DeviceID while device was open')
  249.     else
  250.         if (DeviceID >= midioutGetNumDevs) And (DeviceID <> MIDI_MAPPER) then
  251.             raise EmidioutputError.Create('Invalid device ID')
  252.         else
  253.             begin
  254.             FDeviceID := DeviceID;
  255.  
  256.             { Set the name and other midioutCAPS properties to match the ID }
  257.             FError :=
  258.                 midioutGetDevCaps(DeviceID, @midioutCaps, sizeof(TmidioutCaps));
  259.             if Ferror > 0 then
  260.                 raise EmidioutputError.Create(midioutErrorString(FError));
  261.  
  262.             with midiOutCaps do
  263.                 begin
  264.                 FProductName := StrPas(szPname);
  265.                 FDriverVersion := vDriverVersion;
  266.                 FTechnology := wTechnology;
  267.                 FVoices := wVoices;
  268.                 FNotes := wNotes;
  269.                 FChannelMask := wChannelMask;
  270.                 FSupport := dwSupport;
  271.                 end;
  272.  
  273.             end;
  274. end;
  275.  
  276. {-------------------------------------------------------------------}
  277. { Set the product name property and put the matching output device number 
  278.   in FDeviceID.
  279.   This is handy if you want to save a configured output/output device
  280.   by device name instead of device number, because device numbers may
  281.   change if users install or remove MIDI devices.
  282.   Exception if output device with matching name not found,
  283.   or if output device is open }
  284. procedure Tmidioutput.SetProductName( NewProductName: String );
  285. var
  286.     midioutCaps: TmidioutCaps;
  287.     testDeviceID: Integer;
  288.     testProductName: String;
  289. begin
  290.     if FState = mosOpen then
  291.         raise EmidioutputError.Create('Change to ProductName while device was open')
  292.     else
  293.         begin
  294.        { Loop uses -1 to test for MIDI_MAPPER as well }
  295.         for testDeviceID := -1 To (midioutGetNumDevs-1) do
  296.             begin
  297.             FError :=
  298.                 midioutGetDevCaps(testDeviceID, @midioutCaps, sizeof(TmidioutCaps));
  299.             if Ferror > 0 then
  300.                 raise EmidioutputError.Create(midioutErrorString(FError));
  301.             testProductName := StrPas(midioutCaps.szPname);
  302.             if testProductName = NewProductName then
  303.                 begin
  304.                 FProductName := NewProductName;
  305.                 Break;
  306.                 end;
  307.             end;
  308.         if FProductName <> NewProductName then
  309.             raise EmidioutputError.Create('MIDI output Device ' +
  310.                 NewProductName + ' not installed')
  311.         else
  312.             SetDeviceID(testDeviceID);
  313.     end;
  314. end;
  315.  
  316.  
  317. {-------------------------------------------------------------------}
  318. function Tmidioutput.Open: Boolean;
  319. var
  320.     hMem: THandle;
  321. begin
  322.     try  
  323.         { Create the control info for the DLL }
  324.         if (PCtlInfo = Nil) then
  325.             begin
  326.             PCtlInfo := GlobalSharedLockedAlloc( Sizeof(TCtlInfo), hMem );
  327.             PctlInfo^.hMem := hMem;
  328.             end;
  329.  
  330.         PctlInfo^.pBuffer := Nil;    { Not used for output }
  331.         Pctlinfo^.hWindow := Handle;    { Control's window handle }
  332.         FError := midioutOpen(@FMidiHandle, FDeviceId,
  333.                         Longint(@midiHandler),
  334.                         Longint(PCtlInfo),
  335.                         CALLBACK_FUNCTION);
  336.         If (FError <> 0) then
  337.             { TODO: use CreateFmtHelp to add MIDI device name/ID to message }
  338.             raise EmidioutputError.Create(midioutErrorString(FError));
  339.  
  340.         FState := mosOpen;
  341.  
  342.     except
  343.         if PCtlInfo <> Nil then
  344.             begin
  345.             GlobalSharedLockedFree(PCtlInfo^.hMem, PCtlInfo);
  346.             PCtlInfo := Nil;
  347.             end;
  348.     end;
  349.  
  350. end;
  351.  
  352. {-------------------------------------------------------------------}
  353. procedure TMidiOutput.PutShort(MidiMessage: Byte; Data1: Byte; Data2: Byte);
  354. var
  355.     thisMsg: Longint;
  356. begin
  357.     thisMsg := Longint(MidiMessage) Or
  358.         (Longint(Data1) shl 8) Or
  359.         (Longint(Data2) shl 16);
  360.  
  361.     FError := midiOutShortMsg(FMidiHandle, thisMsg);
  362.     if Ferror > 0 then
  363.         raise EmidioutputError.Create(midioutErrorString(FError));
  364. end;
  365.  
  366. {-------------------------------------------------------------------}
  367. procedure TMidiOutput.PutLong(SysexPointer: Pointer; msgLength: Word);
  368. { Notes: This works asynchronously; you send your sysex output by
  369. calling this function, which returns immediately. When the MIDI device
  370. driver has finished sending the data the MidiOutPut function in this
  371. component is called, which will in turn call the OnMidiOutput method
  372. if the component user has defined one. }
  373.  
  374. var
  375.     MyMidiHdr: TMyMidiHdr;
  376. begin
  377.     { Initialize the header and allocate buffer memory }
  378.     MyMidiHdr := TMyMidiHdr.Create(msgLength);
  379.  
  380.     { Copy the data over to the MidiHdr buffer
  381.       We can't just use the caller's PChar because the buffer memory
  382.       has to be global, shareable, and locked. }
  383.     StrMove(MyMidiHdr.SysexPointer, SysexPointer, msgLength);
  384.  
  385.     { Store the MyMidiHdr address in the header so we can find it again quickly }
  386.     MyMidiHdr.hdrPointer^.dwUser := Longint(MyMidiHdr);
  387.  
  388.     { Get MMSYSTEM's blessing for this header }
  389.     FError := midiOutPrepareHeader(FMidiHandle,MyMidiHdr.hdrPointer,
  390.         sizeof(TMIDIHDR));
  391.     if Ferror > 0 then
  392.         raise EMidiOutputError.Create(MidiOutErrorString(FError));
  393.  
  394.     { Send it }
  395.     FError := midiOutLongMsg(FMidiHandle, MyMidiHdr.hdrPointer,
  396.         sizeof(TMIDIHDR));
  397.     if Ferror > 0 then
  398.         raise EMidiOutputError.Create(MidiOutErrorString(FError));
  399.  
  400. end;
  401.  
  402. {-------------------------------------------------------------------}
  403. procedure Tmidioutput.PutMidiEvent(theEvent:TMidiEvent);
  404. var
  405.     thisMsg: Longint;
  406. begin
  407.     if FState <> mosOpen then
  408.         raise EMidiOutputError.Create('MIDI Output device not open');
  409.  
  410.     with theEvent do
  411.         begin
  412.         if Sysex = Nil then
  413.             begin
  414.             PutShort(MidiMessage, Data1, Data2)
  415.             end
  416.         else
  417.             PutLong(Sysex, SysexLength);
  418.     end;
  419. end;
  420.  
  421. {-------------------------------------------------------------------}
  422. function Tmidioutput.Close: Boolean;
  423. begin
  424.     if FState = mosOpen then
  425.         begin
  426.         FError := midioutReset(FMidiHandle);
  427.         if Ferror <> 0 then
  428.             raise EMidiOutputError.Create(MidiOutErrorString(FError));
  429.     
  430.         FError := midioutClose(FMidiHandle);
  431.         if Ferror <> 0 then
  432.             raise EMidiOutputError.Create(MidiOutErrorString(FError));
  433.         end;
  434.  
  435.     FMidiHandle := 0;
  436.     FState := mosClosed;
  437.  
  438. end;
  439.  
  440. {-------------------------------------------------------------------}
  441. procedure TMidiOutput.SetVolume( Left: Word; Right:Word );
  442. var
  443.     dwVolume: Longint;
  444. begin
  445.     dwVolume := (Longint(Left) shl 16) Or Right;
  446.     FError := midiOutSetVolume(DeviceID, dwVolume);
  447.     if Ferror <> 0 then
  448.         raise EMidiOutputError.Create(MidiOutErrorString(FError));
  449. end;
  450.  
  451. {-------------------------------------------------------------------}
  452. procedure Tmidioutput.midioutput( var Message: TMessage );
  453. { Triggered when sysex output from PutLong is complete }
  454. var
  455.     MyMidiHdr: TMyMidiHdr;
  456.     thisHdr: PMidiHdr;
  457. begin
  458.     if Message.Msg = Mom_Done then
  459.         begin
  460.         { Find the MIDIHDR we used for the output. Message.lParam is its address }
  461.         thisHdr := PMidiHdr(Message.lParam);
  462.  
  463.         { Remove it from the output device }
  464.         midiOutUnprepareHeader(FMidiHandle, thisHdr, sizeof(TMIDIHDR));
  465.  
  466.         { Get the address of the MyMidiHdr containing this MIDIHDR.
  467.             We stored this address in the PutLong procedure }
  468.         MyMidiHdr := TMyMidiHdr(thisHdr^.dwUser);
  469.  
  470.         { Header and copy of sysex data no longer required since output is complete }
  471.         MyMidiHdr.Free;
  472.  
  473.         { Call the user's event handler if any }
  474.         if Assigned(FOnmidioutput) then
  475.             FOnmidioutput(Self);
  476.         end;
  477. end;
  478.  
  479. {-------------------------------------------------------------------}
  480. procedure Register;
  481. begin
  482.   RegisterComponents('Samples', [Tmidioutput]);
  483. end;
  484.  
  485. end.
  486.