Delphi控制摄像头

Delphi对摄像头的控制很简单,在System,windows和messages三个单元内已定义了所有的底层消息函数,我们只需要合理的调用它们就行了。我把摄像头的有关操作做成一个控件,这样就可以拖动窗体上直接使用了。

{************************************
 *    Camera Control for Delphi7    *
 *          Made by Rarnu           *
 *        Credit 2006.08.27         *
 *       http://rarnu.ik8.com       *
 ************************************}

unit RaCameraEye;

interface

uses
  SysUtils, Classes, Controls, Windows, Messages;

{事件声明}
type
  {开始摄像事件}
  TOnStart = procedure(Sender: TObject) of object;
  {停止摄像事件}
  TOnStop = procedure(Sender: TObject) of object;
  {开始录像事件}
  TOnStartRecord = procedure(Sender: TObject) of object;
  {停止录像事件}
  TOnStopRecord = procedure(Sender: TObject) of object;

type
  TRaCameraEye = class(TComponent)
  private
    {图像显示容器}
    fDisplay: TWinControl;
    {事件关联变量}
    fOnStart: TOnStart;
    fOnStartRecord: TOnStartRecord;
    fOnStop: TOnStop;
    fOnStopRecord: TOnStopRecord;
  protected
  public
    {构造&析构,由TComponent类覆盖而来}
    constructor Create(AOwner: TComponent); override;
    destructor Destroy; override;
    {开始摄像}
    procedure Start;
    {停止摄像}
    procedure Stop;
    {截图并保存到bmp}
    procedure SaveToBmp(FileName: string);
    {录制AVI}
    procedure RecordToAVI(FileName: string);
    {停止录制}
    procedure StopRecord;
  published
    property Display: TWinControl read fDisplay write fDisplay;
    property OnStart: TOnStart read fOnStart write fOnStart;
    property OnStop: TOnStop read fOnStop write fOnStop;
    property OnStartRecord: TOnStartRecord read fOnStartRecord write fOnStartRecord;
    property OnStopRecord: TOnStopRecord read fOnStopRecord write fOnStopRecord;
  end;

{消息常量声明}
const
  WM_CAP_START = WM_USER;
  WM_CAP_STOP = WM_CAP_START + 68;
  WM_CAP_DRIVER_CONNECT = WM_CAP_START + 10;
  WM_CAP_DRIVER_DISCONNECT = WM_CAP_START + 11;
  WM_CAP_SAVEDIB = WM_CAP_START + 25;
  WM_CAP_GRAB_FRAME = WM_CAP_START + 60;
  WM_CAP_SEQUENCE = WM_CAP_START + 62;
  WM_CAP_FILE_SET_CAPTURE_FILEA = WM_CAP_START + 20;
  WM_CAP_SEQUENCE_NOFILE = WM_CAP_START + 63;
  WM_CAP_SET_OVERLAY = WM_CAP_START + 51;
  WM_CAP_SET_PREVIEW = WM_CAP_START + 50;
  WM_CAP_SET_CALLBACK_VIDEOSTREAM = WM_CAP_START + 6;
  WM_CAP_SET_CALLBACK_ERROR = WM_CAP_START + 2;
  WM_CAP_SET_CALLBACK_STATUSA = WM_CAP_START + 3;
  WM_CAP_SET_CALLBACK_FRAME = WM_CAP_START + 5;
  WM_CAP_SET_SCALE = WM_CAP_START + 53;
  WM_CAP_SET_PREVIEWRATE = WM_CAP_START + 52;

{声明动态函数,此函数从DLL中调入,动态判断是否可用}
type
  TFunCap = function(
    lpszWindowName: PCHAR;
    dwStyle: longint;
    x: integer;
    y: integer;
    nWidth: integer;
    nHeight: integer;
    ParentWin: HWND;
    nId: integer): HWND; stdcall;

{全局变量声明}
var
  hWndC: THandle;
  FunCap: TFunCap;
  DllHandle: THandle;

procedure Register;

implementation

