Delphi-OpenCV/component/uOCVTypes.pas

430 lines
11 KiB
ObjectPascal
Raw Normal View History

// *****************************************************************
// Delphi-OpenCV Demo
// Copyright (C) 2013 Project Delphi-OpenCV
// ****************************************************************
// Contributor:
// Laentir Valetov
// email:laex@bk.ru
// ****************************************************************
// You may retrieve the latest version of this file at the GitHub,
// located at git://github.com/Laex/Delphi-OpenCV.git
// ****************************************************************
// The contents of this file are used with permission, subject to
// the Mozilla Public License Version 1.1 (the "License"); you may
// not use this file except in compliance with the License. You may
// obtain a copy of the License at
// http://www.mozilla.org/MPL/MPL-1_1Final.html
//
// Software distributed under the License is distributed on an
// "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
// implied. See the License for the specific language governing
// rights and limitations under the License.
// *******************************************************************
{$IFNDEF CLR}
{$I OpenCV.inc}
unit uOCVTypes;
{$ENDIF}
interface
Uses
{$IFDEF VER6P}
System.SysUtils,
System.Classes,
System.Generics.Collections,
Vcl.Graphics,
{$ELSE}
SysUtils,
Classes,
Generics,
Graphics,
{$ENDIF VER6P}
core.types_c,
System.Types;
Type
IocvImage = interface
['{84567F57-A399-4179-AA0F-6F8A2788F89B}']
function GetIplImage: pIplImage;
function GetisGray: Boolean;
function GrayImage: IocvImage;
function Clone: IocvImage;
function Same: IocvImage;
function AsBitmap: TBitmap;
function Crop(const roi: TCvRect): IocvImage;
property IpImage: pIplImage Read GetIplImage;
property isGray: Boolean read GetisGray;
end;
TocvImage = class(TInterfacedObject, IocvImage)
private
FImage: pIplImage;
protected
function GetIplImage: pIplImage;
function GetisGray: Boolean;
public
constructor Create(const AImage: pIplImage); overload;
constructor Create(const Bitmap: TBitmap); overload;
constructor CreateClone(const AImage: pIplImage);
constructor LoadFormFile(const FileName: String);
destructor Destroy; override;
function GrayImage: IocvImage;
function Clone: IocvImage;
function Same: IocvImage;
function AsBitmap: TBitmap;
function Crop(const roi: TCvRect): IocvImage;
property IplImage: pIplImage Read GetIplImage;
property isGray: Boolean read GetisGray;
end;
TOnOcvNotify = procedure(Sender: TObject; const IplImage: IocvImage) of object;
TOnOcvAfterTransform = TOnOcvNotify;
TOnOcvBeforeTransform = procedure(Sender: TObject; const IplImage: IocvImage; Var Transorm: Boolean) of object;
TOnOcvContour = procedure(Sender: TObject; const IplImage: IocvImage; const ContourCount: Integer; const Contours: pCvSeq)
of object;
TocvRect = Type TRect;
TocvRects = TArray<TocvRect>;
TOnOcvHaarCascade = procedure(Sender: TObject; const IplImage: IocvImage; const HaarRects: TocvRects) of object;
TOnOcvRect = procedure(Sender: TObject; const IplImage: IocvImage; const Rect: TocvRect) of object;
TOnOcvRects = procedure(Sender: TObject; const IplImage: IocvImage; const Rects: TocvRects) of object;
IocvDataReceiver = interface
['{F67DEC9E-CCE0-49D2-AB9B-AD7E1020C5DC}']
procedure TakeImage(const IplImage: IocvImage);
procedure SetVideoSource(const Value: TObject);
end;
IocvDataSource = interface
['{80640C0A-6828-42F8-83E7-DA5FD9036DFF}']
procedure AddReceiver(const OpenCVVideoReceiver: IocvDataReceiver);
procedure RemoveReceiver(const OpenCVVideoReceiver: IocvDataReceiver);
function GetName: string;
function GetImage: IocvImage;
function GetEnabled: Boolean;
property Enabled: Boolean read GetEnabled;
end;
TocvReceiverList = class(TThreadList) // <IocvDataReceiver>;
public
procedure Add(Item: IocvDataReceiver);
procedure Remove(Item: IocvDataReceiver); inline;
end;
TocvDataSource = class(TComponent, IocvDataSource)
protected
FOpenCVVideoReceivers: TocvReceiverList;
FImage: IocvImage;
function GetName: string; virtual;
procedure NotifyReceiver(const IplImage: IocvImage); virtual;
function GetImage: IocvImage;
function GetEnabled: Boolean; virtual;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
procedure AddReceiver(const OpenCVVideoReceiver: IocvDataReceiver); virtual;
procedure RemoveReceiver(const OpenCVVideoReceiver: IocvDataReceiver); virtual;
property Image: IocvImage read GetImage;
end;
TocvDataReceiver = class(TComponent, IocvDataReceiver)
private
FocvVideoSource: IocvDataSource;
protected
procedure TakeImage(const IplImage: IocvImage); virtual;
procedure SetVideoSource(const Value: TObject); virtual;
procedure SetOpenCVVideoSource(const Value: IocvDataSource); virtual;
public
destructor Destroy; override;
published
property VideoSource: IocvDataSource Read FocvVideoSource write SetOpenCVVideoSource;
end;
TocvDataSourceAndReceiver = class(TocvDataSource, IocvDataReceiver)
private
FocvVideoSource: IocvDataSource;
protected
procedure TakeImage(const IplImage: IocvImage); virtual;
procedure SetVideoSource(const Value: TObject); virtual;
procedure SetOpenCVVideoSource(const Value: IocvDataSource); virtual;
public
destructor Destroy; override;
published
property VideoSource: IocvDataSource Read FocvVideoSource write SetOpenCVVideoSource;
end;
function ocvRect(Left, Top, Right, Bottom: Integer): TocvRect;
implementation
uses
core_c,
imgproc_c,
imgproc.types_c, highgui_c;
function ocvRect(Left, Top, Right, Bottom: Integer): TocvRect;
begin
Result.Left := Left;
Result.Top := Top;
Result.Bottom := Bottom;
Result.Right := Right;
end;
{TOpenCVDataSource}
procedure TocvDataSource.AddReceiver(const OpenCVVideoReceiver: IocvDataReceiver);
begin
FOpenCVVideoReceivers.Add(OpenCVVideoReceiver);
end;
constructor TocvDataSource.Create(AOwner: TComponent);
begin
inherited;
FOpenCVVideoReceivers := TocvReceiverList.Create;
end;
destructor TocvDataSource.Destroy;
begin
FOpenCVVideoReceivers.Free;
FImage := nil;
inherited;
end;
function TocvDataSource.GetEnabled: Boolean;
begin
Result := False;
end;
function TocvDataSource.GetImage: IocvImage;
begin
Result := FImage;
end;
function TocvDataSource.GetName: string;
begin
Result := Name;
end;
procedure TocvDataSource.NotifyReceiver(const IplImage: IocvImage);
Var
R: Pointer; // IocvDataReceiver;
LockList: TList; // <IocvDataReceiver>;
begin
LockList := FOpenCVVideoReceivers.LockList;
try
for R in LockList do
IocvDataReceiver(R).TakeImage(IplImage);
finally
FOpenCVVideoReceivers.UnlockList;
end;
end;
procedure TocvDataSource.RemoveReceiver(const OpenCVVideoReceiver: IocvDataReceiver);
begin
FOpenCVVideoReceivers.Remove(OpenCVVideoReceiver);
end;
{TOpenCVDataSourceAndReceiver}
destructor TocvDataSourceAndReceiver.Destroy;
begin
if Assigned(FocvVideoSource) then
FocvVideoSource.RemoveReceiver(Self);
inherited;
end;
procedure TocvDataSourceAndReceiver.SetOpenCVVideoSource(const Value: IocvDataSource);
begin
if (FocvVideoSource <> Value) then
begin
if Assigned(FocvVideoSource) then
FocvVideoSource.RemoveReceiver(Self);
FocvVideoSource := Value;
if Assigned(FocvVideoSource) then
FocvVideoSource.AddReceiver(Self);
end;
end;
procedure TocvDataSourceAndReceiver.SetVideoSource(const Value: TObject);
begin
VideoSource := Value as TocvDataSource;
end;
procedure TocvDataSourceAndReceiver.TakeImage(const IplImage: IocvImage);
begin
end;
{TocvDataReceiver}
destructor TocvDataReceiver.Destroy;
begin
if Assigned(FocvVideoSource) then
FocvVideoSource.RemoveReceiver(Self);
inherited;
end;
procedure TocvDataReceiver.SetOpenCVVideoSource(const Value: IocvDataSource);
begin
if (FocvVideoSource <> Value) then
begin
if Assigned(FocvVideoSource) then
FocvVideoSource.RemoveReceiver(Self);
FocvVideoSource := Value;
if Assigned(FocvVideoSource) then
FocvVideoSource.AddReceiver(Self);
end;
end;
procedure TocvDataReceiver.SetVideoSource(const Value: TObject);
begin
if (Value <> Self) then
VideoSource := Value as TocvDataSource;
end;
procedure TocvDataReceiver.TakeImage(const IplImage: IocvImage);
begin
end;
{TocvImage}
function TocvImage.AsBitmap: TBitmap;
var
deep: Integer;
i, j, K, wStep, Channels: Integer;
data: PByteArray;
pb: PByteArray;
begin
if (FImage <> NIL) then
begin
Result := TBitmap.Create;
Result.Width := FImage^.Width;
Result.Height := FImage^.Height;
deep := FImage^.nChannels * FImage^.depth;
case deep of
8:
Result.PixelFormat := pf8bit;
16:
Result.PixelFormat := pf16bit;
24:
Result.PixelFormat := pf24bit;
32:
Result.PixelFormat := pf32bit;
End;
wStep := FImage^.WidthStep;
Channels := FImage^.nChannels;
data := Pointer(FImage^.imageData);
for i := 0 to FImage^.Height - 1 do
begin
pb := Result.Scanline[i];
for j := 0 to FImage^.Width - 1 do
begin
for K := 0 to Channels - 1 do
pb[3 * j + K] := data[i * wStep + j * Channels + K]
End;
End;
Result := Result;
End
else
Result := NIL;
end;
function TocvImage.Clone: IocvImage;
begin
Result := TocvImage.CreateClone(FImage);
end;
constructor TocvImage.Create(const AImage: pIplImage);
begin
FImage := AImage;
end;
constructor TocvImage.Create(const Bitmap: TBitmap);
Var
bitmapData: PByte;
begin
Assert(Bitmap.PixelFormat = pf24bit, 'only 24bit'); // <20><><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> - IPL_DEPTH_8U, 3
bitmapData := Bitmap.Scanline[0];
FImage := cvCreateImage(cvSize(Bitmap.Width, Bitmap.Height), IPL_DEPTH_8U, 3);
Move(bitmapData^, FImage^.imageData^, FImage^.imageSize);
FImage^.imageDataOrigin := nil;
FImage^.imageId := nil;
FImage^.maskROI := nil;
FImage^.roi := nil;
end;
constructor TocvImage.CreateClone(const AImage: pIplImage);
begin
FImage := cvCloneImage(AImage);
end;
function TocvImage.Crop(const roi: TCvRect): IocvImage;
Var
CropIplImage: pIplImage;
begin
CropIplImage := cvCreateImage(cvSize(roi.Width, roi.Height), FImage^.depth, FImage^.nChannels);
cvSetImageROI(FImage, roi);
cvCopyImage(FImage, CropIplImage);
cvResetImageROI(FImage);
Result := TocvImage.Create(CropIplImage);
end;
destructor TocvImage.Destroy;
begin
cvReleaseImage(FImage);
inherited;
end;
function TocvImage.GetIplImage: pIplImage;
begin
Result := FImage;
end;
function TocvImage.GetisGray: Boolean;
begin
Result := FImage^.nChannels = 1;
end;
function TocvImage.GrayImage: IocvImage;
Var
iImage: pIplImage;
begin
if isGray then
Result := Self
else
begin
iImage := cvCreateImage(cvGetSize(FImage), IPL_DEPTH_8U, 1);
cvCvtColor(FImage, iImage, CV_RGB2GRAY);
Result := TocvImage.Create(iImage);
end;
end;
constructor TocvImage.LoadFormFile(const FileName: String);
begin
FImage := cvLoadImage(PAnsiChar(AnsiString(FileName)));
end;
function TocvImage.Same: IocvImage;
begin
Result := TocvImage.Create(cvCreateImage(cvGetSize(FImage), FImage^.depth, FImage^.nChannels));
end;
{TocvReceiverList}
procedure TocvReceiverList.Add(Item: IocvDataReceiver);
begin
inherited Add(Pointer(Item));
end;
procedure TocvReceiverList.Remove(Item: IocvDataReceiver);
begin
inherited Remove(Pointer(Item));
end;
end.