mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-06 00:37:55 +02:00
3196 lines
87 KiB
ObjectPascal
3196 lines
87 KiB
ObjectPascal
{
|
|
*****************************************************************************
|
|
* gtk3objects.pas *
|
|
* ----------------- *
|
|
* *
|
|
* *
|
|
*****************************************************************************
|
|
|
|
*****************************************************************************
|
|
This file is part of the Lazarus Component Library (LCL)
|
|
|
|
See the file COPYING.modifiedLGPL.txt, included in this distribution,
|
|
for details about the license.
|
|
*****************************************************************************
|
|
}
|
|
unit Gtk3Objects;
|
|
|
|
{$mode objfpc}{$H+}
|
|
{$i gtk3defines.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, SysUtils, Types, math, FPCanvas,
|
|
// LazUtils
|
|
LazUTF8, IntegerList, LazStringUtils,
|
|
// LCL
|
|
LCLType, LCLProc, Graphics,
|
|
LazGtk3, LazGdk3, LazGObject2, LazGLib2, LazGdkPixbuf2,
|
|
LazPango1, LazPangoCairo1, LazCairo1, gtk3procs, LazLogger;
|
|
|
|
type
|
|
TGtk3DeviceContext = class;
|
|
|
|
{ TGtk3Object }
|
|
|
|
TGtk3Object = class(TObject)
|
|
private
|
|
FUpdateCount: Integer;
|
|
public
|
|
constructor Create; virtual; overload;
|
|
procedure Release; virtual;
|
|
procedure BeginUpdate; virtual;
|
|
procedure EndUpdate; virtual;
|
|
function InUpdate: Boolean;
|
|
end;
|
|
|
|
{ TGtk3ContextObject }
|
|
|
|
TGtk3ContextObject = class(TGtk3Object)
|
|
private
|
|
FShared: Boolean;
|
|
fContext:TGtk3DeviceContext;
|
|
public
|
|
constructor Create; override;
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; virtual;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; virtual;abstract;
|
|
property Shared: Boolean read FShared write FShared;
|
|
end;
|
|
|
|
{ TGtk3Font }
|
|
|
|
TGtk3Font = class(TGtk3ContextObject)
|
|
private
|
|
FLayout: PPangoLayout;
|
|
FLogFont: TLogFont;
|
|
FFontName: String;
|
|
FHandle: PPangoFontDescription;
|
|
public
|
|
constructor Create(ACairo: Pcairo_t; AWidget: PGtkWidget = nil);
|
|
constructor Create(ALogFont: TLogFont; const ALongFontName: String);
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; override;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; override;
|
|
destructor Destroy; override;
|
|
procedure UpdateLogFont;
|
|
property FontName: String read FFontName write FFontName;
|
|
property Handle: PPangoFontDescription read FHandle;
|
|
property Layout: PPangoLayout read FLayout;
|
|
property LogFont: TLogFont read FLogFont;
|
|
end;
|
|
|
|
{ TGtk3Brush }
|
|
|
|
TGtk3Brush = class(TGtk3ContextObject)
|
|
private
|
|
FColor: TColor;
|
|
FStyle: LongWord;
|
|
procedure SetColor(AValue: TColor);
|
|
procedure SetStyle(AStyle:longword);
|
|
public
|
|
brush_pattern:pcairo_pattern_t;
|
|
pat_buf:pdword;
|
|
LogBrush: TLogBrush;
|
|
constructor Create; override;
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; override;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; override;
|
|
destructor Destroy; override;
|
|
procedure UpdatePattern(const aColor: TColorRef);
|
|
property Color: TColor read FColor write SetColor;
|
|
property Context: TGtk3DeviceContext read FContext write FContext;
|
|
property Style: LongWord read FStyle write SetStyle;
|
|
end;
|
|
|
|
{ TGtk3Pen }
|
|
|
|
TGtk3Pen = class(TGtk3ContextObject)
|
|
private
|
|
FCosmetic: Boolean;
|
|
FEndCap: TPenEndCap;
|
|
FJoinStyle: TPenJoinStyle;
|
|
FPenMode: TPenMode;
|
|
FStyle: TFPPenStyle;
|
|
FWidth: Integer;
|
|
FColor: TColor;
|
|
FIsExtPen: Boolean;
|
|
procedure SetColor(AValue: TColor);
|
|
procedure setCosmetic(b: Boolean);
|
|
procedure setWidth(p1: Integer);
|
|
public
|
|
LogPen: TLogPen;
|
|
constructor Create; override;
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; override;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; override;
|
|
property Color: TColor read FColor write SetColor;
|
|
property Context: TGtk3DeviceContext read FContext write FContext;
|
|
|
|
property Cosmetic: Boolean read FCosmetic write SetCosmetic;
|
|
property EndCap: TPenEndCap read FEndCap write FEndCap;
|
|
property IsExtPen: Boolean read FIsExtPen write FIsExtPen;
|
|
property JoinStyle: TPenJoinStyle read FJoinStyle write FJoinStyle;
|
|
property Mode: TPenMode read FPenMode write FPenMode;
|
|
property Style: TFPPenStyle read FStyle write FStyle;
|
|
property Width: Integer read FWidth write SetWidth;
|
|
end;
|
|
|
|
{ TGtk3Region }
|
|
|
|
TGtk3Region = class(TGtk3ContextObject)
|
|
private
|
|
FHandle: Pcairo_region_t;
|
|
public
|
|
property Handle: Pcairo_region_t read FHandle write FHandle;
|
|
constructor Create({%H-}CreateHandle: Boolean); virtual; overload;
|
|
constructor Create({%H-}CreateHandle: Boolean; X1,Y1,X2,Y2: Integer); virtual; overload;
|
|
constructor Create(X1,Y1,X2,Y2,nW,nH: Integer); virtual; overload;
|
|
constructor CreateEllipse(X1,Y1,X2,Y2: Integer); virtual; overload;
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; override;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; override;
|
|
destructor Destroy; override;
|
|
function GetExtents: TRect;
|
|
function ContainsRect(ARect: TRect): Boolean;
|
|
function ContainsPoint(APoint: TPoint): Boolean;
|
|
end;
|
|
|
|
{ TGtk3Image }
|
|
|
|
TGtk3Image = class(TGtk3ContextObject)
|
|
private
|
|
FData: PByte;
|
|
FDataOwner: Boolean;
|
|
FHandle: PGdkPixbuf;
|
|
FFormat : Tcairo_format_t;
|
|
public
|
|
constructor Create; override;
|
|
constructor Create(vHandle: PGdkPixbuf); overload;
|
|
constructor Create(AData: PByte; width: Integer; height: Integer; format: Tcairo_format_t; const ADataOwner: Boolean = False); overload;
|
|
constructor Create(AData: PByte; width: Integer; height: Integer; bytesPerLine: Integer; format: Tcairo_format_t; const ADataOwner: Boolean = False); overload;
|
|
function Select(ACtx:TGtk3DeviceContext):TGtk3ContextObject; override;
|
|
function Get(szbuf:integer;pbuf:pointer):integer; override;
|
|
destructor Destroy; override;
|
|
procedure CopyFrom(AImage: PGdkPixbuf; x, y, w, h: integer);
|
|
function height: Integer;
|
|
function width: Integer;
|
|
function depth: Integer;
|
|
function dotsPerMeterX: Integer;
|
|
function dotsPerMeterY: Integer;
|
|
function bits: PByte;
|
|
function numBytes: LongWord;
|
|
function bytesPerLine: Integer;
|
|
property Format: Tcairo_format_t read FFormat;
|
|
property Handle: PGdkPixbuf read FHandle;
|
|
end;
|
|
|
|
{ TGtk3Cursor }
|
|
|
|
TGtk3Cursor = class(TGtk3Object)
|
|
private
|
|
fHandle:PGdkCursor;
|
|
public
|
|
constructor Create(ACur:integer); overload;
|
|
constructor Create(pixbuf:PGdkPixbuf;x,y:gint); overload;
|
|
constructor Create(img:TGtk3Image); overload;
|
|
constructor Create(ACur: TGdkCursorType); overload;
|
|
destructor Destroy; override;
|
|
property Handle:PGdkCursor read fHandle;
|
|
end;
|
|
|
|
{ TGtk3DeviceContext }
|
|
|
|
TGtk3DeviceContext = class (TGtk3Object)
|
|
private
|
|
FDCSaveCounter: integer;
|
|
FBkColor:TColorRef;
|
|
FBgBrush: TGtk3Brush; //solid brush created when BkColor is setted up.
|
|
FBrush: TGtk3Brush;
|
|
FFont: TGtk3Font;
|
|
FvImage: TGtk3Image;
|
|
FCanRelease: Boolean;
|
|
FCurrentBrush: TGtk3Brush;
|
|
FCurrentFont: TGtk3Font;
|
|
FCurrentImage: TGtk3Image;
|
|
FCurrentTextColor: TColorRef;
|
|
FCurrentRegion: TGtk3Region;
|
|
FOwnsCairo: Boolean;
|
|
FOwnsSurface: Boolean;
|
|
FPen: TGtk3Pen;
|
|
FvClipRect: TRect;
|
|
FCurrentPen: TGtk3Pen;
|
|
FBkMode: integer;
|
|
FCanvasScaleFactor: double;
|
|
FXorMode: boolean;
|
|
function GetBkColor:TColorRef;
|
|
function GetOffset: TPoint;
|
|
function GetRasterOp: integer;
|
|
procedure setBrush(AValue: TGtk3Brush);
|
|
procedure setBkColor(AValue: TColorRef);
|
|
procedure SetFont(AValue: TGtk3Font);
|
|
procedure SetOffset(AValue: TPoint);
|
|
procedure setPen(AValue: TGtk3Pen);
|
|
procedure SetRasterOp(AValue: integer);
|
|
procedure SetvImage(AValue: TGtk3Image);
|
|
function SX(const x: double): Double;
|
|
function SY(const y: double): Double;
|
|
function SX2(const x: double): Double;
|
|
function SY2(const y: double): Double;
|
|
procedure ApplyBrush;
|
|
procedure ApplyFont;
|
|
procedure ApplyPen;
|
|
procedure FillAndStroke;
|
|
procedure SetCanvasScaleFactor(const AValue: double);
|
|
procedure SetXorMode(var xorSurface: Pcairo_surface_t; AMap: Tcairo_operator_t); // SetRasterOp
|
|
procedure ApplyXorDrawing(xorSurface: Pcairo_surface_t; AMap: Tcairo_operator_t); // after each paint
|
|
protected
|
|
FCairo: Pcairo_t;
|
|
FXorCairo: PCairo_t;
|
|
FXorSurface: Pcairo_surface_t;
|
|
FLastPenX, FLastPenY: double;
|
|
property XorMode: boolean read FXorMode write FXorMode;
|
|
public
|
|
CairoSurface: Pcairo_surface_t;
|
|
Parent: PGtkWidget;
|
|
ParentPixmap: PGdkPixbuf;
|
|
Window: PGdkWindow;
|
|
fncOrigin:TPoint; // non-client area offsets surface origin
|
|
constructor Create(AWidget: PGtkWidget; const APaintEvent: Boolean = False); virtual;
|
|
constructor Create(AWindow: PGdkWindow; const APaintEvent: Boolean); virtual;
|
|
constructor CreateFromCairo(AWidget: PGtkWidget; ACairo: PCairo_t); virtual;
|
|
destructor Destroy; override;
|
|
procedure CreateObjects;
|
|
procedure DeleteObjects;
|
|
public
|
|
procedure drawPixel(x, y: Integer; AColor: TColor);
|
|
function pcr: Pcairo_t;
|
|
function getPixel(x, y: Integer): TColor;
|
|
procedure drawRect(x1, y1, w, h: Integer; const AFill, ABorder: Boolean);
|
|
procedure drawRoundRect(x, y, w, h, rx, ry: Integer);
|
|
procedure drawText(x, y: Integer; AText: PChar; ALen: Integer; const ABgFilled: boolean);
|
|
procedure drawEllipse(x, y, w, h: Integer; AFill, ABorder: Boolean);
|
|
procedure drawSurface(targetRect: PRect; Surface: Pcairo_surface_t; sourceRect: PRect;
|
|
aPixBuf: PGdkPixBuf; mask: PGdkPixBuf; maskRect: PRect);
|
|
procedure drawImage(targetRect: PRect; image: PGdkPixBuf; sourceRect: PRect;
|
|
mask: PGdkPixBuf; maskRect: PRect);
|
|
procedure drawImage1(targetRect: PRect; image: PGdkPixBuf; sourceRect: PRect;
|
|
mask: PGdkPixBuf; maskRect: PRect);
|
|
procedure drawPixmap(p: PPoint; pm: PGdkPixbuf; sr: PRect);
|
|
procedure drawPolyLine(P: PPoint; NumPts: Integer);
|
|
procedure drawPolygon(P: PPoint; NumPts: Integer; FillRule: Integer; AFill,
|
|
ABorder: Boolean);
|
|
procedure drawPolyBezier(P: PPoint; NumPoints: Integer; Filled, Continuous: boolean);
|
|
procedure EllipseArcPath(CX, CY, RX, RY: Double; Angle1, Angle2: Double; Clockwise, Continuous: Boolean);
|
|
procedure eraseRect(ARect: PRect);
|
|
procedure fillRect(ARect: PRect; ABrush: HBRUSH); overload;
|
|
procedure fillRect(x, y, w, h: Integer; ABrush: HBRUSH); overload;
|
|
procedure fillRect(x, y, w, h: Integer); overload;
|
|
function RoundRect(X1, Y1, X2, Y2: Integer; RX, RY: Integer): Boolean;
|
|
function drawFrameControl(arect:TRect;uType,uState:cardinal):boolean;
|
|
function drawFocusRect(const aRect: TRect): boolean;
|
|
function getBpp: integer;
|
|
function getDepth: integer;
|
|
function getDeviceSize: TPoint;
|
|
function LineTo(X, Y: Integer): Boolean;
|
|
function MoveTo(const X, Y: Integer; OldPoint: PPoint): Boolean;
|
|
function SetClipRegion(ARgn: TGtk3Region): Integer;
|
|
procedure SetSourceColor(AColor: TColor);
|
|
procedure SetImage(AImage: TGtk3Image);
|
|
function ResetClip: Integer;
|
|
procedure TranslateCairoToDevice;
|
|
procedure Translate(APoint: TPoint);
|
|
procedure set_antialiasing(aamode:boolean);
|
|
procedure Save;
|
|
procedure Restore;
|
|
property BkMode: integer read FBkMode write FBkMode;
|
|
property BkColor: TColorRef read GetBkColor write SetBkColor;
|
|
property BgBrush: TGtk3Brush read FBgBrush; {bgBrush is created when SetBk is called, otherwise is nil}
|
|
property CanRelease: Boolean read FCanRelease write FCanRelease;
|
|
property CurrentBrush: TGtk3Brush read FCurrentBrush write FCurrentBrush;
|
|
property CurrentFont: TGtk3Font read FCurrentFont write FCurrentFont;
|
|
property CurrentImage: TGtk3Image read FCurrentImage write FCurrentImage;
|
|
property CurrentPen: TGtk3Pen read FCurrentPen write FCurrentPen;
|
|
property CurrentRegion: TGtk3Region read FCurrentRegion;
|
|
property CurrentTextColor: TColorRef read FCurrentTextColor write FCurrentTextColor;
|
|
property CanvasScaleFactor: double read FCanvasScaleFactor write SetCanvasScaleFactor;
|
|
property Offset: TPoint read GetOffset write SetOffset;
|
|
property OwnsSurface: Boolean read FOwnsSurface;
|
|
property RasterOp: integer read GetRasterOp write SetRasterOp; //automatically maps cairo_operator_t to winapi ROP.
|
|
property vBrush: TGtk3Brush read FBrush write setBrush;
|
|
property vClipRect: TRect read FvClipRect write FvClipRect;
|
|
property vFont: TGtk3Font read FFont write SetFont;
|
|
property vImage: TGtk3Image read FvImage write SetvImage;
|
|
property vPen: TGtk3Pen read FPen write setPen;
|
|
end;
|
|
|
|
function CheckBitmap(const ABitmap: HBITMAP; const AMethodName: String;
|
|
const AParamName: String = ''): Boolean;
|
|
procedure Gtk3WordWrap(DC: HDC; AText: PChar;
|
|
MaxWidthInPixel: integer; out Lines: PPChar; out LineCount: integer);
|
|
|
|
function Gtk3DefaultContext: TGtk3DeviceContext;
|
|
function Gtk3ScreenContext: TGtk3DeviceContext;
|
|
|
|
function ReplaceAmpersandsWithUnderscores(const S: string): string; inline;
|
|
function ReplaceUnderscoresWithAmpersands(const S: string): string; inline;
|
|
|
|
implementation
|
|
|
|
uses gtk3int,controls;
|
|
|
|
const
|
|
PixelOffset = 0.5; // Cairo API needs 0.5 pixel offset to not make blurry lines
|
|
|
|
const
|
|
Dash_Dash: array [0..1] of double = (3, 2); //____ ____
|
|
Dash_Dot: array [0..1] of double = (1, 2); //.........
|
|
Dash_DashDot: array [0..3] of double = (3, 2, 1, 2); //__ . __ .
|
|
Dash_DashDotDot: array [0..5] of double = (3, 2, 1, 2, 1, 2); //__ . . __
|
|
|
|
var
|
|
FDefaultContext: TGtk3DeviceContext = nil;
|
|
FScreenContext: TGtk3DeviceContext = nil;
|
|
|
|
function create_stipple(stipple_data:pbyte;width,height:integer):pcairo_pattern_t;forward;
|
|
|
|
const clr_A = $FF008000;// $3093BA52;
|
|
const clr_B = $FFFFFFFF;// $30FFFFFF;
|
|
|
|
const
|
|
(* the stipple patten should look like that
|
|
* 1 1 1 0 0 0 0 1
|
|
* 1 1 0 0 0 0 1 1
|
|
* 1 0 0 0 0 1 1 1
|
|
* 0 0 0 0 1 1 1 1
|
|
*
|
|
* 0 0 0 1 1 1 1 0
|
|
* 0 0 1 1 1 1 0 0
|
|
* 0 1 1 1 1 0 0 0
|
|
* 1 1 1 1 0 0 0 0
|
|
*)
|
|
|
|
stipple_bdiag: array[0..8 * 8-1] of dword = (
|
|
clr_A, clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A,
|
|
clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A,
|
|
clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A, clr_A,
|
|
clr_B, clr_B, clr_B, clr_B, clr_A, clr_A, clr_A, clr_A,
|
|
//-----------------------------------------------------
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B,
|
|
clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B,
|
|
clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_A, clr_A, clr_A, clr_A, clr_B, clr_B, clr_B, clr_B);
|
|
|
|
stipple_fdiag: array[0..8 * 8-1] of dword = (
|
|
clr_A, clr_A, clr_A, clr_A, clr_B, clr_B, clr_B, clr_B,
|
|
clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B,
|
|
//-----------------------------------------------------
|
|
clr_B, clr_B, clr_B, clr_B, clr_A, clr_A, clr_A, clr_A,
|
|
clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A, clr_A,
|
|
clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A,
|
|
clr_A, clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A);
|
|
|
|
|
|
|
|
//bsHorizontal
|
|
stipple_horz: array[0..15] of dword = (
|
|
clr_B, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_B,
|
|
clr_A, clr_A, clr_A, clr_A
|
|
);
|
|
|
|
stipple_vert: array[0..15] of dword = (
|
|
clr_A, clr_B, clr_B, clr_B,
|
|
clr_A, clr_B, clr_B, clr_B,
|
|
clr_A, clr_B, clr_B, clr_B,
|
|
clr_A, clr_B, clr_B, clr_B
|
|
);
|
|
|
|
(* , bsVertical, bsFDiagonal,
|
|
bsBDiagonal, bsCross, bsDiagCross, bsImage, bsPattern);*)
|
|
|
|
|
|
stipple_cross0: array[0..8] of dword = (
|
|
clr_B, clr_A, clr_B,
|
|
clr_A, clr_A, clr_A,
|
|
clr_B, clr_A, clr_B
|
|
);
|
|
|
|
stipple_cross1: array[0..63] of dword = (
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_A, clr_A, clr_A, clr_A, clr_A, clr_A, clr_A, clr_A,
|
|
clr_A, clr_A, clr_A, clr_A, clr_A, clr_A, clr_A, clr_A,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B
|
|
);
|
|
|
|
stipple_dcross0: array[0..8] of dword = (
|
|
clr_A, clr_B, clr_A,
|
|
clr_B, clr_A, clr_B,
|
|
clr_A, clr_B, clr_A
|
|
);
|
|
|
|
stipple_dcross: array[0..63] of dword = (
|
|
clr_A, clr_B, clr_B, clr_B, clr_B, clr_B, clr_B, clr_A,
|
|
clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A,
|
|
clr_B, clr_A, clr_A, clr_B, clr_B, clr_A, clr_A, clr_B,
|
|
clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B,
|
|
//----------------------------------------------------
|
|
clr_B, clr_B, clr_B, clr_A, clr_A, clr_B, clr_B, clr_B,
|
|
clr_B, clr_B, clr_A, clr_A, clr_A, clr_A, clr_B, clr_B,
|
|
clr_B, clr_A, clr_A, clr_B, clr_B, clr_A, clr_A, clr_B,
|
|
clr_A, clr_A, clr_B, clr_B, clr_B, clr_B, clr_A, clr_A
|
|
);
|
|
|
|
function Gtk3DefaultContext: TGtk3DeviceContext;
|
|
begin
|
|
if FDefaultContext = nil then
|
|
FDefaultContext := TGtk3DeviceContext.Create(PGtkWidget(nil), False);
|
|
Result := FDefaultContext;
|
|
end;
|
|
|
|
function Gtk3ScreenContext: TGtk3DeviceContext;
|
|
begin
|
|
if FScreenContext = nil then
|
|
FScreenContext := TGtk3DeviceContext.Create(gdk_get_default_root_window, False);
|
|
Result := FScreenContext;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Name: CheckBitmap
|
|
Params: Bitmap - Handle to a bitmap (TGtk3Image)
|
|
AMethodName - Method name
|
|
AParamName - Param name
|
|
Returns: If the bitmap is valid
|
|
------------------------------------------------------------------------------}
|
|
function CheckBitmap(const ABitmap: HBITMAP; const AMethodName: String;
|
|
const AParamName: String): Boolean;
|
|
begin
|
|
Result := TObject(ABitmap) is TGtk3Image;
|
|
if Result then Exit;
|
|
|
|
if Pos('.', AMethodName) = 0 then
|
|
DebugLn('Gtk3WidgetSet ' + AMethodName + ' Error - invalid bitmap ' +
|
|
AParamName + ' = ' + DbgS(ABitmap) + '!')
|
|
else
|
|
DebugLn(AMethodName + ' Error - invalid bitmap ' + AParamName + ' = ' +
|
|
DbgS(ABitmap) + '!');
|
|
end;
|
|
|
|
procedure TColorToRGB(AColor: TColor; out R, G, B: double);
|
|
begin
|
|
R := (AColor and $FF) / 255;
|
|
G := ((AColor shr 8) and $FF) / 255;
|
|
B := ((AColor shr 16) and $FF) / 255;
|
|
end;
|
|
|
|
{Map winapi ROP to Tcairo_operator_t}
|
|
function MapRasterOpToCairo(AValue: Integer): Tcairo_operator_t;
|
|
begin
|
|
case AValue of
|
|
BLACKNESS,
|
|
R2_BLACK:
|
|
Result := CAIRO_OPERATOR_CLEAR;
|
|
|
|
SRCCOPY,
|
|
R2_COPYPEN:
|
|
Result := CAIRO_OPERATOR_SOURCE;
|
|
|
|
MERGEPAINT,
|
|
R2_MASKNOTPEN:
|
|
Result := CAIRO_OPERATOR_OUT;
|
|
|
|
SRCAND,
|
|
R2_MASKPEN:
|
|
Result := CAIRO_OPERATOR_IN;
|
|
|
|
SRCERASE,
|
|
R2_MASKPENNOT:
|
|
Result := CAIRO_OPERATOR_OUT;
|
|
|
|
R2_MERGENOTPEN:
|
|
Result := CAIRO_OPERATOR_OVER;
|
|
|
|
SRCPAINT,
|
|
R2_MERGEPEN:
|
|
Result := CAIRO_OPERATOR_OVER;
|
|
|
|
R2_MERGEPENNOT:
|
|
Result := CAIRO_OPERATOR_OVER;
|
|
|
|
R2_NOP:
|
|
Result := CAIRO_OPERATOR_DEST;
|
|
|
|
R2_NOT:
|
|
Result := CAIRO_OPERATOR_OUT;
|
|
|
|
NOTSRCCOPY,
|
|
R2_NOTCOPYPEN:
|
|
Result := CAIRO_OPERATOR_SOURCE;
|
|
|
|
PATPAINT,
|
|
R2_NOTMASKPEN:
|
|
Result := CAIRO_OPERATOR_XOR;
|
|
|
|
NOTSRCERASE,
|
|
R2_NOTMERGEPEN:
|
|
Result := CAIRO_OPERATOR_CLEAR;
|
|
|
|
DSTINVERT,
|
|
R2_NOTXORPEN:
|
|
Result := CAIRO_OPERATOR_XOR; //just for testing CAIRO_OPERATOR_DIFFERENCE;
|
|
|
|
WHITENESS,
|
|
R2_WHITE:
|
|
Result := CAIRO_OPERATOR_SCREEN;
|
|
|
|
SRCINVERT,
|
|
R2_XORPEN:
|
|
Result := CAIRO_OPERATOR_XOR;
|
|
|
|
else
|
|
Result := CAIRO_OPERATOR_OVER;
|
|
end;
|
|
end;
|
|
|
|
{Map cairo_operator_t to winapi ROP}
|
|
function MapCairoRasterOpToRasterOp(AValue: Tcairo_operator_t): Integer;
|
|
begin
|
|
case AValue of
|
|
CAIRO_OPERATOR_CLEAR:
|
|
Result := R2_BLACK;
|
|
|
|
CAIRO_OPERATOR_SOURCE:
|
|
Result := R2_COPYPEN;
|
|
|
|
CAIRO_OPERATOR_OVER:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_IN:
|
|
Result := R2_MASKPEN;
|
|
|
|
CAIRO_OPERATOR_OUT:
|
|
Result := R2_MASKPENNOT;
|
|
|
|
CAIRO_OPERATOR_ATOP:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_DEST:
|
|
Result := R2_NOP;
|
|
|
|
CAIRO_OPERATOR_DEST_OVER:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_DEST_IN:
|
|
Result := R2_MASKPEN;
|
|
|
|
CAIRO_OPERATOR_DEST_OUT:
|
|
Result := R2_MASKPENNOT;
|
|
|
|
CAIRO_OPERATOR_DEST_ATOP:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_XOR:
|
|
Result := R2_XORPEN;
|
|
|
|
CAIRO_OPERATOR_ADD:
|
|
Result := R2_MERGEPEN; // Similar to add-blend effect
|
|
|
|
CAIRO_OPERATOR_SATURATE:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_MULTIPLY:
|
|
Result := R2_MASKPEN;
|
|
|
|
CAIRO_OPERATOR_SCREEN:
|
|
Result := R2_WHITE;
|
|
|
|
CAIRO_OPERATOR_OVERLAY:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_DARKEN:
|
|
Result := R2_MASKPEN;
|
|
|
|
CAIRO_OPERATOR_LIGHTEN:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_COLOR_DODGE:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_COLOR_BURN:
|
|
Result := R2_MASKPEN;
|
|
|
|
CAIRO_OPERATOR_HARD_LIGHT:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_SOFT_LIGHT:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_DIFFERENCE:
|
|
Result := R2_NOTXORPEN;
|
|
|
|
CAIRO_OPERATOR_EXCLUSION:
|
|
Result := R2_XORPEN;
|
|
|
|
CAIRO_OPERATOR_HSL_HUE:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_HSL_SATURATION:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_HSL_COLOR:
|
|
Result := R2_MERGEPEN;
|
|
|
|
CAIRO_OPERATOR_HSL_LUMINOSITY:
|
|
Result := R2_MERGEPEN;
|
|
|
|
else
|
|
Result := R2_NOP; // Default fallback
|
|
end;
|
|
end;
|
|
|
|
{ TGtk3Cursor }
|
|
|
|
constructor TGtk3Cursor.Create(ACur:integer);
|
|
var gdk_cur: TGdkCursorType;
|
|
begin
|
|
inherited Create;
|
|
case ACur of
|
|
crArrow: gdk_cur:=GDK_ARROW;
|
|
else
|
|
gdk_cur:=GDK_ARROW;
|
|
end;
|
|
|
|
Fhandle:=TGdkCursor.new_for_display(gdk_display_get_default, gdk_cur);
|
|
end;
|
|
|
|
constructor TGtk3Cursor.Create(pixbuf: PGdkPixbuf; x,y:gint);
|
|
begin
|
|
inherited Create;
|
|
FHandle := TGdkCursor.new_from_pixbuf(TGdkDisplay.get_default(),pixbuf,x,y);
|
|
end;
|
|
|
|
constructor TGtk3Cursor.Create(img: TGtk3Image);
|
|
begin
|
|
inherited Create;
|
|
FHandle := TGdkCursor.new_from_pixbuf(TGdkDisplay.get_default(),img.Handle, img.Width, img.Height);
|
|
end;
|
|
|
|
constructor TGtk3Cursor.Create(ACur: TGdkCursorType);
|
|
begin
|
|
inherited Create;
|
|
FHandle := gdk_cursor_new(ACur);
|
|
end;
|
|
|
|
destructor TGtk3Cursor.Destroy;
|
|
begin
|
|
FHandle^.unref;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
{ TGtk3ContextObject }
|
|
|
|
constructor TGtk3ContextObject.Create;
|
|
begin
|
|
inherited Create;
|
|
FShared := False;
|
|
end;
|
|
|
|
function TGtk3ContextObject.Select(ACtx:TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
DbgS('Default context object selected, please implement');
|
|
Result:=nil;
|
|
end;
|
|
|
|
{ TGtk3Region }
|
|
|
|
constructor TGtk3Region.Create(CreateHandle: Boolean);
|
|
begin
|
|
inherited Create;
|
|
FHandle := cairo_region_create;
|
|
end;
|
|
|
|
constructor TGtk3Region.Create(CreateHandle: Boolean; X1, Y1, X2, Y2: Integer);
|
|
var
|
|
ARect: Tcairo_rectangle_int_t;
|
|
begin
|
|
inherited Create;
|
|
FHandle := nil;
|
|
ARect.x := x1;
|
|
ARect.y := y1;
|
|
ARect.width := x2 - x1;
|
|
ARect.height := y2 - y1;
|
|
FHandle := cairo_region_create_rectangle(@ARect);
|
|
end;
|
|
|
|
constructor TGtk3Region.Create(X1,Y1,X2,Y2,nW,nH: Integer);
|
|
var
|
|
ASurface: pcairo_surface_t;
|
|
cr:Pcairo_t;
|
|
rr:double;
|
|
w,h:integer;
|
|
begin
|
|
inherited Create;
|
|
FHandle := nil;
|
|
w:=x2-x1;
|
|
h:=y2-y1;
|
|
rr:=nW/2;
|
|
|
|
ASurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, w, h);
|
|
cr:=cairo_create(ASurface);
|
|
try
|
|
cairo_new_path(cr);
|
|
|
|
cairo_move_to(cr,x1,y2-rr);
|
|
cairo_line_to(cr,x1,y1+rr);
|
|
cairo_arc(cr,x1 + rr, y1 + rr, rr, pi, 3*pi/2);
|
|
cairo_line_to(cr,x2-rr,y1);
|
|
cairo_arc(cr,x2 - rr, y1 + rr, rr, 3*pi/2, 0);
|
|
cairo_line_to(cr,x2,y2-rr);
|
|
cairo_arc(cr,x2 - rr, y2 - rr, rr, 0, pi/2);
|
|
cairo_line_to(cr,x1-rr,y2);
|
|
cairo_arc(cr,x1 + rr, y2 - rr, rr, pi/2, pi);
|
|
|
|
cairo_close_path(cr);
|
|
cairo_set_source_rgba(cr,1,1,1,1);
|
|
cairo_fill_preserve(cr);
|
|
|
|
FHandle := gdk_cairo_region_create_from_surface(ASurface);
|
|
finally
|
|
cairo_destroy(cr);
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
end;
|
|
|
|
constructor TGtk3Region.CreateEllipse(X1,Y1,X2,Y2: Integer);
|
|
var
|
|
ASurface: pcairo_surface_t;
|
|
cr:Pcairo_t;
|
|
w,h:integer;
|
|
save_matrix: Tcairo_matrix_t;
|
|
begin
|
|
inherited Create;
|
|
FHandle := nil;
|
|
w:=x2-x1;
|
|
h:=y2-y1;
|
|
|
|
ASurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, w, h);
|
|
cr:=cairo_create(ASurface);
|
|
try
|
|
cairo_save(cr);
|
|
try
|
|
cairo_get_matrix(cr, @save_matrix);
|
|
cairo_translate (cr, x1 + w / 2.0 + PixelOffset, y1 + h / 2.0 + PixelOffset);
|
|
cairo_scale (cr, w / 2.0, h / 2.0);
|
|
cairo_new_path(cr);
|
|
cairo_arc
|
|
(
|
|
(*cr =*) cr,
|
|
(*xc =*) 0,
|
|
(*yc =*) 0,
|
|
(*radius =*) 1,
|
|
(*angle1 =*) 0,
|
|
(*angle2 =*) 2 * Pi
|
|
);
|
|
cairo_close_path(cr);
|
|
cairo_set_source_rgba(cr,1,1,1,1);
|
|
cairo_fill_preserve(cr);
|
|
finally
|
|
cairo_restore(cr);
|
|
end;
|
|
FHandle := gdk_cairo_region_create_from_surface(ASurface);
|
|
finally
|
|
cairo_destroy(cr);
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
end;
|
|
|
|
function TGtk3Region.Select(ACtx: TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
fContext:=ACtx;
|
|
if not Assigned(fContext) then exit(nil);
|
|
fContext.setClipRegion(Self);
|
|
Result:=Self;
|
|
end;
|
|
|
|
function TGtk3Region.Get(szbuf: integer; pbuf: pointer): integer;
|
|
begin
|
|
Result:=0;
|
|
end;
|
|
|
|
destructor TGtk3Region.Destroy;
|
|
begin
|
|
if Assigned(FHandle) then
|
|
begin
|
|
cairo_region_destroy(FHandle);
|
|
FHandle := nil;
|
|
end;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
function TGtk3Region.GetExtents: TRect;
|
|
var
|
|
ARect: Tcairo_rectangle_int_t;
|
|
begin
|
|
Result := Rect(0, 0, 0, 0);
|
|
if Assigned(FHandle) then
|
|
begin
|
|
cairo_region_get_extents(FHandle, @ARect);
|
|
Result.Left := ARect.x;
|
|
Result.Top := ARect.y;
|
|
Result.Right := ARect.width + ARect.x;
|
|
Result.Bottom := ARect.height + ARect.y;
|
|
end;
|
|
end;
|
|
|
|
function TGtk3Region.ContainsRect(ARect: TRect): Boolean;
|
|
var
|
|
ACairoRect: Tcairo_rectangle_int_t;
|
|
begin
|
|
with ACairoRect do
|
|
begin
|
|
x := ARect.Left;
|
|
y := ARect.Top;
|
|
width := ARect.Right - ARect.Left;
|
|
height := ARect.Bottom - ARect.Top;
|
|
end;
|
|
Result := cairo_region_contains_rectangle(FHandle, @ACairoRect) = CAIRO_REGION_OVERLAP_IN;
|
|
end;
|
|
|
|
function TGtk3Region.ContainsPoint(APoint: TPoint): Boolean;
|
|
begin
|
|
Result := cairo_region_contains_point(FHandle, APoint.x, APoint.y);
|
|
end;
|
|
|
|
{ TGtk3Font }
|
|
|
|
procedure TGtk3Font.UpdateLogFont;
|
|
var
|
|
sz:integer;
|
|
members:TPangoFontMask;
|
|
AStyle: TPangoStyle;
|
|
AGravity: TPangoGravity;
|
|
stretch: TPangoStretch;
|
|
weight: TPangoWeight;
|
|
begin
|
|
if not Assigned(fHandle) then exit;
|
|
fillchar(fLogFont,sizeof(fLogFont),0);
|
|
members:=fHandle^.get_set_fields;
|
|
if PANGO_FONT_MASK_FAMILY in members then
|
|
begin
|
|
fLogFont.lfFaceName:=PChar(fHandle^.get_family);
|
|
end else
|
|
begin
|
|
if PANGO_FONT_MASK_STRETCH in members then
|
|
stretch := fHandle^.get_stretch
|
|
else
|
|
stretch := PANGO_STRETCH_NORMAL;
|
|
|
|
fLogFont.lfFaceName:=AppendPangoFontFaceSuffixes(
|
|
PChar(fHandle^.get_family), stretch, PANGO_WEIGHT_NORMAL);
|
|
end;
|
|
if PANGO_FONT_MASK_STYLE in members then
|
|
begin
|
|
AStyle := fHandle^.get_style;
|
|
if AStyle = PANGO_STYLE_ITALIC then
|
|
fLogFont.lfItalic:=1;
|
|
end;
|
|
if PANGO_FONT_MASK_WEIGHT in members then
|
|
fLogFont.lfWeight := Integer(fHandle^.get_weight());
|
|
if PANGO_FONT_MASK_GRAVITY in members then
|
|
begin
|
|
AGravity := fHandle^.get_gravity;
|
|
if AGravity = PANGO_GRAVITY_SOUTH then
|
|
fLogFont.lfOrientation := 0
|
|
else
|
|
if AGravity = PANGO_GRAVITY_EAST then
|
|
fLogFont.lfOrientation := 900
|
|
else
|
|
if AGravity = PANGO_GRAVITY_NORTH then
|
|
fLogFont.lfOrientation := 1800
|
|
else
|
|
if AGravity = PANGO_GRAVITY_WEST then
|
|
fLogFont.lfOrientation := 2700;
|
|
end;
|
|
if PANGO_FONT_MASK_SIZE in members then
|
|
begin
|
|
sz:=fHandle^.get_size;
|
|
if fHandle^.get_size_is_absolute then
|
|
begin
|
|
sz:= PANGO_PIXELS(sz);
|
|
end else
|
|
begin
|
|
{ in points }
|
|
//sz:=round(96*sz/PANGO_SCALE/72);//round(2.03*sz/PANGO_SCALE);
|
|
sz := MulDiv(PANGO_PIXELS(sz), Round(gdk_screen_get_resolution(gdk_screen_get_default)), 72 );
|
|
end;
|
|
|
|
fLogFont.lfHeight:=sz;//round(sz/PANGO_SCALE);
|
|
end;
|
|
end;
|
|
|
|
constructor TGtk3Font.Create(ACairo: Pcairo_t; AWidget: PGtkWidget);
|
|
var
|
|
AContext: PPangoContext;
|
|
AOwnsContext: Boolean;
|
|
begin
|
|
inherited Create;
|
|
AOwnsContext := not Gtk3IsWidget(AWidget);
|
|
if not AOwnsContext then
|
|
begin
|
|
AContext := gtk_widget_get_pango_context(AWidget);
|
|
//DebugLn('TGtk3Font.Create AContext created from widget ....context=',dbgHex(PtrUInt(AContext)));
|
|
end else
|
|
begin
|
|
AContext := pango_cairo_create_context(ACairo);
|
|
//DebugLn('TGtk3Font.Create AContext created from pango cairo ....');
|
|
end;
|
|
FHandle := pango_font_description_copy(pango_context_get_font_description(AContext));
|
|
FFontName := pango_font_description_get_family(FHandle);
|
|
|
|
FLayout := pango_layout_new(AContext);
|
|
if FHandle^.get_size_is_absolute then
|
|
begin
|
|
FHandle^.set_absolute_size(FHandle^.get_size);
|
|
//DebugLn(['**TGtk3Font.Create size is absolute ',FFontName,' size ',FHandle^.get_size]);
|
|
end else
|
|
begin
|
|
FHandle^.set_size(FHandle^.get_size);
|
|
//DebugLn(['*TGtk3Font.Create size is not absolute ',FFontName,' size ',FHandle^.get_size]);
|
|
end;
|
|
|
|
FLayout^.set_font_description(FHandle);
|
|
//DebugLn('TGtk3Font.Create1 ',FFontName);
|
|
if AOwnsContext then
|
|
g_object_unref(AContext);
|
|
//DebugLn('TGtk3Font.Create1 ',FFontName);
|
|
end;
|
|
|
|
constructor TGtk3Font.Create(ALogFont: TLogFont; const ALongFontName: String);
|
|
var
|
|
AContext: PPangoContext;
|
|
AttrList: PPangoAttrList;
|
|
Attr: PPangoAttribute;
|
|
Family: string;
|
|
Stretch: TPangoStretch;
|
|
Weight: TPangoWeight;
|
|
begin
|
|
inherited Create;
|
|
FLogFont := ALogFont;
|
|
FFontName := ALogFont.lfFaceName;
|
|
|
|
Family := FFontName;
|
|
ExtractPangoFontFaceSuffixes(Family, Stretch, Weight);
|
|
|
|
AContext := gdk_pango_context_get;
|
|
if IsFontNameDefault(FFontName) or (FFontName = '') then
|
|
begin
|
|
if Gtk3WidgetSet.DefaultAppFontName <> '' then
|
|
FHandle := pango_font_description_from_string(PgChar(Gtk3WidgetSet.DefaultAppFontName))
|
|
else
|
|
FHandle := pango_font_description_copy(pango_context_get_font_description(AContext));
|
|
end else
|
|
begin
|
|
FHandle := TPangoFontDescription.new;
|
|
FHandle^.set_family(PgChar(Family));
|
|
end;
|
|
FFontName := FHandle^.get_family;
|
|
if ALogFont.lfHeight <> 0 then
|
|
FHandle^.set_absolute_size(Abs(ALogFont.lfHeight) * PANGO_SCALE);
|
|
if ALogFont.lfItalic > 0 then
|
|
FHandle^.set_style(PANGO_STYLE_ITALIC);
|
|
if Stretch <> PANGO_STRETCH_NORMAL then
|
|
FHandle^.set_stretch(Stretch);
|
|
if (ALogFont.lfWeight = FW_DONTCARE) or
|
|
(ALogFont.lfWeight = FW_NORMAL) or
|
|
((ALogFont.lfWeight = FW_BOLD) and (Weight >= PANGO_WEIGHT_SEMIBOLD)) then
|
|
FHandle^.set_weight(TPangoWeight(Weight))
|
|
else
|
|
FHandle^.set_weight(TPangoWeight(ALogFont.lfWeight));
|
|
FLayout := pango_layout_new(AContext);
|
|
FLayout^.set_font_description(FHandle);
|
|
|
|
if (ALogFont.lfUnderline<>0) or (ALogFont.lfStrikeOut<>0) then
|
|
begin
|
|
AttrList := pango_layout_get_attributes(FLayout);
|
|
if (AttrList = nil) then
|
|
AttrList := pango_attr_list_new();
|
|
if ALogFont.lfUnderline <> 0 then
|
|
Attr := pango_attr_underline_new(PANGO_UNDERLINE_SINGLE)
|
|
else
|
|
Attr := pango_attr_underline_new(PANGO_UNDERLINE_NONE);
|
|
pango_attr_list_change(AttrList, Attr);
|
|
|
|
Attr := pango_attr_strikethrough_new(ALogFont.lfStrikeOut<>0);
|
|
pango_attr_list_change(AttrList, Attr);
|
|
pango_layout_set_attributes(FLayout, AttrList);
|
|
pango_attr_list_unref(AttrList);
|
|
end;
|
|
g_object_unref(AContext);
|
|
end;
|
|
|
|
function TGtk3Font.Select(ACtx:TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
fContext:=ACtx;
|
|
if not Assigned(fContext) then exit(nil);
|
|
Result := fContext.CurrentFont;
|
|
fContext.CurrentFont:= Self;
|
|
end;
|
|
|
|
function TGtk3Font.Get(szbuf: integer; pbuf: pointer): integer;
|
|
begin
|
|
Result:=sizeof(Self.LogFont);
|
|
if pbuf=nil then exit;
|
|
Self.UpdateLogFont;
|
|
move(LogFont,pbuf^,min(szbuf,Result));
|
|
end;
|
|
|
|
destructor TGtk3Font.Destroy;
|
|
begin
|
|
if Assigned(FLayout) then
|
|
begin
|
|
g_object_unref(FLayout);
|
|
FLayout := nil;
|
|
end;
|
|
if Assigned(FHandle) then
|
|
begin
|
|
pango_font_description_free(FHandle);
|
|
FHandle := nil;
|
|
end;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
{ TGtk3Object }
|
|
|
|
constructor TGtk3Object.Create;
|
|
begin
|
|
FUpdateCount := 0;
|
|
end;
|
|
|
|
procedure TGtk3Object.Release;
|
|
begin
|
|
Free;
|
|
end;
|
|
|
|
procedure TGtk3Object.BeginUpdate;
|
|
begin
|
|
inc(FUpdateCount);
|
|
end;
|
|
|
|
procedure TGtk3Object.EndUpdate;
|
|
begin
|
|
if FUpdateCount > 0 then
|
|
dec(FUpdateCount);
|
|
end;
|
|
|
|
function TGtk3Object.InUpdate: Boolean;
|
|
begin
|
|
Result := FUpdateCount > 0;
|
|
end;
|
|
|
|
{ TGtk3Image }
|
|
|
|
constructor TGtk3Image.Create;
|
|
var
|
|
ACairo: Pcairo_t;
|
|
ASurface: Pcairo_surface_t;
|
|
ARect: TGdkRectangle;
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3Image.Create 1');
|
|
{$ENDIF}
|
|
inherited Create;
|
|
ACairo := gdk_cairo_create(gdk_get_default_root_window);
|
|
gdk_cairo_get_clip_rectangle(ACairo, @ARect);
|
|
cairo_destroy(ACairo);
|
|
ASurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, ARect.width, ARect.height);
|
|
try
|
|
FHandle := gdk_pixbuf_get_from_surface(ASurface, 0 ,0, ARect.Width, ARect.Height);
|
|
finally
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
FData := nil;
|
|
FDataOwner := False;
|
|
FFormat := CAIRO_FORMAT_ARGB32;
|
|
end;
|
|
|
|
constructor TGtk3Image.Create(vHandle: PGdkPixbuf);
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3Image.Create 2 vHandle=',dbgs(vHandle),' channels ',dbgs(vHandle^.get_n_channels),' bps ',dbgs(vHandle^.get_bits_per_sample),' has_alpha=',dbgs(vHandle^.get_has_alpha));
|
|
{$ENDIF}
|
|
inherited Create;
|
|
FHandle := vHandle^.copy;
|
|
FData := nil;
|
|
FDataOwner := False;
|
|
|
|
if FHandle^.get_has_alpha then
|
|
FFormat := CAIRO_FORMAT_ARGB32
|
|
else
|
|
FFormat := CAIRO_FORMAT_RGB24;
|
|
end;
|
|
|
|
function IsPixelDataEmpty(AData: PByte; const Height, BytesPerLine: Integer): Boolean;
|
|
var
|
|
y: Integer;
|
|
PixelPtr: PByte;
|
|
begin
|
|
Result := True;
|
|
|
|
if AData = nil then Exit;
|
|
|
|
for y := 0 to Height - 1 do
|
|
begin
|
|
PixelPtr := AData + (y * BytesPerLine);
|
|
if not CompareMem(PixelPtr, AData, BytesPerLine) then
|
|
exit(False);
|
|
end;
|
|
end;
|
|
|
|
constructor TGtk3Image.Create(AData: PByte; width: Integer; height: Integer;
|
|
format: Tcairo_format_t; const ADataOwner: Boolean);
|
|
var
|
|
ASurface: Pcairo_surface_t;
|
|
w,h: Integer;
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3Image.Create 3 AData=',dbgs(AData <> nil),' format=',dbgs(Ord(format)),' w=',dbgs(width),' h=',dbgs(height),' dataowner=',dbgs(ADataOwner));
|
|
{$ENDIF}
|
|
inherited Create;
|
|
FFormat := format;
|
|
FData := AData;
|
|
FDataOwner := ADataOwner;
|
|
if FData = nil then
|
|
begin
|
|
w := width;
|
|
h := height;
|
|
if w <= 0 then
|
|
w := 16;
|
|
if h <= 0 then
|
|
h := 16;
|
|
|
|
ASurface := cairo_image_surface_create(format, w, h);
|
|
try
|
|
FHandle := gdk_pixbuf_get_from_surface(ASurface, 0 ,0, w, h);
|
|
finally
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
gdk_pixbuf_fill(FHandle, 0);
|
|
end else
|
|
FHandle := TGdkPixbuf.new_from_data(AData, GDK_COLORSPACE_RGB, format=CAIRO_FORMAT_ARGB32, 8, width, height, 0, nil, nil);
|
|
end;
|
|
|
|
constructor TGtk3Image.Create(AData: PByte; width: Integer; height: Integer;
|
|
bytesPerLine: Integer; format: Tcairo_format_t; const ADataOwner: Boolean);
|
|
var
|
|
ASurface: Pcairo_surface_t;
|
|
w, h: Integer;
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3Image.Create 4 AData=',dbgs(AData <> nil),' format=',dbgs(Ord(format)),' w=',dbgs(width),' h=',dbgs(height),' dataowner=',dbgs(ADataOwner),' bpl=',dbgs(bytesPerLine));
|
|
{$endif}
|
|
inherited Create;
|
|
FFormat := format;
|
|
FData := AData;
|
|
FDataOwner := ADataOwner;
|
|
|
|
if FData = nil then
|
|
begin
|
|
w := width;
|
|
h := height;
|
|
if (w <= 0) then
|
|
w := 16;
|
|
if (h <= 0) then
|
|
h := 16;
|
|
ASurface := cairo_image_surface_create(format, w, h);
|
|
try
|
|
FHandle := gdk_pixbuf_get_from_surface(ASurface, 0 ,0, w, h);
|
|
finally
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
gdk_pixbuf_fill(FHandle, 0);
|
|
end else
|
|
begin
|
|
FHandle := TGdkPixbuf.new_from_data(AData, GDK_COLORSPACE_RGB, format=CAIRO_FORMAT_ARGB32, 8, width, height, bytesPerLine, nil, nil);
|
|
if (format = CAIRO_FORMAT_ARGB32) then
|
|
begin
|
|
if IsPixelDataEmpty(AData, Height, BytesPerLine) then
|
|
g_object_set_data(FHandle,'lcl_color_swap', Self)
|
|
else
|
|
g_object_set_data(FHandle,'lcl_no_color_swap', Self);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function TGtk3Image.Select(ACtx: TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
fContext:=ACtx;
|
|
if not Assigned(ACtx) then exit(nil);
|
|
Result := fContext.CurrentImage;
|
|
fContext.SetImage(Self);
|
|
end;
|
|
|
|
function TGtk3Image.Get(szbuf: integer; pbuf: pointer): integer;
|
|
begin
|
|
Result:=0;
|
|
end;
|
|
|
|
destructor TGtk3Image.Destroy;
|
|
begin
|
|
if FHandle <> nil then
|
|
begin
|
|
FHandle^.unref;
|
|
FHandle := nil;
|
|
end;
|
|
if (FDataOwner) and (FData <> nil) then
|
|
FreeMem(FData);
|
|
|
|
inherited Destroy;
|
|
end;
|
|
|
|
procedure TGtk3Image.CopyFrom(AImage: PGdkPixbuf; x, y, w, h: integer);
|
|
begin
|
|
if FHandle = nil then
|
|
begin
|
|
DebugLn('*TGtk3Image.CopyFrom create subpixbuf ...');
|
|
FHandle := gdk_pixbuf_new_subpixbuf(AImage, x, y, w, h);
|
|
//TODO: must
|
|
// FHandle := gdk_pixbuf_copy(AImage);
|
|
end else
|
|
begin
|
|
DebugLn('*TGtk3Image.CopyFrom AImage ...');
|
|
g_object_unref(FHandle);
|
|
FHandle := gdk_pixbuf_new_subpixbuf(AImage, x, y, w, h);
|
|
// gdk_pixbuf_copy_area(AImage, x, y, w, h, FHandle, 0, 0);
|
|
end;
|
|
end;
|
|
|
|
function TGtk3Image.height: Integer;
|
|
begin
|
|
Result := FHandle^.get_height;
|
|
end;
|
|
|
|
function TGtk3Image.width: Integer;
|
|
begin
|
|
Result := FHandle^.get_width;
|
|
end;
|
|
|
|
function TGtk3Image.depth: Integer;
|
|
var
|
|
AOption: Pgchar;
|
|
begin
|
|
Result := 32;
|
|
AOption := FHandle^.get_option('depth');
|
|
if AOption <> nil then
|
|
begin
|
|
TryStrToInt(StrPas(AOption), Result);
|
|
end;
|
|
end;
|
|
|
|
function TGtk3Image.dotsPerMeterX: Integer;
|
|
begin
|
|
Result := 0;
|
|
end;
|
|
|
|
function TGtk3Image.dotsPerMeterY: Integer;
|
|
begin
|
|
Result := 0;
|
|
end;
|
|
|
|
function TGtk3Image.bits: PByte;
|
|
begin
|
|
Result := FHandle^.pixels;
|
|
end;
|
|
|
|
function TGtk3Image.numBytes: LongWord;
|
|
begin
|
|
Result := FHandle^.get_byte_length;
|
|
end;
|
|
|
|
function TGtk3Image.bytesPerLine: Integer;
|
|
begin
|
|
Result := FHandle^.rowstride;
|
|
end;
|
|
|
|
{ TGtk3Pen }
|
|
|
|
procedure TGtk3Pen.SetColor(AValue: TColor);
|
|
var
|
|
ARed, AGreen, ABlue: Double;
|
|
begin
|
|
FColor := AValue;
|
|
ColorToCairoRGB(FColor, ARed, AGreen, ABlue);
|
|
if Assigned(FContext) and Assigned(FContext.pcr) then
|
|
begin
|
|
cairo_stroke(fContext.Pcr);
|
|
cairo_new_path(fContext.Pcr);
|
|
cairo_set_source_rgb(FContext.pcr, ARed, AGreen, ABlue);
|
|
end;
|
|
end;
|
|
|
|
constructor TGtk3Pen.Create;
|
|
begin
|
|
inherited Create;
|
|
FillChar(LogPen, SizeOf(LogPen), #0);
|
|
FIsExtPen := False;
|
|
FContext := nil;
|
|
FColor := clBlack;
|
|
FCosmetic := True;
|
|
FWidth := 0;
|
|
FStyle := psSolid;
|
|
FEndCap := pecFlat;
|
|
FJoinStyle := pjsRound;
|
|
FPenMode := pmCopy; // default pen mode
|
|
end;
|
|
|
|
function TGtk3Pen.Select(ACtx:TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
fContext:=ACtx;
|
|
if not Assigned(fContext) then exit(nil);
|
|
Result := FContext.CurrentPen;
|
|
fContext.CurrentPen := Self;
|
|
Self.SetColor(fColor); // update Cairo
|
|
end;
|
|
|
|
function TGtk3Pen.Get(szbuf: integer; pbuf: pointer): integer;
|
|
begin
|
|
Result:=sizeof(LogPen);
|
|
if pbuf=nil then exit;
|
|
move(LogPen,pbuf^,min(result,szbuf));
|
|
end;
|
|
|
|
procedure TGtk3Pen.setCosmetic(b: Boolean);
|
|
begin
|
|
FCosmetic := B;
|
|
if Assigned(FContext) and Assigned(FContext.pcr) then
|
|
begin
|
|
if b then
|
|
cairo_set_line_width(FContext.pcr, 0)
|
|
else
|
|
cairo_set_line_width(FContext.pcr, 1);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3Pen.setWidth(p1: Integer);
|
|
begin
|
|
FWidth := p1;
|
|
if Assigned(FContext) then
|
|
cairo_set_line_width(FContext.pcr, p1);
|
|
end;
|
|
|
|
{ TGtk3Brush }
|
|
|
|
procedure TGtk3Brush.SetColor(AValue: TColor);
|
|
var
|
|
ARed, AGreen, ABlue: Double;
|
|
begin
|
|
FColor := AValue;
|
|
ColorToCairoRGB(FColor, ARed, AGreen, ABlue);
|
|
if Assigned(FContext) then
|
|
cairo_set_source_rgb(FContext.pcr, ARed, AGreen, ABlue);
|
|
end;
|
|
|
|
procedure TGtk3Brush.SetStyle(AStyle: longword);
|
|
begin
|
|
if AStyle=fStyle then exit;
|
|
fStyle:=AStyle;
|
|
Self.UpdatePattern(ColorToRGB(FColor));
|
|
end;
|
|
|
|
constructor TGtk3Brush.Create;
|
|
begin
|
|
inherited Create;
|
|
FColor := clNone;
|
|
FillChar(LogBrush, SizeOf(TLogBrush), #0);
|
|
end;
|
|
|
|
function TGtk3Brush.Select(ACtx:TGtk3DeviceContext): TGtk3ContextObject;
|
|
begin
|
|
fContext:=ACtx;
|
|
if not Assigned(fContext) then exit(nil);
|
|
Result := fContext.CurrentBrush;
|
|
Self.UpdatePattern(ColorToRGB(FColor));
|
|
fContext.CurrentBrush := Self;
|
|
end;
|
|
|
|
function TGtk3Brush.Get(szbuf: integer; pbuf: pointer): integer;
|
|
begin
|
|
Result:=sizeof(LogBrush);
|
|
if pbuf=nil then exit;
|
|
move(LogBrush,pbuf^,min(Result,szbuf));
|
|
end;
|
|
|
|
destructor TGtk3Brush.Destroy;
|
|
begin
|
|
if Assigned(brush_pattern) then
|
|
cairo_pattern_destroy(brush_pattern);
|
|
if Assigned(pat_buf) then
|
|
FreeMemAndNil(pat_buf);
|
|
inherited Destroy;
|
|
end;
|
|
|
|
procedure TGtk3Brush.UpdatePattern(const aColor: TColorRef);
|
|
var
|
|
w,h,i,j:integer;
|
|
rgb:array[0..3] of byte absolute aColor;
|
|
pat_sample,psrc,pdst:pdword;
|
|
begin
|
|
if Self.LogBrush.lbStyle<>BS_HATCHED then exit;
|
|
|
|
if Assigned(Self.brush_pattern) then
|
|
begin
|
|
cairo_pattern_destroy(brush_pattern);
|
|
FreeMemAndNil(pat_buf);
|
|
brush_pattern := nil;
|
|
end;
|
|
case TBrushStyle(Self.LogBrush.lbHatch+ord(bsHorizontal)) of
|
|
bsHorizontal:
|
|
begin
|
|
w:=4; h:=4;
|
|
pat_sample:=@stipple_horz[0];
|
|
end;
|
|
bsVertical:
|
|
begin
|
|
w:=4; h:=4;
|
|
pat_sample:=@stipple_vert[0];
|
|
end;
|
|
bsFDiagonal:
|
|
begin
|
|
w:=8; h:=8;
|
|
pat_sample:=@stipple_fdiag[0];
|
|
end;
|
|
bsBDiagonal:
|
|
begin
|
|
w:=8; h:=8;
|
|
pat_sample:=@stipple_bdiag[0];
|
|
end;
|
|
bsCross:
|
|
begin
|
|
w:=8; h:=8;
|
|
pat_sample:=@stipple_cross1[0];
|
|
end;
|
|
bsDiagCross:
|
|
begin
|
|
w:=8; h:=8;
|
|
pat_sample:=@stipple_dcross[0];
|
|
end;
|
|
else
|
|
exit
|
|
end;
|
|
psrc:=pat_sample;
|
|
getmem(pat_buf,w*h*sizeof(dword));
|
|
pdst:=pat_buf;
|
|
for i:=0 to h-1 do
|
|
for j:=0 to w-1 do
|
|
begin
|
|
case psrc^ of
|
|
clr_A: pdst^:=$ff000000 or (rgb[0] shl 16) or (rgb[1] shl 8) or (rgb[2]);
|
|
clr_B: pdst^:=$ffffffff;
|
|
end;
|
|
inc(psrc); inc(pdst);
|
|
end;
|
|
{GTK3 states the buffer must exist, until image that uses the buffer - destroyed}
|
|
brush_pattern:=create_stipple(PByte(pat_buf),w,w);
|
|
end;
|
|
|
|
|
|
|
|
function create_stipple(stipple_data:pbyte;width,height:integer):pcairo_pattern_t;
|
|
var
|
|
surface:pcairo_surface_t;
|
|
pattern:pcairo_pattern_t;
|
|
stride:integer;
|
|
begin
|
|
stride := cairo_format_stride_for_width (CAIRO_FORMAT_ARGB32, width);
|
|
surface := cairo_image_surface_create_for_data (stipple_data, CAIRO_FORMAT_ARGB32, width, height,
|
|
stride);
|
|
pattern := cairo_pattern_create_for_surface (surface);
|
|
cairo_surface_destroy (surface);
|
|
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
|
|
|
result:= pattern;
|
|
end;
|
|
|
|
function DebugColor(AColor: TColor): string;
|
|
var
|
|
R, G, B: double;
|
|
begin
|
|
TColorToRGB(AColor, R, G, B);
|
|
Result := Format('DebugColor: R %2.2n G %2.2n B %2.2n',[R, G, B]);
|
|
end;
|
|
|
|
{ TGtk3DeviceContext }
|
|
|
|
function TGtk3DeviceContext.GetBkColor:TColorRef;
|
|
begin
|
|
Result := FBkColor;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.GetOffset: TPoint;
|
|
var
|
|
dx,dy: Double;
|
|
begin
|
|
cairo_surface_get_device_offset(cairo_get_target(pcr), @dx, @dy);
|
|
Result := Point(Round(dx), Round(dy));
|
|
end;
|
|
|
|
function TGtk3DeviceContext.GetRasterOp: integer;
|
|
begin
|
|
if FXorMode then
|
|
Result := R2_XORPEN
|
|
else
|
|
Result := MapCairoRasterOpToRasterOp(cairo_get_operator(pcr));
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.setBrush(AValue: TGtk3Brush);
|
|
begin
|
|
if Assigned(FBrush) then
|
|
FBrush.Free;
|
|
FBrush := AValue;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.setBkColor(AValue:TColorRef);
|
|
begin
|
|
FBkColor := AValue;
|
|
if Assigned(FBgBrush) then
|
|
begin
|
|
FBgBrush.Free;
|
|
FBgBrush := nil;
|
|
end;
|
|
if FBkColor = clNone then
|
|
exit;
|
|
FBgBrush := TGtk3Brush.Create;
|
|
FBgBrush.Context := Self;
|
|
FBgBrush.Style := BS_SOLID;
|
|
FBgBrush.Color := ColorToRGB(LongInt(FBkColor));
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetFont(AValue: TGtk3Font);
|
|
begin
|
|
if Assigned(FFont) then
|
|
FFont.Free;
|
|
FFont := AValue;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetOffset(AValue: TPoint);
|
|
var
|
|
dx, dy: Double;
|
|
begin
|
|
dx := AValue.X;
|
|
dy := AValue.Y;
|
|
cairo_surface_set_device_offset(cairo_get_target(pcr), dx, dy);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.setPen(AValue: TGtk3Pen);
|
|
begin
|
|
if Assigned(FPen) then
|
|
FPen.Free;
|
|
FPen := AValue;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetRasterOp(AValue: integer);
|
|
var
|
|
AMap: Tcairo_operator_t;
|
|
begin
|
|
if MapCairoRasterOpToRasterOp(cairo_get_operator(pcr)) = AValue then
|
|
exit;
|
|
if FXorMode and ((AValue <> R2_XORPEN) and (AValue <> R2_NOTXORPEN)) then
|
|
begin
|
|
FXorMode := False;
|
|
ApplyXorDrawing(FXorSurface, cairo_get_operator(FXorCairo));
|
|
if FXorCairo <> nil then
|
|
begin
|
|
cairo_destroy(FXorCairo);
|
|
FXorCairo := nil;
|
|
end;
|
|
if FXorSurface <> nil then
|
|
begin
|
|
cairo_surface_destroy(FXorSurface);
|
|
FXorSurface := nil;
|
|
end;
|
|
end;
|
|
AMap := MapRasterOpToCairo(AValue);
|
|
if AMap <> CAIRO_OPERATOR_XOR then
|
|
cairo_set_operator(pcr, MapRasterOpToCairo(AValue))
|
|
else
|
|
begin
|
|
if FXorMode then
|
|
exit;
|
|
FXorMode := True;
|
|
if FXorSurface <> nil then
|
|
begin
|
|
if FXorCairo <> nil then
|
|
cairo_destroy(FXorCairo);
|
|
FXorCairo := nil;
|
|
cairo_surface_destroy(FXorSurface);
|
|
FXorSurface := nil;
|
|
end;
|
|
if AValue = R2_NOTXORPEN then
|
|
AMap := CAIRO_OPERATOR_OVER;
|
|
SetXorMode(FXorSurface, AMap);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetXorMode(var xorSurface: Pcairo_surface_t;
|
|
AMap: Tcairo_operator_t);
|
|
var
|
|
R: TGdkRectangle;
|
|
begin
|
|
if xorSurface <> nil then
|
|
raise Exception.Create('TGtk3DeviceContext: xorSurface <> nil !');
|
|
|
|
gdk_cairo_get_clip_rectangle(FCairo, @R);
|
|
|
|
xorSurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, R.width, R.height);
|
|
FXorCairo := cairo_create(xorSurface);
|
|
cairo_set_source_rgba(FXorCairo, 0, 0, 0, 0);
|
|
cairo_set_operator(FXorCairo, CAIRO_OPERATOR_CLEAR);
|
|
cairo_paint(FXorCairo);
|
|
cairo_set_operator(FXorCairo, AMap);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.ApplyXorDrawing(xorSurface: Pcairo_surface_t;
|
|
AMap: Tcairo_operator_t);
|
|
var
|
|
width, height: Integer;
|
|
AOperator: Tcairo_operator_t;
|
|
begin
|
|
AOperator := cairo_get_operator(FCairo);
|
|
cairo_surface_flush(xorSurface);
|
|
cairo_set_source_surface(FCairo, xorSurface, 0, 0);
|
|
if aMap = CAIRO_OPERATOR_XOR then
|
|
cairo_set_operator(FCairo, CAIRO_OPERATOR_DIFFERENCE)
|
|
else
|
|
cairo_set_operator(FCairo, CAIRO_OPERATOR_OVER);
|
|
cairo_paint(FCairo);
|
|
cairo_set_operator(FCairo, AOperator);
|
|
end;
|
|
|
|
|
|
procedure TGtk3DeviceContext.SetvImage(AValue: TGtk3Image);
|
|
begin
|
|
if Assigned(FvImage) then
|
|
FvImage.Free;
|
|
FvImage.Free;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.SX(const x: double): Double;
|
|
begin
|
|
Result := 1*(x+vClipRect.Left);
|
|
end;
|
|
|
|
function TGtk3DeviceContext.SY(const y: double): Double;
|
|
begin
|
|
Result := 1*(y+vClipRect.Top);
|
|
end;
|
|
|
|
function TGtk3DeviceContext.SX2(const x: double): Double;
|
|
begin
|
|
Result := x;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.SY2(const y: double): Double;
|
|
begin
|
|
Result := y;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.ApplyBrush;
|
|
begin
|
|
SetSourceColor(FCurrentBrush.Color);
|
|
if Self.FCurrentBrush.Style<>0 then
|
|
begin
|
|
if Assigned(Self.FCurrentBrush.brush_pattern) then
|
|
begin
|
|
//According to MSDN hatched brushes uses BkColor when BkMode = opaque
|
|
if (FCurrentBrush.Style = BS_HATCHED) and (BkMode <> TRANSPARENT) then
|
|
begin
|
|
//Setting another style will update pattern again.
|
|
FCurrentBrush.UpdatePattern(FBkColor);
|
|
end;
|
|
cairo_set_source(pcr,Self.FCurrentBrush.brush_pattern);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.ApplyFont;
|
|
var
|
|
AFont: TGtk3Font;
|
|
begin
|
|
if Assigned(FCurrentFont) then
|
|
AFont := FCurrentFont
|
|
else
|
|
AFont := FFont;
|
|
if AFont<>nil then ;
|
|
DebugLn(['TGtk3DeviceContext.ApplyFont ToDo']);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.ApplyPen;
|
|
|
|
procedure SetDash(d: array of double);
|
|
begin
|
|
cairo_set_dash(pcr, @d[0], High(d)+1, 0);
|
|
end;
|
|
|
|
var
|
|
cap: Tcairo_line_cap_t;
|
|
w: Double;
|
|
begin
|
|
SetSourceColor(FCurrentPen.Color);
|
|
(*
|
|
case FCurrentPen.Mode of
|
|
pmBlack: begin
|
|
SetSourceColor(clBlack);
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
end;
|
|
pmWhite: begin
|
|
SetSourceColor(clWhite);
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
end;
|
|
pmCopy: cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
pmXor: cairo_set_operator(pcr, CAIRO_OPERATOR_XOR);
|
|
pmNotXor: cairo_set_operator(pcr, CAIRO_OPERATOR_DIFFERENCE);
|
|
{pmNop,
|
|
pmNot,
|
|
pmCopy,
|
|
pmNotCopy,
|
|
pmMergePenNot,
|
|
pmMaskPenNot,
|
|
pmMergeNotPen,
|
|
pmMaskNotPen,
|
|
pmMerge,
|
|
pmNotMerge,
|
|
pmMask,
|
|
pmNotMask,}
|
|
else
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
end;
|
|
*)
|
|
|
|
if FCurrentPen.Cosmetic then
|
|
cairo_set_line_width(pcr, 1.0)
|
|
else
|
|
begin
|
|
w := FCurrentPen.Width;
|
|
if w <= 1 then
|
|
w := 0.5;
|
|
cairo_set_line_width(pcr, w {* ScaleX}); //line_width is diameter of the pen circle
|
|
end;
|
|
|
|
cap := CAIRO_LINE_CAP_BUTT;
|
|
|
|
case FCurrentPen.Style of
|
|
psSolid: cairo_set_dash(pcr, nil, 0, 0);
|
|
psDash: SetDash(Dash_Dash);
|
|
psDot: SetDash(Dash_Dot);
|
|
psDashDot: SetDash(Dash_DashDot);
|
|
psDashDotDot: SetDash(Dash_DashDotDot);
|
|
else
|
|
cairo_set_dash(pcr, nil, 0, 0);
|
|
end;
|
|
|
|
case FCurrentPen.EndCap of
|
|
pecRound: cap := CAIRO_LINE_CAP_ROUND;
|
|
pecSquare: cap := CAIRO_LINE_CAP_SQUARE;
|
|
pecFlat: cap := CAIRO_LINE_CAP_BUTT;
|
|
end;
|
|
|
|
// dashed patterns do not look ok combined with round or squared caps
|
|
// make it flat until a solution is found
|
|
case FCurrentPen.Style of
|
|
psDash, psDot, psDashDot, psDashDotDot:
|
|
cap := CAIRO_LINE_CAP_BUTT
|
|
else
|
|
;
|
|
end;
|
|
cairo_set_line_cap(pcr, cap);
|
|
|
|
case FCurrentPen.JoinStyle of
|
|
pjsRound: cairo_set_line_join(pcr, CAIRO_LINE_JOIN_ROUND);
|
|
pjsBevel: cairo_set_line_join(pcr, CAIRO_LINE_JOIN_BEVEL);
|
|
pjsMiter: cairo_set_line_join(pcr, CAIRO_LINE_JOIN_MITER);
|
|
end;
|
|
|
|
end;
|
|
|
|
constructor TGtk3DeviceContext.Create(AWidget: PGtkWidget;
|
|
const APaintEvent: Boolean);
|
|
var
|
|
W: gint;
|
|
H: gint;
|
|
ACairo:pcairo_t;
|
|
ARect: TGdkRectangle;
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.Create (',
|
|
' WidgetHandle: ', dbghex(PtrInt(AWidget)),
|
|
' FromPaintEvent:',BoolToStr(APaintEvent),' )');
|
|
{$endif}
|
|
inherited Create;
|
|
FXorSurface := nil;
|
|
FCanvasScaleFactor := 1;
|
|
FvClipRect := Rect(0, 0, 0, 0);
|
|
Window := nil;
|
|
Parent := nil;
|
|
ParentPixmap := nil;
|
|
CairoSurface := nil;
|
|
FCanRelease := False;
|
|
FOwnsCairo := True;
|
|
FOwnsSurface := False;
|
|
FCurrentTextColor := clBlack;
|
|
FBkColor := clWhite;
|
|
FDCSaveCounter := 0;
|
|
if AWidget = nil then
|
|
begin
|
|
//no need for ParentPixmap, this is Mem DC.
|
|
CairoSurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 10, 10);
|
|
FCairo := cairo_create(CairoSurface);
|
|
FOwnsSurface := True;
|
|
end else
|
|
begin
|
|
Parent := AWidget;
|
|
if not APaintEvent then
|
|
begin
|
|
{avoid paints on null pixmaps !}
|
|
W := gtk_widget_get_allocated_width(AWidget);
|
|
H := gtk_widget_get_allocated_height(AWidget);
|
|
if W <= 0 then W := 1;
|
|
if H <= 0 then H := 1;
|
|
CairoSurface := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, W, H);
|
|
FOwnsSurface := True;
|
|
ParentPixmap := gdk_pixbuf_get_from_surface(CairoSurface, 0, 0, W, H);
|
|
FCairo := cairo_create(CairoSurface);
|
|
end else
|
|
begin
|
|
W := gtk_widget_get_allocated_width(AWidget);
|
|
H := gtk_widget_get_allocated_height(AWidget);
|
|
if W <= 0 then W := 1;
|
|
if H <= 0 then H := 1;
|
|
FCairo := gdk_cairo_create(gtk_widget_get_window(AWidget));
|
|
end;
|
|
end;
|
|
if not FOwnsSurface then
|
|
CairoSurface := cairo_get_target(FCairo);
|
|
CreateObjects;
|
|
end;
|
|
|
|
constructor TGtk3DeviceContext.Create(AWindow: PGdkWindow;
|
|
const APaintEvent: Boolean);
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.Create (',
|
|
' WindowHandle: ', dbghex(PtrInt(AWindow)),
|
|
' FromPaintEvent:',BoolToStr(APaintEvent),' )');
|
|
{$endif}
|
|
inherited Create;
|
|
FDCSaveCounter := 0;
|
|
FXorSurface := nil;
|
|
FCanvasScaleFactor := 1;
|
|
FvClipRect := Rect(0, 0, 0, 0);
|
|
Parent := nil;
|
|
ParentPixmap := nil;
|
|
CairoSurface := nil;
|
|
Window := AWindow;
|
|
FOwnsSurface := False;
|
|
FCanRelease := False;
|
|
FOwnsCairo := True;
|
|
|
|
FCurrentTextColor := clBlack;
|
|
FBkColor := clWhite;
|
|
FCairo := gdk_cairo_create(AWindow);
|
|
gdk_cairo_set_source_window(FCairo, AWindow, 0, 0);
|
|
CairoSurface := cairo_get_target(FCairo);
|
|
CreateObjects;
|
|
end;
|
|
|
|
constructor TGtk3DeviceContext.CreateFromCairo(AWidget: PGtkWidget;
|
|
ACairo: PCairo_t);
|
|
var
|
|
AGdkRect: TGdkRectangle;
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.CreateFromCairo (',
|
|
' WidgetHandle: ', dbghex(PtrInt(AWidget)),
|
|
' FromPaintEvent:',BoolToStr(True),' )');
|
|
{$endif}
|
|
inherited Create;
|
|
FDCSaveCounter := 0;
|
|
FXorSurface := nil;
|
|
FCanvasScaleFactor := 1;
|
|
FOwnsCairo := False;
|
|
Window := nil;
|
|
Parent := AWidget;
|
|
ParentPixmap := nil;
|
|
CairoSurface := nil;
|
|
FOwnsSurface := False;
|
|
FCurrentTextColor := clBlack;
|
|
FBkColor := clWhite;
|
|
gdk_cairo_get_clip_rectangle(ACairo, @AGdkRect);
|
|
FvClipRect := RectFromGdkRect(AGdkRect);
|
|
FCairo := ACairo;
|
|
CairoSurface := cairo_get_target(FCairo);
|
|
CreateObjects;
|
|
end;
|
|
|
|
destructor TGtk3DeviceContext.Destroy;
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.Destroy ',dbgHex(PtrUInt(Self)));
|
|
{$endif}
|
|
DeleteObjects;
|
|
|
|
if FXorMode then
|
|
RasterOp := R2_COPYPEN;
|
|
|
|
if FDCSaveCounter > 0 then
|
|
DebugLn('WARNING: TGtk3DeviceContext: Unpaired Cairo save/restore calls. Current count = ',dbgs(FDCSaveCounter),', but should be 0.');
|
|
|
|
if FOwnsCairo and (FCairo <> nil) then
|
|
cairo_destroy(FCairo);
|
|
|
|
if FXorCairo <> nil then
|
|
begin
|
|
//never should happen
|
|
cairo_destroy(FXorCairo);
|
|
FXorCairo := nil;
|
|
end;
|
|
if (ParentPixmap <> nil) then
|
|
g_object_unref(ParentPixmap);
|
|
if FOwnsSurface and (CairoSurface <> nil) then
|
|
cairo_surface_destroy(CairoSurface);
|
|
Parent := nil;
|
|
FCairo := nil;
|
|
ParentPixmap := nil;
|
|
CairoSurface := nil;
|
|
Window := nil;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.CreateObjects;
|
|
var
|
|
Matrix: Tcairo_matrix_t;
|
|
begin
|
|
FLastPenX := 0;
|
|
FLastPenY := 0;
|
|
FBgBrush := nil; // created on demand
|
|
FBkMode := TRANSPARENT;
|
|
FCurrentImage := nil;
|
|
FCurrentRegion := nil;
|
|
FBrush := TGtk3Brush.Create;
|
|
FBrush.Context := Self;
|
|
FBrush.Color := clNone;
|
|
FBrush.Style := BS_SOLID;
|
|
FPen := TGtk3Pen.Create;
|
|
FPen.Context := Self;
|
|
FPen.Color := clBlack;
|
|
FCurrentPen := FPen;
|
|
FCurrentBrush := FBrush;
|
|
FFont := TGtk3Font.Create(FCairo, Parent);
|
|
FCurrentFont := FFont;
|
|
FvImage := TGtk3Image.Create(nil, 1, 1, 8, CAIRO_FORMAT_ARGB32);
|
|
FCurrentImage := FvImage;
|
|
|
|
cairo_get_matrix(FCairo, @Matrix);
|
|
// widget with menu or other non-client exclusions have offset in trasform matrix
|
|
fncOrigin:=Point(round(Matrix.x0),round(Matrix.y0));
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.DeleteObjects;
|
|
begin
|
|
if Assigned(FBrush) then
|
|
FreeAndNil(FBrush);
|
|
if Assigned(FPen) then
|
|
FreeAndNil(FPen);
|
|
if Assigned(FFont) then
|
|
FreeAndNil(FFont);
|
|
if Assigned(FvImage) then
|
|
FreeAndNil(FvImage);
|
|
if Assigned(FBgBrush) then
|
|
FreeAndNil(FBgBrush);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawPixel(x, y: Integer; AColor: TColor);
|
|
// Seems that painting line from (a-1, b-1) to (a,b) gives one pixel
|
|
begin
|
|
SetSourceColor(AColor);
|
|
cairo_new_path(pcr);
|
|
cairo_set_line_width(pcr, 1);
|
|
cairo_move_to(pcr, x - PixelOffset, y - PixelOffset);
|
|
cairo_line_to(pcr, x + PixelOffset, y + PixelOffset);
|
|
cairo_stroke(pcr);
|
|
end;
|
|
|
|
function TGtk3DeviceContext.pcr: Pcairo_t;
|
|
begin
|
|
Result := FXorCairo;
|
|
if Result = nil then
|
|
Result := FCairo;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.getPixel(x, y: Integer): TColor;
|
|
var
|
|
APixbuf: PGdkPixbuf;
|
|
AData: PByte;
|
|
APixelValue: Longword;
|
|
ASurfaceWidth, ASurfaceHeight, ARowStride: Integer;
|
|
AOutSize: Tcairo_rectangle_int_t;
|
|
ARegion: Pcairo_region_t;
|
|
pixels,row: pointer;
|
|
stride,r,c:integer;
|
|
|
|
st:Tcairo_surface_type_t;
|
|
cr:Pcairo_t;
|
|
img_surf,view:Pcairo_surface_t;
|
|
begin
|
|
Result := 0;
|
|
|
|
if CairoSurface = nil then
|
|
exit;
|
|
|
|
cairo_surface_flush(CairoSurface);
|
|
|
|
st := cairo_surface_get_type(CairoSurface);
|
|
if st in [CAIRO_SURFACE_TYPE_XLIB, CAIRO_SURFACE_TYPE_XCB] then
|
|
begin
|
|
(* Allocate an image surface of a suitable size *)
|
|
view:=cairo_surface_create_for_rectangle(CairoSurface,fncOrigin.X + x -PixelOffset, fncOrigin.Y + y - PixelOffset,1 + PixelOffset, 1 + PixelOffset);
|
|
img_surf := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 1, 1);
|
|
cr := cairo_create(img_surf);
|
|
cairo_set_source_surface(cr, view, 0, 0);
|
|
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
|
|
cairo_paint(cr);
|
|
cairo_surface_flush (img_surf);
|
|
// cairo_surface_write_to_png(img_surf, 'image.png');
|
|
cairo_destroy(cr);
|
|
pixels := cairo_image_surface_get_data(img_surf);
|
|
APixelValue:=PLongInt(pixels)^;
|
|
cairo_surface_destroy(img_surf);
|
|
cairo_surface_destroy(view);
|
|
end else
|
|
if st=CAIRO_SURFACE_TYPE_IMAGE then
|
|
begin
|
|
pixels := cairo_image_surface_get_data(CAirosurface);
|
|
if Assigned(pixels) then
|
|
begin
|
|
stride := cairo_image_surface_get_stride(CairoSurface);
|
|
row:=pixels+(fncOrigin.Y+Y)*stride;
|
|
inc(row,(fncOrigin.X+X)*sizeof(longint));
|
|
APixelValue:=PLongInt(row)^;
|
|
end;
|
|
end;
|
|
Result := ((APixelValue and $FF0000) shr 16) or
|
|
(APixelValue and $00FF00) or
|
|
((APixelValue and $0000FF) shl 16);
|
|
|
|
end;
|
|
|
|
|
|
procedure TGtk3DeviceContext.drawRect(x1, y1, w, h: Integer; const AFill, ABorder: Boolean);
|
|
var
|
|
aOp: Tcairo_operator_t;
|
|
begin
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
cairo_rectangle(pcr, x1 + PixelOffset, y1 + PixelOffset, w - 2*PixelOffset, h - 2*PixelOffset);
|
|
if AFill then
|
|
begin
|
|
ApplyBrush;
|
|
cairo_fill_preserve(pcr);
|
|
end;
|
|
if ABorder then
|
|
begin
|
|
ApplyPen;
|
|
cairo_stroke(pcr);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawRoundRect(x, y, w, h, rx, ry: Integer);
|
|
begin
|
|
RoundRect(x, y, w, h, rx, ry);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawText(x, y: Integer; AText: PChar; ALen: Integer;
|
|
const ABgFilled: Boolean);
|
|
var
|
|
R, G, B: Double;
|
|
gColor: TGdkColor;
|
|
Attr: PPangoAttribute;
|
|
AttrList: PPangoAttrList;
|
|
ornt: Integer;
|
|
begin
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
cairo_move_to(pcr, x, y);
|
|
|
|
ornt := Self.FCurrentFont.FLogFont.lfOrientation;
|
|
if ornt <> 0 then
|
|
cairo_rotate(pcr, -Pi * (ornt / 10) / 180);
|
|
|
|
ColorToCairoRGB(ColorToRgb(TColor(CurrentTextColor)), R, G, B);
|
|
cairo_set_source_rgb(pcr, R, G, B);
|
|
|
|
FCurrentFont.Layout^.set_text(AText, ALen);
|
|
|
|
if ABgFilled then
|
|
begin
|
|
gColor := TColorToTGDKColor(FBgBrush.Color);
|
|
AttrList := pango_attr_list_new;
|
|
Attr := pango_attr_background_new(gColor.red, gColor.green, gColor.blue);
|
|
pango_attr_list_insert(AttrList, Attr);
|
|
FCurrentFont.Layout^.set_attributes(AttrList);
|
|
pango_attr_list_unref(AttrList);
|
|
end;
|
|
|
|
pango_cairo_show_layout(pcr, FCurrentFont.Layout);
|
|
|
|
if ABgFilled then
|
|
FCurrentFont.Layout^.set_attributes(nil);
|
|
|
|
if ornt <> 0 then
|
|
cairo_rotate(pcr, Pi * (ornt / 10) / 180);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawEllipse(x, y, w, h: Integer; AFill, ABorder: Boolean);
|
|
var
|
|
scale_x, scale_y: Double;
|
|
begin
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
scale_x := w / 2.0;
|
|
scale_y := h / 2.0;
|
|
|
|
cairo_translate(pcr, x + scale_x + PixelOffset, y + scale_y + PixelOffset);
|
|
cairo_scale(pcr, scale_x, scale_y);
|
|
|
|
cairo_new_path(pcr);
|
|
cairo_arc(pcr, 0, 0, 1, 0, 2 * Pi);
|
|
cairo_close_path(pcr);
|
|
|
|
if AFill then
|
|
begin
|
|
ApplyBrush;
|
|
cairo_fill_preserve(pcr);
|
|
end;
|
|
|
|
cairo_scale(pcr, 1 / scale_x, 1 / scale_y);
|
|
cairo_translate(pcr, -(x + scale_x + PixelOffset), -(y + scale_y + PixelOffset));
|
|
|
|
if ABorder then
|
|
begin
|
|
ApplyPen;
|
|
cairo_stroke(pcr);
|
|
end;
|
|
end;
|
|
|
|
procedure SwapRedBlueChannels(PixBuf: PGdkPixbuf);
|
|
var
|
|
Pixels: PByte;
|
|
x, y, RowStride, Width, Height, Channels: Integer;
|
|
Temp: Byte;
|
|
begin
|
|
if PixBuf = nil then
|
|
begin
|
|
DebugLn('ERROR: SwapRedBlueChannels: Pixbuf is nil!');
|
|
Exit;
|
|
end;
|
|
|
|
Width := gdk_pixbuf_get_width(PixBuf);
|
|
Height := gdk_pixbuf_get_height(PixBuf);
|
|
RowStride := gdk_pixbuf_get_rowstride(PixBuf);
|
|
Channels := gdk_pixbuf_get_n_channels(PixBuf);
|
|
|
|
if Channels < 3 then
|
|
begin
|
|
DebugLn('ERROR: SwapRedBlueChannels: Pixbuf does not have enough color channels (expected at least 3).');
|
|
Exit;
|
|
end;
|
|
|
|
Pixels := gdk_pixbuf_get_pixels(PixBuf);
|
|
|
|
for y := 0 to Height - 1 do
|
|
begin
|
|
for x := 0 to Width - 1 do
|
|
begin
|
|
Temp := Pixels[x * Channels];
|
|
Pixels[x * Channels] := Pixels[x * Channels + 2];
|
|
Pixels[x * Channels + 2] := Temp;
|
|
end;
|
|
Inc(Pixels, RowStride);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawSurface(targetRect: PRect;
|
|
Surface: Pcairo_surface_t; sourceRect: PRect; aPixBuf: PGdkPixBuf;
|
|
mask: PGdkPixBuf; maskRect: PRect);
|
|
var
|
|
M: Tcairo_matrix_t;
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.DrawSurface ');
|
|
{$ENDIF}
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
with targetRect^ do
|
|
cairo_rectangle(pcr, Left + PixelOffset, Top + PixelOffset, Right - Left, Bottom - Top);
|
|
|
|
if aPixBuf <> nil then
|
|
begin
|
|
aPixBuf^.ref;
|
|
//this fixes problem with some images being R & B swapped when blitted onto dest.
|
|
if (cairo_surface_get_type(CairoSurface) <> cairo_surface_get_type(Surface)) and
|
|
(g_object_get_data(aPixBuf,'lcl_color_swap') <> nil) then
|
|
begin
|
|
SwapRedBlueChannels(aPixBuf);
|
|
gdk_cairo_set_source_pixbuf(pcr, aPixBuf, 0, 0);
|
|
end else
|
|
begin
|
|
if g_object_get_data(aPixBuf,'lcl_no_color_swap') <> nil then
|
|
gdk_cairo_set_source_pixbuf(pcr, aPixBuf, 0, 0)
|
|
else
|
|
cairo_set_source_surface(pcr, Surface, 0, 0);
|
|
end;
|
|
aPixBuf^.unref;
|
|
end else
|
|
cairo_set_source_surface(pcr, Surface, 0, 0);
|
|
|
|
cairo_matrix_init_identity(@M);
|
|
cairo_matrix_translate(@M, SourceRect^.Left, SourceRect^.Top);
|
|
cairo_matrix_scale(@M,
|
|
(sourceRect^.Right - sourceRect^.Left) / (targetRect^.Right - targetRect^.Left),
|
|
(sourceRect^.Bottom - sourceRect^.Top) / (targetRect^.Bottom - targetRect^.Top)
|
|
);
|
|
cairo_matrix_translate(@M, -targetRect^.Left, -targetRect^.Top);
|
|
cairo_pattern_set_matrix(cairo_get_source(pcr), @M);
|
|
|
|
cairo_clip(pcr);
|
|
cairo_paint(pcr);
|
|
|
|
cairo_reset_clip(pcr);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawImage(targetRect: PRect; image: PGdkPixBuf;
|
|
sourceRect: PRect; mask: PGdkPixBuf; maskRect: PRect);
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.DrawImage ');
|
|
{$ENDIF}
|
|
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
gdk_cairo_set_source_pixbuf(pcr, Image, 0, 0);
|
|
|
|
with targetRect^ do
|
|
cairo_rectangle(pcr, Left + PixelOffset, Top + PixelOffset, Right - Left, Bottom - Top);
|
|
|
|
cairo_paint(pcr);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawImage1(targetRect: PRect; image: PGdkPixBuf;
|
|
sourceRect: PRect; mask: PGdkPixBuf; maskRect: PRect);
|
|
var
|
|
M: Tcairo_matrix_t;
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.DrawImage ');
|
|
{$ENDIF}
|
|
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
gdk_cairo_set_source_pixbuf(pcr, Image, 0, 0);
|
|
|
|
with targetRect^ do
|
|
cairo_rectangle(pcr, Left + PixelOffset, Top + PixelOffset, Right - Left, Bottom - Top);
|
|
|
|
cairo_matrix_init_identity(@M);
|
|
cairo_matrix_translate(@M, SourceRect^.Left, SourceRect^.Top);
|
|
cairo_matrix_scale(@M,
|
|
(sourceRect^.Right - sourceRect^.Left) / (targetRect^.Right - targetRect^.Left),
|
|
(sourceRect^.Bottom - sourceRect^.Top) / (targetRect^.Bottom - targetRect^.Top)
|
|
);
|
|
cairo_matrix_translate(@M, -targetRect^.Left, -targetRect^.Top);
|
|
|
|
cairo_pattern_set_matrix(cairo_get_source(pcr), @M);
|
|
|
|
cairo_clip(pcr);
|
|
cairo_paint(pcr);
|
|
cairo_reset_clip(pcr);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawPixmap(p: PPoint; pm: PGdkPixbuf; sr: PRect);
|
|
var
|
|
ASurface: Pcairo_surface_t;
|
|
AData: PByte;
|
|
begin
|
|
{$IFDEF VerboseGtk3DeviceContext}
|
|
DebugLn('TGtk3DeviceContext.DrawPixmap ');
|
|
{$ENDIF}
|
|
|
|
if pm = nil then
|
|
Exit;
|
|
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
AData := PByte(gdk_pixbuf_get_pixels(pm));
|
|
|
|
ASurface := cairo_image_surface_create_for_data(AData, CAIRO_FORMAT_ARGB32,
|
|
gdk_pixbuf_get_width(pm), gdk_pixbuf_get_height(pm), gdk_pixbuf_get_rowstride(pm));
|
|
|
|
if cairo_surface_status(ASurface) <> CAIRO_STATUS_SUCCESS then
|
|
begin
|
|
cairo_surface_destroy(ASurface);
|
|
DebugLn('TGtk3DeviceContext.drawPixmap() error. cairo_surface_status() returned bad status.');
|
|
Exit;
|
|
end;
|
|
|
|
cairo_set_source_surface(pcr, ASurface, p^.X, p^.Y);
|
|
|
|
cairo_paint(pcr);
|
|
cairo_surface_destroy(ASurface);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawPolyLine(P: PPoint; NumPts: Integer);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
ApplyPen;
|
|
cairo_move_to(pcr, P[0].X+PixelOffset, P[0].Y+PixelOffset);
|
|
for i := 1 to NumPts-1 do
|
|
cairo_line_to(pcr, P[i].X+PixelOffset, P[i].Y+PixelOffset);
|
|
cairo_stroke(pcr);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawPolygon(P: PPoint; NumPts: Integer;
|
|
FillRule: Integer; AFill, ABorder: Boolean);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
cairo_new_path(pcr);
|
|
|
|
cairo_move_to(pcr, P[0].X + PixelOffset, P[0].Y + PixelOffset);
|
|
|
|
for i := 1 to NumPts - 1 do
|
|
cairo_line_to(pcr, P[i].X + PixelOffset, P[i].Y + PixelOffset);
|
|
|
|
cairo_close_path(pcr);
|
|
|
|
if AFill then
|
|
begin
|
|
ApplyBrush;
|
|
cairo_set_fill_rule(pcr, Tcairo_fill_rule_t(FillRule));
|
|
cairo_fill_preserve(pcr);
|
|
end;
|
|
|
|
if ABorder then
|
|
begin
|
|
ApplyPen;
|
|
cairo_stroke(pcr);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.drawPolyBezier(P: PPoint; NumPoints: Integer; Filled, Continuous: Boolean);
|
|
var
|
|
MaxIndex, i: Integer;
|
|
bFill, bBorder: Boolean;
|
|
begin
|
|
// 3 points per curve + a starting point for the first curve
|
|
if NumPoints < 4 then
|
|
Exit;
|
|
|
|
bFill := CurrentBrush.Style <> BS_NULL;
|
|
bBorder := CurrentPen.Style <> psClear;
|
|
|
|
MaxIndex := NumPoints - 3 - Ord(not Continuous);
|
|
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
// Start drawing
|
|
i := 0;
|
|
while i <= MaxIndex do
|
|
begin
|
|
if i = 0 then
|
|
begin
|
|
cairo_move_to(pcr, P[i].X + PixelOffset, P[i].Y + PixelOffset);
|
|
Inc(i);
|
|
end
|
|
else if not Continuous then
|
|
begin
|
|
cairo_line_to(pcr, P[i].X + PixelOffset, P[i].Y + PixelOffset);
|
|
Inc(i);
|
|
end;
|
|
|
|
cairo_curve_to(pcr,
|
|
P[i].X + PixelOffset, P[i].Y + PixelOffset,
|
|
P[i + 1].X + PixelOffset, P[i + 1].Y + PixelOffset,
|
|
P[i + 2].X + PixelOffset, P[i + 2].Y + PixelOffset);
|
|
Inc(i, 3);
|
|
end;
|
|
|
|
if Filled then
|
|
begin
|
|
cairo_close_path(pcr);
|
|
if bFill then
|
|
begin
|
|
ApplyBrush;
|
|
cairo_fill_preserve(pcr);
|
|
end;
|
|
end;
|
|
|
|
if bBorder then
|
|
begin
|
|
ApplyPen;
|
|
cairo_stroke(pcr);
|
|
end
|
|
else
|
|
cairo_new_path(pcr); {clear path if no border is drawn}
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.eraseRect(ARect: PRect);
|
|
begin
|
|
// cairo_surface_
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.fillRect(ARect: PRect; ABrush: HBRUSH);
|
|
begin
|
|
with ARect^ do
|
|
fillRect(Left, Top, Right - Left, Bottom - Top, ABrush);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.fillRect(x, y, w, h: Integer; ABrush: HBRUSH);
|
|
var
|
|
ATempBrush: TGtk3Brush;
|
|
begin
|
|
{$ifdef VerboseGtk3DeviceContext}
|
|
//DebugLn('TGtk3DeviceContext.fillRect ',Format('x %d y %d w %d h %d',[x, y, w, h]));
|
|
{$endif}
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
if ABrush <> 0 then
|
|
begin
|
|
ATempBrush := FCurrentBrush;
|
|
CurrentBrush := TGtk3Brush(ABrush);
|
|
end
|
|
else
|
|
ATempBrush := FCurrentBrush;
|
|
|
|
applyBrush;
|
|
|
|
cairo_rectangle(pcr, x + PixelOffset, y + PixelOffset, w - 1, h - 1);
|
|
|
|
if (CurrentBrush.Style <> BS_NULL) then
|
|
begin
|
|
cairo_fill_preserve(pcr); // Preserve path for border
|
|
// Must paint border, filling is not enough
|
|
SetSourceColor(FCurrentBrush.Color);
|
|
cairo_set_line_width(pcr, 1);
|
|
cairo_stroke(pcr);
|
|
end;
|
|
CurrentBrush := ATempBrush;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.fillRect(x, y, w, h: Integer);
|
|
begin
|
|
fillRect(x, y, w, h , 0);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.FillAndStroke;
|
|
begin
|
|
if Assigned(FCurrentBrush) and (FCurrentBrush.Style <> BS_NULL) then
|
|
begin
|
|
ApplyBrush;
|
|
if Assigned(FCurrentPen) and (FCurrentPen.Style = psClear) then
|
|
cairo_fill(pcr)
|
|
else
|
|
cairo_fill_preserve(pcr);
|
|
end;
|
|
if Assigned(FCurrentPen) and (FCurrentPen.Style <> psClear) then
|
|
begin
|
|
ApplyPen;
|
|
cairo_stroke(pcr);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.EllipseArcPath(CX, CY, RX, RY: Double; Angle1, Angle2: Double; Clockwise, Continuous: Boolean);
|
|
begin
|
|
if (RX = 0) or (RY = 0) then // Avoid zero scale issues
|
|
Exit;
|
|
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
|
|
cairo_translate(pcr, SX(CX), SY(CY));
|
|
cairo_scale(pcr, SX2(RX), SY2(RY));
|
|
|
|
if not Continuous then
|
|
cairo_move_to(pcr, cos(Angle1), sin(Angle1));
|
|
|
|
if Clockwise then
|
|
cairo_arc(pcr, 0, 0, 1, Angle1, Angle2)
|
|
else
|
|
cairo_arc_negative(pcr, 0, 0, 1, Angle1, Angle2);
|
|
|
|
cairo_scale(pcr, 1 / SX2(RX), 1 / SY2(RY));
|
|
cairo_translate(pcr, -SX(CX), -SY(CY));
|
|
end;
|
|
|
|
function TGtk3DeviceContext.RoundRect(X1, Y1, X2, Y2: Integer; RX, RY: Integer): Boolean;
|
|
var
|
|
DX, DY,drx,dry: Double;
|
|
begin
|
|
Result := False;
|
|
cairo_surface_get_device_offset(cairo_get_target(pcr), @DX, @DY);
|
|
DX := DX+PixelOffset;
|
|
DY := DY+PixelOffset;
|
|
drx:=rx/2;
|
|
dry:=ry/2;
|
|
cairo_translate(pcr, DX, DY);
|
|
try
|
|
cairo_new_path(pcr);
|
|
cairo_move_to(pcr, X1+dRX, Y1);
|
|
cairo_line_to(pcr, X2-dRX-PixelOffset, Y1);
|
|
EllipseArcPath(X2-dRX-PixelOffset, Y1+dRY, dRX, dRY, -PI/2, 0, True, True);
|
|
cairo_line_to(pcr, X2-PixelOffset, Y2-dRY-PixelOffset);
|
|
EllipseArcPath(X2-dRX-PixelOffset, Y2-dRY-PixelOffset, dRX, dRY, 0, PI/2, True, True);
|
|
cairo_line_to(pcr, X1+dRX, Y2-PixelOffset);
|
|
EllipseArcPath(X1+dRX, Y2-dRY-PixelOffset, dRX, dRY, PI/2, PI, True, True);
|
|
cairo_line_to(pcr, X1, Y1+dRX);
|
|
EllipseArcPath(X1+dRX, Y1+dRY, dRX, dRY, PI, PI*1.5, True, True);
|
|
cairo_close_path(pcr);
|
|
FillAndStroke;
|
|
Result := True;
|
|
finally
|
|
cairo_translate(pcr, -DX, -DY);
|
|
end;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.drawFrameControl(arect:TRect;uType,uState:cardinal):boolean;
|
|
var
|
|
Context: PGtkStyleContext;
|
|
pw:PGtkWidget;
|
|
path:PGtkwIdgetPath;
|
|
w:PgtkWidget;
|
|
begin
|
|
|
|
Result:=false;
|
|
|
|
w:=nil;
|
|
|
|
case uType of
|
|
DFC_BUTTON,DFC_CAPTION:
|
|
w:=GetStyleWidget(lgsButton);
|
|
DFC_MENU:
|
|
w:=GetStyleWidget(lgsMenu);
|
|
else
|
|
w:=GetStyleWidget(lgsDefault);
|
|
end;
|
|
|
|
if not Assigned(w) then exit;
|
|
|
|
Context:=w^.get_style_context;
|
|
path:=w^.get_path;
|
|
//it is wrong to call gtk_style_context_set_path here. Zeljan.
|
|
//https://docs.gtk.org/gtk3/method.StyleContext.set_path.html
|
|
//gtk_style_context_set_path (context, path);
|
|
gtk_style_context_set_state(context,(* gtk_widget_path_iter_get_state (path, -1)*) [TGtkStateFlagsIdxMinValue..TGtkStateFlagsIdxMaxValue]);
|
|
gtk_style_context_set_state(context, [GTK_STATE_FLAG_FOCUSED, GTK_STATE_FLAG_PRELIGHT]);
|
|
|
|
pw:=w;
|
|
while Assigned(pw) do
|
|
begin
|
|
|
|
Context:=pw^.get_style_context;
|
|
path:=pw^.get_path;
|
|
with aRect do
|
|
begin
|
|
gtk_render_background(Context,pcr, Left, Top, Right - Left, Bottom - Top);
|
|
gtk_render_frame(Context,pcr, Left, Top, Right - Left, Bottom - Top);
|
|
end;
|
|
pw:=pw^.parent;
|
|
end;
|
|
|
|
Result := True;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.drawFocusRect(const aRect: TRect): boolean;
|
|
var
|
|
Context: PGtkStyleContext;
|
|
UnRefContext: boolean;
|
|
begin
|
|
Result := False;
|
|
UnRefContext := False;
|
|
if Parent <> nil then
|
|
Context := Parent^.get_style_context
|
|
else
|
|
begin
|
|
UnRefContext := True;
|
|
Context:=TGtkStyleContext.new();
|
|
Context^.add_class('button');
|
|
end;
|
|
if Context = nil then
|
|
begin
|
|
DebugLn('WARNING: TGtk3WidgetSet.DrawFocusRect drawing focus on non widget context isn''t implemented.');
|
|
exit;
|
|
end;
|
|
with aRect do
|
|
gtk_render_focus(Context ,pcr, Left, Top, Right - Left, Bottom - Top);
|
|
if UnRefContext then
|
|
Context^.unref;
|
|
Result := True;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.getBpp: integer;
|
|
var
|
|
AVisual: PGdkVisual;
|
|
begin
|
|
if (Parent <> nil) and (Parent^.get_has_window) then
|
|
begin
|
|
AVisual := gdk_window_get_visual(Parent^.get_window);
|
|
Result := gdk_visual_get_bits_per_rgb(AVisual);
|
|
end else
|
|
if (ParentPixmap <> nil) and (Parent = nil) then
|
|
begin
|
|
Result := ParentPixmap^.get_bits_per_sample;
|
|
end else
|
|
begin
|
|
AVisual := gdk_window_get_visual(gdk_get_default_root_window);
|
|
Result := gdk_visual_get_bits_per_rgb(AVisual);
|
|
end;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.getDepth: integer;
|
|
var
|
|
AVisual: PGdkVisual;
|
|
begin
|
|
Result := 0;
|
|
if (Parent <> nil) and Gtk3IsGdkWindow(Parent^.get_window) then
|
|
begin
|
|
AVisual := gdk_window_get_visual(Parent^.get_window);
|
|
if Gtk3IsGdkVisual(AVisual) then
|
|
begin
|
|
Result := gdk_visual_get_depth(AVisual);
|
|
exit;
|
|
end;
|
|
end;
|
|
AVisual := gdk_window_get_visual(gdk_get_default_root_window);
|
|
if Gtk3IsGdkVisual(AVisual) then
|
|
Result := gdk_visual_get_depth(AVisual);
|
|
end;
|
|
|
|
function TGtk3DeviceContext.getDeviceSize: TPoint;
|
|
begin
|
|
Result := Point(0 , 0);
|
|
if Parent <> nil then
|
|
begin
|
|
Result.y := Parent^.get_allocated_height;
|
|
Result.x := Parent^.get_allocated_width;
|
|
end else
|
|
if ParentPixmap <> nil then
|
|
begin
|
|
Result.y := ParentPixmap^.height;
|
|
Result.x := ParentPixmap^.width;
|
|
end else
|
|
if Gtk3IsGdkWindow(Window) then
|
|
begin
|
|
Result.X := Window^.get_width;
|
|
Result.y := Window^.get_height;
|
|
end;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.LineTo(X, Y: Integer): Boolean;
|
|
var
|
|
X0, Y0,dx,dy:integer;
|
|
begin
|
|
if not Assigned(pcr) then
|
|
exit(False);
|
|
cairo_set_operator(pcr, CAIRO_OPERATOR_OVER);
|
|
ApplyPen;
|
|
|
|
|
|
if fCurrentPen.Width<=1 then // optimizations
|
|
begin
|
|
//cairo_get_current_point(pcr, @FX, @FY);
|
|
X0:=round(FLastPenX);
|
|
Y0:=round(FLastPenY);
|
|
dx:=X-X0;
|
|
dy:=Y-Y0;
|
|
|
|
if (dx=0) and (dy=0) then exit;
|
|
|
|
if (dx=0) then
|
|
begin
|
|
if dy>0 then
|
|
begin
|
|
cairo_move_to(pcr,X+PixelOffset,Y0);
|
|
cairo_line_to(pcr,X+PixelOffset,Y);
|
|
end else
|
|
begin
|
|
cairo_move_to(pcr,X+PixelOffset,Y0+1);
|
|
cairo_line_to(pcr,X+PixelOffset,Y+1);
|
|
end;
|
|
end else
|
|
if (dy=0) then
|
|
begin
|
|
if dx>0 then
|
|
begin
|
|
cairo_move_to(pcr,X0,Y+PixelOffset);
|
|
cairo_line_to(pcr,X,Y+PixelOffset);
|
|
end else
|
|
begin
|
|
cairo_move_to(pcr,X0+1,Y+PixelOffset);
|
|
cairo_line_to(pcr,X+1,Y+PixelOffset);
|
|
end;
|
|
end else
|
|
if abs(dx)=abs(dy) then
|
|
begin
|
|
// here is required more Cairo magic
|
|
if (dx>0) and (dy>0) then
|
|
begin
|
|
cairo_move_to(pcr,FLastPenX-PixelOffset,FLastPenY-PixelOffset);
|
|
cairo_line_to(pcr,X, Y);
|
|
end else
|
|
if (dx>0) and (dy<0) then
|
|
begin
|
|
cairo_move_to(pcr,FLastPenX+PixelOffset,FLastPenY+PixelOffset);
|
|
cairo_line_to(pcr,X+PixelOffset, Y+PixelOffset);
|
|
end else
|
|
if (dx<0) and (dy>0) then
|
|
begin
|
|
cairo_move_to(pcr,FLastPenX+1+PixelOffset,FLastPenY-PixelOffset);
|
|
cairo_line_to(pcr,X+1, Y);
|
|
end else {dx<0 and dy<0}
|
|
begin
|
|
cairo_move_to(pcr,FLastPenX+PixelOffset,FLastPenY+PixelOffset);
|
|
cairo_line_to(pcr,X+PixelOffset, Y+PixelOffset);
|
|
end;
|
|
end else
|
|
cairo_line_to(pcr,X+PixelOffset, Y+PixelOffset);
|
|
|
|
end else
|
|
cairo_line_to(pcr,X+PixelOffset, Y+PixelOffset);
|
|
|
|
cairo_stroke(pcr);
|
|
Result := True;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.MoveTo(const X, Y: Integer; OldPoint: PPoint): Boolean;
|
|
var
|
|
dx: Double;
|
|
dy: Double;
|
|
begin
|
|
if not Assigned(pcr) then
|
|
exit(False);
|
|
if OldPoint <> nil then
|
|
begin
|
|
cairo_get_current_point(pcr, @dx, @dy);
|
|
OldPoint^.X := Round(dx);
|
|
OldPoint^.Y := Round(dy);
|
|
end;
|
|
dx := X;
|
|
dy := Y;
|
|
if CurrentPen.Width > 1 then
|
|
begin
|
|
dx := X + PixelOffset;
|
|
dy := Y + PixelOffset;
|
|
end;
|
|
cairo_move_to(pcr, dx, dy);
|
|
FLastPenX := dx;
|
|
FLastPenY := dy;
|
|
//TODO: check if we need here cairo_get_current_point or we can assign it like above
|
|
//cairo_get_current_point(pcr,@FLastPenX,@FLastPenY);
|
|
Result := True;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.SetClipRegion(ARgn: TGtk3Region): Integer;
|
|
begin
|
|
Result := SimpleRegion;
|
|
if Assigned(pcr) then
|
|
begin
|
|
cairo_reset_clip(pcr);
|
|
gdk_cairo_region(pcr, ARgn.FHandle);
|
|
cairo_clip(pcr);
|
|
end;
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetSourceColor(AColor: TColor);
|
|
var
|
|
R, G, B: double;
|
|
begin
|
|
TColorToRGB(AColor, R, G, B);
|
|
cairo_set_source_rgb(pcr, R, G, B);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetImage(AImage: TGtk3Image);
|
|
var
|
|
APixBuf: PGdkPixbuf;
|
|
begin
|
|
FCurrentImage := AImage;
|
|
cairo_destroy(pcr);
|
|
APixBuf := AImage.Handle;
|
|
if not Gtk3IsGdkPixbuf(APixBuf) then
|
|
begin
|
|
DebugLn('ERROR: TGtk3DeviceContext.SetImage image handle isn''t PGdkPixbuf.');
|
|
exit;
|
|
end;
|
|
(*
|
|
DebugLn('TGtk3DeviceContext.SetImage w=',dbgs(APixBuf^.width),' h=',dbgs(APixBuf^.height),
|
|
' RowStride ',dbgs(APixBuf^.rowstride),' BPS=',dbgs(APixBuf^.get_bits_per_sample),
|
|
' BLEN ',dbgs(APixbuf^.get_byte_length),' channels ',dbgs(APixBuf^.get_n_channels),
|
|
' ALPHA ',dbgs(APixbuf^.get_has_alpha));
|
|
*)
|
|
if FOwnsSurface and (CairoSurface <> nil) then
|
|
cairo_surface_destroy(CairoSurface);
|
|
|
|
CairoSurface := cairo_image_surface_create_for_data(APixBuf^.pixels,
|
|
AImage.Format,
|
|
APixBuf^.get_width,
|
|
APixBuf^.get_height,
|
|
APixBuf^.rowstride);
|
|
FCairo := cairo_create(CairoSurface);
|
|
FOwnsSurface := true;
|
|
end;
|
|
|
|
function TGtk3DeviceContext.ResetClip: Integer;
|
|
begin
|
|
Result := NullRegion;
|
|
if Assigned(pcr) then
|
|
cairo_reset_clip(pcr);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.TranslateCairoToDevice;
|
|
var
|
|
Pt: TPoint;
|
|
begin
|
|
Pt := Offset;
|
|
Translate(Pt);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.Translate(APoint: TPoint);
|
|
begin
|
|
cairo_translate(pcr, APoint.X, APoint.Y);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.set_antialiasing(aamode: boolean);
|
|
const
|
|
caa:array[boolean] of Tcairo_antialias_t = (CAIRO_ANTIALIAS_NONE,CAIRO_ANTIALIAS_DEFAULT);
|
|
begin
|
|
cairo_set_antialias(pcr, caa[aamode]);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.Save;
|
|
begin
|
|
cairo_save(pcr);
|
|
inc(FDCSaveCounter);
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.Restore;
|
|
begin
|
|
dec(FDCSaveCounter);
|
|
cairo_restore(pcr);
|
|
if FDCSaveCounter < 0 then
|
|
DebugLn('WARNING: TGtk3DeviceContext: Cairo restore called without save.');
|
|
end;
|
|
|
|
procedure TGtk3DeviceContext.SetCanvasScaleFactor(const AValue: double);
|
|
var
|
|
matrix: Tcairo_matrix_t;
|
|
begin
|
|
if FCanvasScaleFactor <> AValue then
|
|
begin
|
|
FCanvasScaleFactor := AValue;
|
|
cairo_get_matrix(pcr, @matrix);
|
|
cairo_matrix_scale(@matrix, FCanvasScaleFactor, FCanvasScaleFactor);
|
|
cairo_set_matrix(pcr, @matrix);
|
|
end;
|
|
end;
|
|
|
|
//various routines for text
|
|
|
|
// LCL denotes accelerator keys by '&', while Gtk denotes them with '_'.
|
|
// LCL allows to show literal '&' by escaping (doubling) it.
|
|
// Gtk allows to show literal '_' by escaping (doubling) it.
|
|
// As the situation for LCL and Gtk is symmetric, conversion is made by generic function.
|
|
function TransformAmpersandsAndUnderscores(const S: string; const FromChar, ToChar: char): string;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := '';
|
|
i := 1;
|
|
while i <= Length(S) do
|
|
begin
|
|
if S[i] = ToChar then
|
|
Result := Result + ToChar + ToChar
|
|
else
|
|
if S[i] = FromChar then
|
|
if i < Length(S) then
|
|
begin
|
|
if S[i + 1] = FromChar then
|
|
begin
|
|
Result := Result + FromChar;
|
|
inc(i);
|
|
end
|
|
else
|
|
Result := Result + ToChar;
|
|
end
|
|
else
|
|
Result := Result + FromChar
|
|
else
|
|
Result := Result + S[i];
|
|
inc(i);
|
|
end;
|
|
end;
|
|
|
|
function ReplaceAmpersandsWithUnderscores(const S: string): string; inline;
|
|
begin
|
|
Result := TransformAmpersandsAndUnderscores(S, '&', '_');
|
|
end;
|
|
|
|
function ReplaceUnderscoresWithAmpersands(const S: string): string; inline;
|
|
begin
|
|
Result := TransformAmpersandsAndUnderscores(S, '_', '&');
|
|
end;
|
|
|
|
{-------------------------------------------------------------------------------
|
|
function GetTextExtentIgnoringAmpersands(TheFont: PGDKFont;
|
|
Str : PChar; StrLength: integer;
|
|
MaxWidth: Longint; lbearing, rbearing, width, ascent, descent : Pgint);
|
|
|
|
Gets text extent of a string, ignoring escaped Ampersands.
|
|
That means, ampersands are not counted.
|
|
Negative MaxWidth means no limit.
|
|
-------------------------------------------------------------------------------}
|
|
procedure GetTextExtentIgnoringAmpersands(TheFont: TGtk3Font;
|
|
Str : PChar; StrLength: integer;
|
|
lbearing, rbearing, width, ascent, descent : Pgint);
|
|
var
|
|
NewStr : PChar;
|
|
AMetrics: PPangoFontMetrics;
|
|
{ACharWidth,}ATextWidth,ATextHeight: gint;
|
|
begin
|
|
if lbearing<>nil then
|
|
lbearing^:=0;
|
|
if rbearing<>nil then
|
|
rbearing^:=0;
|
|
// check if Str contains an ampersand before removing them all.
|
|
if StrLScan(Str, '&', StrLength) <> nil then
|
|
NewStr := RemoveAmpersands(Str, StrLength)
|
|
else
|
|
NewStr := Str;
|
|
TheFont.Layout^.set_text(NewStr, StrLength);
|
|
// TheFont.Layout^.get_extents(@AInkRect, @ALogicalRect);
|
|
|
|
AMetrics := pango_context_get_metrics(TheFont.Layout^.get_context, TheFont.Handle, TheFont.Layout^.get_context^.get_language);
|
|
// if not Gtk3IsPangoFontMetrics(PGObject(AMetrics)) then
|
|
// exit;
|
|
if AMetrics = nil then
|
|
begin
|
|
Debugln('WARNING: GetTextExtentIgnoringAmpersands AMetrics=nil');
|
|
exit;
|
|
end;
|
|
|
|
if ascent <> nil then
|
|
ascent^ := AMetrics^.get_ascent;
|
|
if descent <> nil then
|
|
descent^ := AMetrics^.get_descent;
|
|
if width <> nil then
|
|
begin
|
|
{ACharWidth := AMetrics^.get_approximate_char_width;
|
|
width^ := (utf8length(Str, StrLength) * ACharWidth) div PANGO_SCALE;}
|
|
TheFont.Layout^.get_pixel_size(@ATextWidth, @ATextHeight);
|
|
width^:=ATextWidth;
|
|
end;
|
|
// PANGO_PIXELS(char_width)
|
|
|
|
// lBearing^ := 0;
|
|
// rBearing^ := 0;
|
|
// gdk_text_extents(TheFont, NewStr, StrLength,
|
|
// lbearing, rBearing, width, ascent, descent);
|
|
if NewStr <> Str then
|
|
StrDispose(NewStr);
|
|
AMetrics^.unref;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
procedure Gtk3WordWrap(DC: HDC; AText: PChar; MaxWidthInPixel: integer;
|
|
var Lines: PPChar; var LineCount: integer); virtual;
|
|
|
|
Breaks AText into several lines and creates a list of PChar. The last entry
|
|
will be nil.
|
|
Lines break at new line chars and at spaces if a line is longer than
|
|
MaxWidthInPixel or in a word.
|
|
Lines will be one memory block so that you can free the list and all lines
|
|
with FreeMem(Lines).
|
|
------------------------------------------------------------------------------}
|
|
procedure Gtk3WordWrap(DC: HDC; AText: PChar;
|
|
MaxWidthInPixel: integer; out Lines: PPChar; out LineCount: integer);
|
|
var
|
|
UseFont: TGtk3Font;
|
|
|
|
function GetLineWidthInPixel(LineStart, LineLen: integer): integer;
|
|
var
|
|
width: LongInt;
|
|
begin
|
|
GetTextExtentIgnoringAmpersands(UseFont, @AText[LineStart], LineLen,
|
|
nil, nil, @width, nil, nil);
|
|
Result := Width;
|
|
end;
|
|
|
|
function FindLineEnd(LineStart: integer): integer;
|
|
var
|
|
CharLen,
|
|
LineStop,
|
|
LineWidth, WordWidth, WordEnd, CharWidth: integer;
|
|
begin
|
|
// first search line break or text break
|
|
Result:=LineStart;
|
|
while not (AText[Result] in [#0,#10,#13]) do inc(Result);
|
|
if Result<=LineStart+1 then exit;
|
|
lineStop:=Result;
|
|
|
|
// get current line width in pixel
|
|
LineWidth:=GetLineWidthInPixel(LineStart,Result-LineStart);
|
|
if LineWidth>MaxWidthInPixel then
|
|
begin
|
|
// line too long
|
|
// -> add words till line size reached
|
|
LineWidth:=0;
|
|
WordEnd:=LineStart;
|
|
WordWidth:=0;
|
|
repeat
|
|
Result:=WordEnd;
|
|
inc(LineWidth,WordWidth);
|
|
// find word start
|
|
while AText[WordEnd] in [' ',#9] do inc(WordEnd);
|
|
// find word end
|
|
while not (AText[WordEnd] in [#0,' ',#9,#10,#13]) do inc(WordEnd);
|
|
// calculate word width
|
|
WordWidth:=GetLineWidthInPixel(Result,WordEnd-Result);
|
|
until LineWidth+WordWidth>MaxWidthInPixel;
|
|
if LineWidth=0 then
|
|
begin
|
|
// the first word is longer than the maximum width
|
|
// -> add chars till line size reached
|
|
Result:=LineStart;
|
|
LineWidth:=0;
|
|
repeat
|
|
charLen:=UTF8CodepointSize(@AText[result]);
|
|
CharWidth:=GetLineWidthInPixel(Result,charLen);
|
|
inc(LineWidth,CharWidth);
|
|
if LineWidth>MaxWidthInPixel then break;
|
|
if result>=lineStop then break;
|
|
inc(Result,charLen);
|
|
until false;
|
|
// at least one char
|
|
if Result=LineStart then begin
|
|
charLen:=UTF8CodepointSize(@AText[result]);
|
|
inc(Result,charLen);
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function IsEmptyText: boolean;
|
|
begin
|
|
if (AText=nil) or (AText[0]=#0) then
|
|
begin
|
|
// no text
|
|
GetMem(Lines,SizeOf(PChar));
|
|
Lines[0]:=nil;
|
|
LineCount:=0;
|
|
Result:=true;
|
|
end else
|
|
Result:=false;
|
|
end;
|
|
|
|
procedure InitFont;
|
|
begin
|
|
UseFont := TGtk3DeviceContext(DC).CurrentFont;
|
|
end;
|
|
|
|
var
|
|
LinesList: TIntegerList;
|
|
LineStart, LineEnd, LineLen: integer;
|
|
ArraySize, TotalSize: integer;
|
|
i: integer;
|
|
CurLineEntry: PPChar;
|
|
CurLineStart: PChar;
|
|
begin
|
|
if IsEmptyText then
|
|
begin
|
|
Lines:=nil;
|
|
LineCount:=0;
|
|
exit;
|
|
end;
|
|
InitFont;
|
|
LinesList:=TIntegerList.Create;
|
|
LineStart:=0;
|
|
|
|
// find all line starts and line ends
|
|
repeat
|
|
LinesList.Add(LineStart);
|
|
// find line end
|
|
LineEnd:=FindLineEnd(LineStart);
|
|
LinesList.Add(LineEnd);
|
|
// find next line start
|
|
LineStart:=LineEnd;
|
|
if AText[LineStart] in [#10,#13] then
|
|
begin
|
|
// skip new line chars
|
|
inc(LineStart);
|
|
if (AText[LineStart] in [#10,#13])
|
|
and (AText[LineStart]<>AText[LineStart-1]) then
|
|
inc(LineStart);
|
|
end else
|
|
if AText[LineStart] in [' ',#9] then
|
|
begin
|
|
// skip space
|
|
while AText[LineStart] in [' ',#9] do
|
|
inc(LineStart);
|
|
end;
|
|
until AText[LineStart]=#0;
|
|
|
|
// create mem block for 'Lines': array of PChar + all lines
|
|
LineCount:=LinesList.Count shr 1;
|
|
ArraySize:=(LineCount+1)*SizeOf(PChar);
|
|
TotalSize:=ArraySize;
|
|
i:=0;
|
|
while i<LinesList.Count do
|
|
begin
|
|
// add LineEnd - LineStart + 1 for the #0
|
|
LineLen:=LinesList[i+1]-LinesList[i]+1;
|
|
inc(TotalSize,LineLen);
|
|
inc(i,2);
|
|
end;
|
|
GetMem(Lines,TotalSize);
|
|
FillChar(Lines^,TotalSize,0);
|
|
|
|
// create Lines
|
|
CurLineEntry:=Lines;
|
|
CurLineStart:=PChar(CurLineEntry)+ArraySize;
|
|
i:=0;
|
|
while i<LinesList.Count do
|
|
begin
|
|
// set the pointer to the start of the current line
|
|
CurLineEntry[i shr 1]:=CurLineStart;
|
|
// copy the line
|
|
LineStart:=LinesList[i];
|
|
LineEnd:=LinesList[i+1];
|
|
LineLen:=LineEnd-LineStart;
|
|
if LineLen>0 then
|
|
Move(AText[LineStart],CurLineStart^,LineLen);
|
|
inc(CurLineStart,LineLen);
|
|
// add #0 as line end
|
|
CurLineStart^:=#0;
|
|
inc(CurLineStart);
|
|
// next line
|
|
inc(i,2);
|
|
end;
|
|
if {%H-}PtrUInt(CurLineStart)-{%H-}PtrUInt(Lines)<>TotalSize then
|
|
RaiseGDBException('Gtk3WordWrap Consistency Error:'
|
|
+' Lines+TotalSize<>CurLineStart');
|
|
CurLineEntry[i shr 1]:=nil;
|
|
|
|
LinesList.Free;
|
|
end;
|
|
|
|
end.
|