procedure Register;
begin
  RegisterComponents(Rarnu Components, [TRaCameraEye]);
end;

{ TRaCameraEye }

constructor TRaCameraEye.Create(AOwner: TComponent);
var
  FPointer: Pointer;{函数指针}
begin
  inherited Create(AOwner);
  fDisplay := nil;
  {通过DLL调入,如果DLL不存在,表示没有驱动}
  DllHandle := LoadLibrary(AVICAP32.DLL);
  if DllHandle <= 0 then
  begin
    MessageBox(TWinControl(Owner).Handle, 未安装摄像头驱动或驱动程序无效,不能使用此控件!, 出错, MB_OK or MB_ICONERROR);
    Destroy;{释放控件}
    Exit;
  end;
  {函数指针指向指定API}
  FPointer := GetProcAddress(DllHandle, capCreateCaptureWindowA);
  {恢复函数指针到实体函数}
  FunCap := TFunCap(FPointer);
end;

destructor TRaCameraEye.Destroy;
begin
  StopRecord;
  Stop;
  fDisplay := nil;
  {如果已加载DLL,则释放掉}
  if DllHandle > 0 then
    FreeLibrary(DllHandle);
  inherited Destroy;
end;

procedure TRaCameraEye.RecordToAVI(FileName: string);
begin
  if hWndC <> 0 then
  begin
    SendMessage(hWndC, WM_CAP_FILE_SET_CAPTURE_FILEA, 0, longint(PCHAR(FileName)));
    SendMessage(hWndC, WM_CAP_SEQUENCE, 0, 0);
    if Assigned(OnStartRecord) then
      OnStartRecord(Self);
  end;
end;

procedure TRaCameraEye.SaveToBmp(FileName: string);
begin
  if hWndC <> 0 then
    SendMessage(hWndC, WM_CAP_SAVEDIB, 0, longint(PCHAR(FileName)));
end;

procedure TRaCameraEye.Start;
var
  OHandle: THandle;
begin
  if fDisplay = nil then Exit;
  OHandle := TWinControl(Owner).Handle;
  {动态函数控制摄像头}
  hWndC := FunCap(
    My Own Capture Window,
    WS_CHILD or WS_VISIBLE,
    {规定显示范围}
    fDisplay.Left, fDisplay.Top, fDisplay.Width, fDisplay.Height,
    OHandle, 0);
  if hWndC <> 0 then
  begin
    {发送指令}
    SendMessage(hWndC, WM_CAP_SET_CALLBACK_VIDEOSTREAM, 0, 0);
    SendMessage(hWndC, WM_CAP_SET_CALLBACK_ERROR, 0, 0);
    SendMessage(hWndC, WM_CAP_SET_CALLBACK_STATUSA, 0, 0);
    SendMessage(hWndC, WM_CAP_DRIVER_CONNECT, 0, 0);
    SendMessage(hWndC, WM_CAP_SET_SCALE, 1, 0);
    SendMessage(hWndC, WM_CAP_SET_PREVIEWRATE, 66, 0);
    SendMessage(hWndC, WM_CAP_SET_OVERLAY, 1, 0);
    SendMessage(hWndC, WM_CAP_SET_PREVIEW, 1, 0);
  end;
  if Assigned(OnStart) then
    OnStart(Self);
end;

procedure TRaCameraEye.Stop;
begin
  if hWndC <> 0 then
  begin
    SendMessage(hWndC, WM_CAP_DRIVER_DISCONNECT, 0, 0);
    hWndC := 0;
    if Assigned(OnStop) then
      OnStop(Self);
  end;
end;

procedure TRaCameraEye.StopRecord;
begin
  if hWndC <> 0 then
  begin
    SendMessage(hWndC, WM_CAP_STOP, 0, 0);
    if Assigned(OnStopRecord) then
      OnStopRecord(Self);
  end;
end;

end.

 

Delphi控制摄像头

上一篇:Photoshop入门教程 美女照片变线稿


下一篇:C#用委托来动态显示日期和时间