In a previous article, I talked about OpenSource GDI+ Library for Delphi. In this article I will present a small application which is the basic of an image processing or image drawing application.
A form to display an image
The application is divided into two forms. One main form and one image display form. The main form creates two instances of the image display form to show two images side by side. The image display forms are created as parented, that is they appears as a child window of the main form.
The most interesting part of the code, involving GDI+ Library is into the image display form. Beside displaying an image, the display form expose a small API to manipulate the image. The main form is very simple and provides a user interface for the display form API.
In this demo, the API is quite simple. It provides zoom and pan and a trivial paint of something above the image. Nevertheless, the code is really serious and you can easily start your own image processing or drawing application.
The display form actually display a bitmap loaded from a file using GDI+ decoders. You can load JPG, GIF, TIF and other format. You could as well create the bitmap from an image capture device such a camera or a scanner. This bitmap is named "FullBitmap" in the code.
The bitmap is drawn into a second bitmap which will be used for display. On this second bitmap the application could paint or draw anything. In this demo, it paints only a simple text but in a real application, you could - for example - have a data structure representing geometrical items and draw those items. You'll get a drawing program. This second bitmap is named "ViewBitmap" in the code.
To create zoom and pan, I used GDI+ built in coordinate transformations and a bunch of variables describing the zoom and pan.
GDI+ also provide a clipping function that I used to make sure the displayed image, zoomed and panned is not drawn outside of the viewing area.
Finally, the display form also display a border around the image. It is used when multiple images are displayed on the same window. The "active" image has his border drawn in a different color.
Below you'll find full source code for your reference. It is also available for download as a full project from my website at:
http://www.overbyte.be/frame_index.html?redirTo=/blog_source_code.html
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 | unit ImageDisplay; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, ExtCtrls, Forms, Dialogs, GdiPlus; const WM_APP_PAINT = WM_USER + 1 ; DEMO_FILE = '..\..\ics_logo.gif' ; type TImageForm = class (TForm) private FFrameWidth : Integer ; FFrameHeight : Integer ; FPaintTop : Integer ; FPaintLeft : Integer ; FPaintMargin : Integer ; FPaintHeight : Integer ; FPaintWidth : Integer ; FYTop : Integer ; FXLeft : Integer ; FZoomFactor : Double ; // 1.0 = no zoom FFullBitMap : IGPBitmap; FViewBitmap : IGPBitmap; FMarginColor : TColor; FAppPaintFlag : Boolean ; function CreateGraphicInterface: IGPGraphics; procedure PaintSomething(Graphics: IGPGraphics); protected procedure Paint; override; procedure Resize; override; procedure InitDrawingArea(ALeft, ATop, AWidth, AHeight, AMargin: Integer ); procedure TriggerAppPaint; procedure WMAppPaint( var Msg: TMessage); message WM_APP_PAINT; procedure SetMarginColor( const Value: TColor); function ZoomFitCompute: Double ; public constructor Create(AOwner : TComponent); override; procedure ZoomIn(Speed: Double ); procedure ZoomOut(Speed: Double ); procedure PanRight; procedure PanDown; procedure PanLeft; procedure PanUp; procedure PanCenter; function LoadFromFile( const AFileName: String ): Boolean ; property MarginColor : TColor read FMarginColor write SetMarginColor; end ; var ImageForm: TImageForm; implementation {$R *.dfm} {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} constructor TImageForm . Create(AOwner: TComponent); begin inherited Create(AOwner); FZoomFactor := 1.0 ; InitDrawingArea( 0 , 0 , Width, Height, 0 ); if FileExists(DEMO_FILE) then LoadFromFile(DEMO_FILE); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} function TImageForm . LoadFromFile( const AFileName : String ) : Boolean ; begin FFullBitMap := TGPBitmap . Create(AFileName); FFrameWidth := FFullBitMap . Width; FFrameHeight := FFullBitMap . Height; FViewBitmap := TGPBitmap . Create(FFrameWidth, FFrameHeight, PixelFormat24bppRGB); TriggerAppPaint; Result := TRUE ; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} function TImageForm . CreateGraphicInterface : IGPGraphics; begin Result := TGPGraphics . Create(Canvas . Handle); Result . ResetTransform; Result . TranslateTransform(FPaintLeft + FXLeft, FPaintTop + FYTop, MatrixOrderPrepend); Result . ScaleTransform(FZoomFactor, FZoomFactor, MatrixOrderPrepend); Result . InterpolationMode := InterpolationModeHighQualityBilinear; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . Paint; var Graphics : IGPGraphics; ViewGraphics : IGPGraphics; Points : array [ 0..4 ] of TGPPoint; WorldPoints : array [ 0..1 ] of TGPPoint; WorldDrawingArea : TGPRect; WorldBitmapArea : TGPRect; begin FAppPaintFlag := FALSE ; Graphics := CreateGraphicInterface; if Assigned(FFullBitMap) then begin Points[ 0 ].X := 0 ; Points[ 0 ].Y := 0 ; Points[ 1 ].X := FFullBitMap . Width; Points[ 1 ].Y := FFullBitMap . Height; Points[ 2 ].X := FPaintWidth; Points[ 2 ].Y := FPaintHeight; Points[ 3 ].X := FXLeft; Points[ 3 ].Y := FYTop; Points[ 4 ].X := FPaintLeft; Points[ 4 ].Y := FPaintTop; Graphics . TransformPoints(CoordinateSpaceWorld, // Destination CoordinateSpaceDevice, // Source Points); // World coordinate space are simply bitmap coordinate space WorldBitmapArea . X := 0 ; WorldBitmapArea . Y := 0 ; WorldBitmapArea . Width := FFullBitMap . Width; WorldBitmapArea . Height := FFullBitMap . Height; WorldDrawingArea . X := Points[ 0 ].X - Points[ 3 ].X; WorldDrawingArea . Y := Points[ 0 ].Y - Points[ 3 ].Y; WorldDrawingArea . Width := (Points[ 2 ].X - Points[ 3 ].X) - WorldDrawingArea . X; WorldDrawingArea . Height := (Points[ 2 ].Y - Points[ 3 ].Y) - WorldDrawingArea . Y; Graphics . SetClip(WorldDrawingArea); ViewGraphics := TGPGraphics . FromImage(FViewBitMap); ViewGraphics . DrawImage(FFullBitMap, 0 , 0 , FFrameWidth, FFrameHeight); PaintSomething(ViewGraphics); Graphics . DrawImage(FViewBitMap, 0 , 0 , FFrameWidth, FFrameHeight); // Draw the rectangle surrounding the image. WorldPoints[ 0 ].X := 0 ; WorldPoints[ 0 ].Y := 0 ; WorldPoints[ 1 ].X := FFullBitMap . Width; WorldPoints[ 1 ].Y := FFullBitMap . Height; Graphics . TransformPoints(CoordinateSpaceDevice, // Destination CoordinateSpaceWorld, // Source WorldPoints); end else begin // FFullBitmap not assigned WorldPoints[ 0 ].X := 0 ; WorldPoints[ 0 ].Y := 0 ; WorldPoints[ 1 ].X := 0 ; WorldPoints[ 1 ].Y := 0 ; end ; Canvas . Pen . Style := psClear; Canvas . Brush . Style := bsSolid; Canvas . Brush . Color := Color; // Left Canvas . Rectangle( 0 , 0 , WorldPoints[ 0 ].X + 1 , FPaintHeight + 1 ); // Right Canvas . Rectangle(WorldPoints[ 1 ].X, 0 , FPaintWidth + 1 , FPaintHeight + 1 ); // Top Canvas . Rectangle(WorldPoints[ 0 ].X, 0 , WorldPoints[ 1 ].X + 1 , WorldPoints[ 0 ].Y + 1 ); // Bottom Canvas . Rectangle(WorldPoints[ 0 ].X, WorldPoints[ 1 ].Y, WorldPoints[ 1 ].X + 1 , FPaintHeight + 1 ); // Paint margin area (used to show selected image) Canvas . Pen . Style := psSolid; Canvas . Pen . Color := FMarginColor; Canvas . Pen . Width := FPaintMargin; Canvas . MoveTo(FPaintMargin div 2 , FPaintMargin div 2 ); Canvas . LineTo(FPaintWidth + 1 , FPaintMargin div 2 ); Canvas . LineTo(FPaintWidth + 1 , FPaintHeight + 1 ); Canvas . LineTo(FPaintMargin div 2 , FPaintHeight + 1 ); Canvas . LineTo(FPaintMargin div 2 , 0 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . InitDrawingArea( ALeft, ATop, AWidth, AHeight, AMargin : Integer ); begin FPaintMargin := AMargin; FPaintTop := ATop + AMargin; FPaintLeft := ALeft + AMargin; FPaintWidth := AWidth - ALeft - AMargin; FPaintHeight := AHeight - ATop - AMargin; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . Resize; var NewXLeft, NewYTop : Integer ; begin InitDrawingArea( 0 , 0 , ClientWidth, ClientHeight, 2 ); NewXLeft := (FPaintWidth - Round(FFrameWidth * FZoomFactor)) div 2 ; NewYTop := (FPaintHeight - Round(FFrameHeight * FZoomFactor)) div 2 ; if NewXLeft > 0 then FXLeft := NewXLeft; if NewYTop > 0 then FYTop := NewYTop; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . WMAppPaint( var Msg: TMessage); begin Paint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . TriggerAppPaint; begin // To avoid too much repainting, we use a flag and a custom message. // The custom message will trigger the painting. // Once the custom message has been posted, the falg is set and no more // message will be posted until the flag is reset by the paint routine. if not FAppPaintFlag then begin FAppPaintFlag := TRUE ; PostMessage(Handle, WM_APP_PAINT, 0 , 0 ); end ; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . SetMarginColor( const Value: TColor); begin FMarginColor := Value; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . ZoomOut(Speed : Double ); begin if Abs (Speed) < 0.001 then FZoomFactor := ZoomFitCompute else if Speed < 0 then FZoomFactor := -Speed else FZoomFactor := FZoomFactor / 1.05 ; if FZoomFactor < 0.01 then FZoomFactor := 0.01 ; if Abs (FZoomFactor - 1.0 ) < 0.001 then FZoomFactor := 1.0 ; // Avoid cumulating error //TriggerZoomChange(FZoomFactor); TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . ZoomIn(Speed : Double ); begin if Abs (Speed) < 0.001 then FZoomFactor := ZoomFitCompute else if Speed < 0 then FZoomFactor := -Speed else FZoomFactor := FZoomFactor * Speed; if Abs (FZoomFactor - 1.0 ) < 0.001 then FZoomFactor := 1.0 ; // Avoid cumulating error //TriggerZoomChange(FZoomFactor); TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} function TImageForm . ZoomFitCompute : Double ; var Z1, Z2 : Double ; begin if (FFrameWidth = 0 ) or (FFrameHeight = 0 ) then begin Result := 1.0 ; FXLeft := 0 ; FYTop := 0 ; Exit; end ; Z1 := FPaintWidth / FFrameWidth; Z2 := FPaintHeight / FFrameHeight; if Z1 < Z2 then Result := Z1 * 0.95 else Result := Z2 * 0.95 ; FXLeft := (FPaintWidth - Round(FFrameWidth * Result)) div 2 ; FYTop := (FPaintHeight - Round(FFrameHeight * Result)) div 2 ; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PanRight; begin FXLeft := FXLeft + 10 ; FYTop := FYTop + 0 ; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PanLeft; begin FXLeft := FXLeft - 10 ; FYTop := FYTop + 0 ; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PanUp; begin FXLeft := FXLeft + 0 ; FYTop := FYTop - 10 ; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PanDown; begin FXLeft := FXLeft + 0 ; FYTop := FYTop + 10 ; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PanCenter; begin FXLeft := (FPaintWidth - Round(FFrameWidth * FZoomFactor)) div 2 ; FYTop := (FPaintHeight - Round(FFrameHeight * FZoomFactor)) div 2 ; TriggerAppPaint; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TImageForm . PaintSomething(Graphics: IGPGraphics); var FontFamily : IGPFontFamily; Font : IGPFont; Point : TGPPointF; SolidBrush : IGPBrush; begin FontFamily := TGPFontFamily . Create( 'Times New Roman' ); Font := TGPFont . Create(FontFamily, 24 , FontStyleRegular, UnitPixel); SolidBrush := TGPSolidBrush . Create(TGPColor . Create( 255 , 255 , 0 , 0 )); Point . Initialize( 10 , 10 ); Graphics . TextRenderingHint := TextRenderingHintAntiAlias; Graphics . DrawString( 'Delphi rocks!' , Font, Point, SolidBrush); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} end . |
Using TImageForm
The form we saw above is used twice in the sample application to display two images side by side. The form has 3 panels: a top panel acting as a tool bar and two panels below for the two images.
The tool bar has been made very simple: only basic buttons to call the image display form API on behalf of the active image. It's up to you to use a nice user interface, you've got the idea.
The two image panels are use to host a display form. Each one showing his independent image.
Finally, an OpenDialog is used to load an image from a file. You can easily add the code to save an image as well since GDI+ does all the work for you.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 | unit ImageMain; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, ImageDisplay, Vcl . ExtCtrls, Vcl . StdCtrls; type TMainForm = class (TForm) TopPanel: TPanel; LeftPanel: TPanel; Splitter1: TSplitter; RightPanel: TPanel; ZoomFitButton: TButton; ZoomInButton: TButton; ZoomOutButton: TButton; PanLeftButton: TButton; PanRightButton: TButton; PanUpButton: TButton; PanDownButton: TButton; PanCenterButton: TButton; Zoom100Button: TButton; OpenButton: TButton; OpenDialog1: TOpenDialog; procedure LeftPanelResize(Sender: TObject); procedure RightPanelResize(Sender: TObject); procedure ZoomFitButtonClick(Sender: TObject); procedure ZoomInButtonClick(Sender: TObject); procedure ZoomOutButtonClick(Sender: TObject); procedure PanLeftButtonClick(Sender: TObject); procedure PanRightButtonClick(Sender: TObject); procedure PanUpButtonClick(Sender: TObject); procedure PanDownButtonClick(Sender: TObject); procedure PanCenterButtonClick(Sender: TObject); procedure Zoom100ButtonClick(Sender: TObject); procedure OpenButtonClick(Sender: TObject); private FLeftImage : TImageForm; FRightImage : TImageForm; FActiveImage : TImageForm; procedure SetActiveImage(Image : TImageForm); procedure ImageClick(Sender: TObject); public constructor Create(AOwner : TComponent); override; destructor Destroy; override; end ; var MainForm: TMainForm; implementation {$R *.dfm} {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} { TMainForm } {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} constructor TMainForm . Create(AOwner: TComponent); begin inherited Create(Aowner); FLeftImage := TImageForm . CreateParented(LeftPanel . Handle); FLeftImage . BorderStyle := bsNone; FLeftImage . OnClick := ImageClick; FLeftImage . Visible := TRUE ; FRightImage := TImageForm . CreateParented(RightPanel . Handle); FRightImage . BorderStyle := bsNone; FRightImage . OnClick := ImageClick; FRightImage . Visible := TRUE ; // Unselect active image and select left image as active // It will set the image borders correctly SetActiveImage( nil ); SetActiveImage(FLeftImage); // Call resize handler for both panels to set images display size LeftPanelResize(LeftPanel); RightPanelResize(LeftPanel); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} destructor TMainForm . Destroy; begin FreeAndNil(FLeftImage); FreeAndNil(FRightImage); inherited Destroy; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . LeftPanelResize(Sender: TObject); begin if Assigned(FLeftImage) then FLeftImage . BoundsRect := Rect( 0 , 0 , LeftPanel . Width - 1 , LeftPanel . Height - 1 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . OpenButtonClick(Sender: TObject); begin if not Assigned(FActiveImage) then SetActiveImage(FLeftImage); OpenDialog1 . Filter := 'JPEG images (*.jpg)|*.jpg|' + 'TIFF images (*.tif)|*.tif|' + 'BMP images (*.bmp)|*.bmp|' + 'GIF images (*.gif)|*.gif|' + 'PNG images (*.png)|*.png|' + 'All files (*.*)|*.*|' + '' ; // OpenDialog1.InitialDir := FInitialDir; OpenDialog1 . Options := OpenDialog1 . Options + [ofPathMustExist, ofFileMustExist]; if not OpenDialog1 . Execute(Handle) then Exit; FActiveImage . LoadFromFile(OpenDialog1 . FileName); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . RightPanelResize(Sender: TObject); begin if Assigned(FRightImage) then FRightImage . BoundsRect := Rect( 0 , 0 , RightPanel . Width - 1 , RightPanel . Height - 1 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . SetActiveImage(Image: TImageForm); begin if not Assigned(Image) then begin FLeftImage . MarginColor := Color; FRightImage . MarginColor := Color; end else begin if Assigned(FActiveImage) then FActiveImage . MarginColor := Color; FActiveImage := Image; FActiveImage . MarginColor := clBlack; end ; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . ImageClick(Sender: TObject); begin SetActiveImage(Sender as TImageForm); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . Zoom100ButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . ZoomIn(- 1.0 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . ZoomFitButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . ZoomIn( 0 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . ZoomInButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . ZoomIn( 1.05 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . ZoomOutButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . ZoomOut( 1.05 ); end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . PanCenterButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . PanCenter; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . PanDownButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . PanDown; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . PanLeftButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . PanLeft; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . PanRightButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . PanRight; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} procedure TMainForm . PanUpButtonClick(Sender: TObject); begin if Assigned(FActiveImage) then FActiveImage . PanUp; end ; {* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *} end . |
Read previous article at:
http://francois-piette.blogspot.be/2013/05/opensource-gdi-library.html
This article is available from:
http://francois-piette.blogspot.be/2013/05/opensource-gdi-library-part-2.html
Download source code at:
http://www.overbyte.be/frame_index.html?redirTo=/blog_source_code.html
Follow me on Twitter
Follow me on LinkedIn
Follow me on Google+
Visit my website: http://www.overbyte.be
No comments:
Post a Comment