mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-11 04:28:10 +02:00
425 lines
12 KiB
ObjectPascal
425 lines
12 KiB
ObjectPascal
unit imgMain;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
Types, Classes, SysUtils,
|
|
LCLIntf, LMessages, Forms, Controls, Graphics, Dialogs, Grids, ExtCtrls,
|
|
StdCtrls, Spin, ComCtrls,
|
|
FileUtil;
|
|
|
|
type
|
|
{ TImageData is stored in the list FImageList of the modified string grid and
|
|
contains the image bitmap as well information on the position of the image. }
|
|
TImageData = class
|
|
Bitmap: TBitmap; // Bitmap to be overlayed
|
|
Col, Row: Integer; // Anchor of the bitmap in the grid
|
|
dx, dy: Integer; // Offset with respect to anchor, in pixels
|
|
destructor Destroy; override;
|
|
end;
|
|
|
|
{ Events for image clicking and moving }
|
|
TImageClickEvent = procedure(Sender: TObject; AShift: TShiftState;
|
|
AImageIndex, X, Y: Integer) of object;
|
|
TImageMoveEvent = procedure(Sender: TObject; AShift: TShiftState;
|
|
AImageIndex, dx, dy: Integer) of object;
|
|
|
|
{ Modified StringGrid with support of embedded images }
|
|
TStringGridEx = class(TStringGrid)
|
|
private
|
|
FImageList: TFPlist;
|
|
FMouseImgIndex: Integer;
|
|
FMouseDownPt: TPoint;
|
|
FOnImageClick: TImageCLickEvent;
|
|
FOnImageMove: TImageMoveEvent;
|
|
protected
|
|
procedure DoImageClick(Shift: TShiftState; X, Y: Integer); virtual;
|
|
procedure DoImageMove(Shift: TShiftState; dx, dy: Integer); virtual;
|
|
procedure DrawAllRows; override;
|
|
procedure MouseDown(Button: TMouseButton; Shift:TShiftState; X,Y:Integer); override;
|
|
procedure MouseMove(Shift: TShiftState; X,Y: Integer); override;
|
|
procedure MouseUp(Button: TMouseButton; Shift:TShiftState; X,Y:Integer); override;
|
|
procedure WMHScroll(var message : TLMHScroll); message LM_HSCROLL;
|
|
procedure WMVScroll(var message : TLMVScroll); message LM_VSCROLL;
|
|
|
|
public
|
|
constructor Create(AOwner: TComponent); override;
|
|
destructor Destroy; override;
|
|
procedure AddImage(APicture: TPicture; ACol, ARow: Integer; dx, dy: Integer;
|
|
AWidth: Integer = 0);
|
|
function GetImageRect(AImageIndex: Integer): TRect;
|
|
procedure MoveImageBy(AImageIndex, dx, dy: Integer);
|
|
function PointInImage(APoint: TPoint): Integer;
|
|
procedure UpdateImageAnchor(AImageIndex: Integer);
|
|
|
|
published
|
|
property OnImageClick: TImageClickEvent read FOnImageClick write FOnImageClick;
|
|
property OnImageMove: TImageMoveEvent read FOnImageMove write FOnImageMove;
|
|
end;
|
|
|
|
type
|
|
|
|
{ TForm1 }
|
|
|
|
TForm1 = class(TForm)
|
|
Button1: TButton;
|
|
Label1: TLabel;
|
|
Label2: TLabel;
|
|
Label3: TLabel;
|
|
OpenDialog1: TOpenDialog;
|
|
Panel1: TPanel;
|
|
Panel2: TPanel;
|
|
SpinEdit1: TSpinEdit;
|
|
SpinEdit2: TSpinEdit;
|
|
StatusBar1: TStatusBar;
|
|
procedure Button1Click(Sender: TObject);
|
|
procedure FormCreate(Sender: TObject);
|
|
private
|
|
Grid: TStringGridEx;
|
|
procedure AddImageFromFile(AFileName: String; ACol, ARow: Integer;
|
|
dx, dy: Integer; AScaledSize: Integer = 0);
|
|
procedure ImageClickHandler(Sender: TObject; Shift: TShiftState;
|
|
AImageIndex, X, Y: Integer);
|
|
procedure ImageMoveHandler(Sender: TObject; Shift: TShiftState;
|
|
AImageIndex, dX, dY: Integer);
|
|
|
|
public
|
|
|
|
end;
|
|
|
|
var
|
|
Form1: TForm1;
|
|
|
|
implementation
|
|
|
|
{$R *.lfm}
|
|
|
|
const
|
|
//JPEG_FILE = '../../../images/splash_source/cheetah.jpg';
|
|
//PNG_FILE1 = '../../../images/codetoolsdefines/da_block.png';
|
|
//PNG_FILE2 = '../../../images/icons/lazarus256x256.png';
|
|
|
|
JPEG_FILE = 'cheetah.jpg';
|
|
PNG_FILE1 = 'da_block.png';
|
|
PNG_FILE2 = 'lazarus256x256.png';
|
|
|
|
|
|
{ TForm1 }
|
|
|
|
procedure TForm1.AddImageFromFile(AFileName: String; ACol, ARow, dx, dy: Integer;
|
|
AScaledSize: Integer = 0);
|
|
// AScaledSize = 0 means: no size scaling
|
|
var
|
|
pic: TPicture;
|
|
begin
|
|
pic := TPicture.Create;
|
|
try
|
|
pic.LoadFromFile(AFileName);
|
|
Grid.AddImage(pic, ACol, ARow, dx, dy, AScaledSize);
|
|
finally
|
|
pic.Free;
|
|
end;
|
|
end;
|
|
|
|
procedure TForm1.Button1Click(Sender: TObject);
|
|
begin
|
|
if OpenDialog1.Execute then
|
|
AddImageFromFile(OpenDialog1.FileName, Grid.Col, Grid.Row, SpinEdit1.Value, SpinEdit2.Value);
|
|
end;
|
|
|
|
procedure TForm1.FormCreate(Sender: TObject);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Grid := TStringGridEx.Create(self);
|
|
Grid.Parent := self;
|
|
Grid.Align := alClient;
|
|
Grid.RowCount := 13; //100;
|
|
Grid.ColCount := 10; //30;
|
|
Grid.Options := Grid.Options + [goThumbTracking, goColSizing, goRowSizing, goEditing];
|
|
Grid.MouseWheelOption := mwGrid;
|
|
Grid.Col := 3;
|
|
Grid.Row := 16;
|
|
Grid.OnImageClick := @ImageClickHandler;
|
|
Grid.OnImageMove := @ImageMoveHandler;
|
|
|
|
for i:=1 to Grid.ColCount-1 do
|
|
Grid.Cells[i, 0] := 'column ' + IntToStr(i);
|
|
for i:=1 to Grid.RowCount-1 do
|
|
Grid.Cells[0, i] := 'row ' + IntToStr(i);
|
|
|
|
ActiveControl := Grid;
|
|
|
|
// col row dx dy width
|
|
AddImageFromFile(JPEG_FILE, 1, 1, 5, 5, 300); // Offset by 5 pixels, scaled to width 300
|
|
AddImageFromFile(PNG_FILE1, 2, 12, 0, 0);
|
|
AddImageFromFile(PNG_FILE2, 5, 2, 0, 0);
|
|
end;
|
|
|
|
procedure TForm1.ImageClickHandler(Sender: TObject; Shift: TShiftState;
|
|
AImageIndex, X, Y: Integer);
|
|
begin
|
|
Statusbar1.SimpleText := Format('Image #%d clicked at x = %d, y = %d', [AImageIndex, X, Y]);
|
|
end;
|
|
|
|
procedure TForm1.ImageMoveHandler(Sender: TObject; Shift: TShiftState;
|
|
AImageIndex, dX, dY: Integer);
|
|
begin
|
|
if [ssLeft, ssCtrl] * Shift = [ssLeft, ssCtrl] then begin
|
|
// Moves the image by dx, dy pixels
|
|
Grid.MoveImageBy(AImageIndex, dx, dy);
|
|
// Resets the image anchor such that the top-left image corner is contained
|
|
// in the anchor cell.
|
|
Grid.UpdateImageAnchor(AImageIndex);
|
|
end;
|
|
end;
|
|
|
|
|
|
{ TImageData }
|
|
|
|
destructor TImageData.Destroy;
|
|
begin
|
|
Bitmap.Free;
|
|
inherited;
|
|
end;
|
|
|
|
|
|
{ Modified TStringGrid }
|
|
|
|
constructor TStringGridEx.Create(AOwner: TComponent);
|
|
begin
|
|
inherited;
|
|
FImageList := TFPList.Create;
|
|
end;
|
|
|
|
destructor TStringGridEx.Destroy;
|
|
var
|
|
j: Integer;
|
|
begin
|
|
for j:=0 to FImageList.Count-1 do
|
|
TImageData(FImageList[j]).Free;
|
|
FImageList.Free;
|
|
inherited;
|
|
end;
|
|
|
|
{ Adds a new image to the image list. ACol and ARow indicate the column and
|
|
row index of the anchor cell to which the image is attached. If the left or
|
|
top edge of the anchor cell is moved then the image follows. The image can
|
|
be shifted by dx and dy pixels from the top left corner of the anchor cell.
|
|
AWidth defines the width of the image in pixels; the image is automatically
|
|
rescaled. If AWidth is missing (or 0) the original size of the image is used.}
|
|
procedure TStringGridEx.AddImage(APicture: TPicture; ACol, ARow: Integer;
|
|
dx, dy: Integer; AWidth: Integer = 0);
|
|
var
|
|
bmp: TBitmap;
|
|
imgdata: TImageData;
|
|
begin
|
|
bmp := TBitmap.Create;
|
|
if AWidth <= 0 then begin
|
|
// Keep original image size
|
|
bmp.Width := APicture.Width;
|
|
bmp.Height := APicture.Height;
|
|
end else begin
|
|
// Scale image
|
|
bmp.Width := AWidth;
|
|
bmp.Height := round(bmp.Width / APicture.Width * APicture.Height);
|
|
end;
|
|
bmp.PixelFormat := pf32Bit;
|
|
bmp.Canvas.Brush.Color := clWhite;
|
|
bmp.Canvas.FillRect(0, 0, bmp.Width, bmp.Height);
|
|
bmp.Canvas.StretchDraw(Rect(0, 0, bmp.Width, bmp.Height), APicture.Graphic);
|
|
|
|
imgData := TImageData.Create;
|
|
imgData.Bitmap := bmp;
|
|
imgData.Col := ACol;
|
|
imgData.Row := ARow;
|
|
imgData.dx := dx;
|
|
imgData.dy := dy;
|
|
FImageList.Add(imgData);
|
|
|
|
Invalidate;
|
|
end;
|
|
|
|
procedure TStringGridEx.DoImageClick(Shift: TShiftState; X, Y: Integer);
|
|
begin
|
|
if Assigned(FOnImageClick) then
|
|
FOnImageClick(self, Shift, FMouseImgIndex, X, Y);
|
|
end;
|
|
|
|
procedure TStringGridEx.DoImageMove(Shift: TShiftState; dx, dy: Integer);
|
|
begin
|
|
if Assigned(FOnImageMove) then
|
|
FOnImageMove(self, Shift, FMouseImgIndex, dx, dy);
|
|
end;
|
|
|
|
procedure TStringGridEx.DrawAllRows;
|
|
var
|
|
imgdata: TImagedata;
|
|
i: Integer;
|
|
clipArea: TRect;
|
|
ImgRect: TRect;
|
|
R: TRect;
|
|
tmp: Integer;
|
|
begin
|
|
inherited;
|
|
|
|
// Calculate the clip area, i.e. the rectangle enclosing the non-fixed cells...
|
|
clipArea := Canvas.ClipRect;
|
|
ColRowToOffset(true, false, FixedCols, clipArea.Left, tmp);
|
|
ColRowToOffset(false, false, FixedRows, clipArea.Top, tmp);
|
|
|
|
Canvas.SaveHandleState;
|
|
try
|
|
// ... and use it for clipping
|
|
IntersectClipRect(Canvas.Handle, clipArea.Left, clipArea.Top, clipArea.Right, clipArea.Bottom);
|
|
|
|
for i := 0 to FImageList.Count-1 do begin
|
|
// Get bounding rectangle of the image
|
|
imgRect := GetImageRect(i);
|
|
|
|
// Calculate the intersection of the image rectangle with the clip rectangle
|
|
// Nothing to do if the image rectangle does not intersect the clip rectangle
|
|
if not IntersectRect(R, clipArea, imgRect) then
|
|
continue;
|
|
|
|
imgdata := TImageData(FImageList[i]);
|
|
Canvas.Draw(imgRect.Left, imgRect.Top, imgData.Bitmap);
|
|
end;
|
|
|
|
finally
|
|
Canvas.RestoreHandlestate;
|
|
end;
|
|
end;
|
|
|
|
{ Extracts, in pixels, the bounding rectangle of the image with the
|
|
specified index }
|
|
function TStringGridEx.GetImageRect(AImageIndex: Integer): TRect;
|
|
var
|
|
imgdata: TImageData;
|
|
begin
|
|
if (AImageIndex >= 0) and (AImageIndex < FImageList.Count) then
|
|
begin
|
|
// Extract image data from image list
|
|
imgdata := TImageData(FImageList[AImageIndex]);
|
|
// Find coordinates of cell anchor
|
|
Result := CellRect(imgdata.Col, imgdata.Row);
|
|
// Fix size of the image rectangle
|
|
Result.Right := Result.Left + imgData.Bitmap.Width;
|
|
Result.Bottom := Result.Top + imgData.Bitmap.Height;
|
|
// Shift image to final position
|
|
OffsetRect(Result, imgdata.dx, imgdata.dy);
|
|
end else
|
|
Result := Rect(0, 0, 0, 0);
|
|
end;
|
|
|
|
procedure TStringGridEx.MouseDown(Button: TMouseButton; Shift:TShiftState;
|
|
X,Y:Integer);
|
|
var
|
|
idx: Integer;
|
|
begin
|
|
idx := PointInImage(Point(X, Y));
|
|
if idx > -1 then begin
|
|
FMouseDownPt := Point(X, Y);
|
|
FMouseImgIndex := idx;
|
|
DoImageClick(Shift, X, Y);
|
|
Abort;
|
|
end else
|
|
inherited;
|
|
end;
|
|
|
|
procedure TStringGridEx.MouseMove(Shift: TShiftState; X,Y: Integer);
|
|
begin
|
|
if (FMouseImgIndex > -1) then begin
|
|
DoImageMove(Shift, X - FMouseDownPt.X, Y - FMouseDownPt.Y);
|
|
FMouseDownPt := Point(X, Y);
|
|
end
|
|
else
|
|
inherited;
|
|
end;
|
|
|
|
procedure TStringGridEx.MouseUp(Button: TMouseButton; Shift:TShiftState;
|
|
X,Y:Integer);
|
|
begin
|
|
FMouseImgIndex := -1;
|
|
inherited;
|
|
end;
|
|
|
|
{ Moves the image by dx pixels horizontally and dy pixels vertically.
|
|
It is recommended to call UpdateImageAnchor afterwards. If this is not done
|
|
then it is not clear whether an image will move if column widths are changed. }
|
|
procedure TStringGridEx.MoveImageBy(AImageIndex, dx, dy: Integer);
|
|
var
|
|
imgData: TImageData;
|
|
begin
|
|
if (AImageIndex > -1) and (AImageIndex < FImageList.Count) then
|
|
begin
|
|
imgData := TImageData(FImageList[AImageIndex]);
|
|
inc(imgData.dx, dx);
|
|
inc(imgData.dy, dy);
|
|
Invalidate;
|
|
end;
|
|
end;
|
|
|
|
{ Finds the index of the image which contains the specified point. Images
|
|
are checked in reverse order, this means that in case of overlapping images
|
|
the "top" one is selected.
|
|
NOTE: Transparent areas of images are ignored, it is always the enclosing
|
|
rectangle of the entire image which is checked. }
|
|
function TStringGridEx.PointInImage(APoint: TPoint): Integer;
|
|
var
|
|
imgRect: TRect;
|
|
topleftPx: TPoint;
|
|
i: Integer;
|
|
begin
|
|
topleftPx := GetPxTopLeft;
|
|
APoint.X := APoint.X + topleftPx.X;
|
|
APoint.Y := APoint.Y + topleftPx.Y;
|
|
for i:=FImageList.Count-1 downto 0 do begin
|
|
imgRect := GetImageRect(i);
|
|
if PtInRect(imgRect, APoint) then
|
|
exit(i);
|
|
end;
|
|
Result := -1;
|
|
end;
|
|
|
|
{ Recalculates the anchor cell to which the image is attached. The anchor cell
|
|
is always the cell which contains the upper left corner of the image. It
|
|
may change when an image is dragged across the grid.
|
|
This method should be called after moving an image because otherwise it will
|
|
not be clear whether an image moves if column widths are changed. }
|
|
procedure TStringGridEx.UpdateImageAnchor(AImageIndex: Integer);
|
|
var
|
|
imgdata: TImageData;
|
|
cell: TPoint;
|
|
R: TRect;
|
|
P: TPoint;
|
|
begin
|
|
if (AImageIndex < 0) or (AImageIndex >= FImageList.Count) then
|
|
exit;
|
|
R := GetImageRect(AImageIndex); // Current pixel coordinates of image
|
|
cell := MouseToLogCell(R.TopLeft); // Col/row of cell with top left image corner
|
|
P := CellRect(cell.X, cell.Y).TopLeft; // Pixel coordinates of top left corner of anchor cell
|
|
imgdata := TImageData(FImageList[AImageIndex]);
|
|
imgData.Col := cell.X;
|
|
imgData.Row := cell.Y;
|
|
imgData.dx := R.Left - P.X;
|
|
imgData.dy := R.Top - P.Y;
|
|
end;
|
|
|
|
procedure TStringGridEx.WMHScroll(var message: TLMHScroll);
|
|
begin
|
|
inherited;
|
|
Invalidate;
|
|
end;
|
|
|
|
procedure TStringGridEx.WMVScroll(var message: TLMVScroll);
|
|
begin
|
|
inherited;
|
|
Invalidate;
|
|
end;
|
|
|
|
end.
|
|
|