mirror of
				https://gitlab.com/freepascal.org/lazarus/lazarus.git
				synced 2025-11-04 04:22:21 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1593 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			1593 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
unit CairoCanvas;
 | 
						|
 | 
						|
{$mode objfpc}{$H+}
 | 
						|
 | 
						|
{$if (FPC_FULLVERSION>=20701)}
 | 
						|
{$Packset 1}
 | 
						|
{$endif}
 | 
						|
 | 
						|
{$define pangocairo}
 | 
						|
{-$define breaklines}   // disabled as it's not UTF-8 safe
 | 
						|
{-$define DebugClip}
 | 
						|
 | 
						|
interface
 | 
						|
 | 
						|
uses
 | 
						|
  Types, SysUtils, Classes, LCLType, LCLProc, Graphics, math, GraphMath,
 | 
						|
  Printers, Cairo
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  ,Pango, PangoCairo, GLib2
 | 
						|
  {$endif}
 | 
						|
  ;
 | 
						|
 | 
						|
type
 | 
						|
  TSquaredCorners = set of (scTopLeft,scBottomLeft,scBottomRight,scTopRight);
 | 
						|
 | 
						|
  { TCairoPrinterCanvas }
 | 
						|
 | 
						|
  TCairoPrinterCanvas = class(TFilePrinterCanvas)
 | 
						|
  strict private
 | 
						|
    cr: Pcairo_t;
 | 
						|
  private
 | 
						|
    FLazClipRect: TRect;
 | 
						|
    FUserClipRect: Pcairo_rectangle_t;
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    fFontDesc: PPangoFontDescription;
 | 
						|
    fFontDescStr: string;
 | 
						|
    function StylesToStr(Styles: TFontStyles):string;
 | 
						|
    procedure UpdatePangoLayout(Layout: PPangoLayout);
 | 
						|
    {$endif}
 | 
						|
    procedure SelectFontEx(AStyle: TFontStyles; const AName: string;
 | 
						|
      ASize: double; aPitch: TFontPitch);
 | 
						|
    function SX(x: double): double;
 | 
						|
    function SY(y: double): double;
 | 
						|
    function SX2(x: double): double;
 | 
						|
    function SY2(y: double): double;
 | 
						|
    procedure SetSourceColor(Color: TColor);
 | 
						|
    procedure SetPenProperties;
 | 
						|
    procedure SetBrushProperties;
 | 
						|
    procedure SelectFont;
 | 
						|
    procedure PolylinePath(Points: PPoint; NumPts: Integer);
 | 
						|
    procedure EllipseArcPath(CX, CY, RX, RY: Double; Angle1, Angle2: Double;
 | 
						|
      Clockwise, Continuous: Boolean);
 | 
						|
    procedure ArcPath(ALeft, ATop, ARight, ABottom, Angle16Deg, Angle16DegLength: double);
 | 
						|
    procedure ArcPath(ALeft, ATop, ARight, ABottom, StX, StY, EX, EY: double);
 | 
						|
    procedure FillAndStroke;
 | 
						|
    procedure FillOnly;
 | 
						|
    procedure StrokeOnly;
 | 
						|
    procedure TColorToRGB(Color: TColor; out R,G,B: double);
 | 
						|
    // debug tools
 | 
						|
    procedure DrawPoint(x,y: double; color: TColor);
 | 
						|
    procedure DrawRefRect(x,y,awidth,aheight: double; color:TColor);
 | 
						|
    procedure DebugSys;
 | 
						|
  protected
 | 
						|
    ScaleX, ScaleY, FontScale: Double;
 | 
						|
    procedure SetLazClipRect(r: TRect);
 | 
						|
    procedure DoLineTo(X1,Y1: Integer); override;
 | 
						|
    procedure DoMoveTo({%H-}x, {%H-}y: integer); override;
 | 
						|
 | 
						|
    function CreateCairoHandle: HDC; virtual; abstract;
 | 
						|
    procedure DestroyCairoHandle; virtual;
 | 
						|
    procedure SetHandle(NewHandle: HDC); override;
 | 
						|
    function GetClipRect: TRect; override;
 | 
						|
    procedure SetClipRect(const ARect: TRect); override;
 | 
						|
    function GetClipping: Boolean; override;
 | 
						|
    procedure SetClipping(const AValue: boolean); override;
 | 
						|
    //
 | 
						|
    procedure CreateBrush; override;
 | 
						|
    procedure CreateFont; override;
 | 
						|
    procedure CreateHandle; override;
 | 
						|
    procedure CreatePen; override;
 | 
						|
    procedure CreateRegion; override;
 | 
						|
    procedure RealizeAntialiasing; override;
 | 
						|
    procedure DestroyHandle;
 | 
						|
  public
 | 
						|
    SurfaceXDPI, SurfaceYDPI: Integer;
 | 
						|
    constructor Create(APrinter : TPrinter); override;
 | 
						|
    constructor Create; overload;
 | 
						|
    destructor Destroy; override;
 | 
						|
    procedure BeginDoc; override;
 | 
						|
    procedure EndDoc; override;
 | 
						|
    procedure NewPage; override;
 | 
						|
    procedure FillRect(const ARect: TRect); override;
 | 
						|
    procedure Rectangle(X1,Y1,X2,Y2: Integer); override;
 | 
						|
    procedure Polyline(Points: PPoint; NumPts: Integer); override;
 | 
						|
    procedure Polygon(Points: PPoint; NumPts: Integer; {%H-}Winding: boolean = False); override;
 | 
						|
    procedure FrameRect(const ARect: TRect); override;
 | 
						|
    procedure Frame(const ARect: TRect); override;
 | 
						|
    procedure RoundRect(X1, Y1, X2, Y2: Integer; RX, RY: Integer); override;
 | 
						|
    procedure Ellipse(X1, Y1, X2, Y2: Integer); override;
 | 
						|
    procedure Arc(ALeft, ATop, ARight, ABottom, Angle16Deg, Angle16DegLength: Integer); override;
 | 
						|
    procedure Arc(ALeft, ATop, ARight, ABottom, StX, StY, EX, EY: Integer); override;
 | 
						|
    procedure Chord(X1, Y1, X2, Y2, Angle1, Angle2: Integer); override;
 | 
						|
    procedure Chord(X1, Y1, X2, Y2, StX, StY, EX, EY: Integer); override;
 | 
						|
    procedure Pie(EllipseX1, EllipseY1, EllipseX2, EllipseY2, StartX, StartY, EndX, EndY: Integer); override;
 | 
						|
    procedure RadialPie(Left, Top, Right, Bottom, Angle1, Angle2: Integer); override;
 | 
						|
    procedure PolyBezier(Points: PPoint; NumPts: Integer; Filled: boolean = False; Continuous: boolean = False); override;
 | 
						|
    procedure TextOut(X,Y: Integer; const Text: String); override;
 | 
						|
    procedure TextRect(ARect: TRect; X1, Y1: integer; const Text: string; const Style: TTextStyle); override;
 | 
						|
    function TextExtent(const Text: string): TSize; override;
 | 
						|
    function GetTextMetrics(out M: TLCLTextMetric): boolean; override;
 | 
						|
    procedure StretchDraw(const DestRect: TRect; SrcGraphic: TGraphic); override;
 | 
						|
    procedure SetPixel(X,Y: Integer; Value: TColor); override;
 | 
						|
  public
 | 
						|
    procedure MixedRoundRect(X1, Y1, X2, Y2: Integer; RX, RY: Integer; SquaredCorners: TSquaredCorners);
 | 
						|
    procedure DrawSurface(const SourceRect, DestRect: TRect; surface: Pcairo_surface_t);
 | 
						|
{  Not implemented
 | 
						|
    procedure FloodFill(X, Y: Integer; FillColor: TColor; FillStyle: TFillStyle); override;
 | 
						|
    procedure CopyRect(const Dest: TRect; SrcCanvas: TCanvas; const Source: TRect); override;
 | 
						|
    procedure Frame3d(var ARect: TRect; const FrameWidth: integer; const Style: TGraphicsBevelCut); override;}
 | 
						|
  end;
 | 
						|
 | 
						|
  { TCairoFileCanvas }
 | 
						|
 | 
						|
  TCairoFileCanvas = class (TCairoPrinterCanvas)
 | 
						|
  protected
 | 
						|
    sf: Pcairo_surface_t;
 | 
						|
    fStream: TStream;
 | 
						|
    procedure DestroyCairoHandle; override;
 | 
						|
  public
 | 
						|
    procedure UpdatePageSize; virtual;
 | 
						|
    property Stream: TStream read fStream write fStream;
 | 
						|
  end;
 | 
						|
 | 
						|
  { TCairoPdfCanvas }
 | 
						|
 | 
						|
  TCairoPdfCanvas = class(TCairoFileCanvas)
 | 
						|
  protected
 | 
						|
    function CreateCairoHandle: HDC; override;
 | 
						|
  public
 | 
						|
    procedure UpdatePageSize; override;
 | 
						|
  end;
 | 
						|
 | 
						|
  { TCairoSvgCanvas }
 | 
						|
 | 
						|
  TCairoSvgCanvas = class(TCairoFileCanvas)
 | 
						|
  protected
 | 
						|
    function CreateCairoHandle: HDC; override;
 | 
						|
  end;
 | 
						|
 | 
						|
  { TCairoPngCanvas }
 | 
						|
 | 
						|
  TCairoPngCanvas = class(TCairoFileCanvas)
 | 
						|
  protected
 | 
						|
    function CreateCairoHandle: HDC; override;
 | 
						|
    procedure DestroyCairoHandle; override;
 | 
						|
  public
 | 
						|
    constructor Create(APrinter: TPrinter); override;
 | 
						|
  end;
 | 
						|
 | 
						|
  { TCairoPsCanvas }
 | 
						|
 | 
						|
  TCairoPsCanvas = class(TCairoFileCanvas)
 | 
						|
  protected
 | 
						|
    function CreateCairoHandle: HDC; override;
 | 
						|
  public
 | 
						|
    procedure UpdatePageSize; override;
 | 
						|
  end;
 | 
						|
 | 
						|
  function GraphicToARGB32(Source: TGraphic; buf: PByte): Boolean;
 | 
						|
 | 
						|
implementation
 | 
						|
 | 
						|
uses
 | 
						|
  IntfGraphics, GraphType, FPimage;
 | 
						|
 | 
						|
const
 | 
						|
  Dash_Dash:        array [0..1] of double = (18, 6);             //____ ____
 | 
						|
  Dash_Dot:         array [0..1] of double = (3, 3);              //.........
 | 
						|
  Dash_DashDot:     array [0..3] of double = (9, 6, 3, 6);        //__ . __ .
 | 
						|
  Dash_DashDotDot:  array [0..5] of double = (9, 3, 3, 3, 3, 3);  //__ . . __
 | 
						|
 | 
						|
function WriteToStream(closure: Pointer; data: PByte; length: LongWord): cairo_status_t; cdecl;
 | 
						|
var
 | 
						|
  Stream: TStream absolute closure;
 | 
						|
begin
 | 
						|
  if Stream.Write(data^, Length) = int64(Length) then
 | 
						|
    result := CAIRO_STATUS_SUCCESS
 | 
						|
  else
 | 
						|
    result := CAIRO_STATUS_WRITE_ERROR;
 | 
						|
end;
 | 
						|
 | 
						|
function GraphicToARGB32(Source: TGraphic; buf: PByte): Boolean;
 | 
						|
var
 | 
						|
  p: PDWord;
 | 
						|
  x, y: Integer;
 | 
						|
  c: TFPColor;
 | 
						|
  Img: TLazIntfImage;
 | 
						|
begin
 | 
						|
  Img := TRasterImage(Source).CreateIntfImage;
 | 
						|
  try
 | 
						|
    if Img.DataDescription.Format=ricfNone then begin
 | 
						|
      Result := False;
 | 
						|
      Exit;
 | 
						|
    end;
 | 
						|
    p := Pointer(buf);
 | 
						|
    for y := 0 to Source.Height-1 do begin
 | 
						|
      for x := 0 to Source.Width-1 do begin
 | 
						|
        c := Img.Colors[x, y];
 | 
						|
        p^ := Hi(c.alpha) shl 24 + Hi(c.red) shl 16 + Hi(c.green) shl 8 + Hi(c.blue);
 | 
						|
        inc(p);
 | 
						|
      end;
 | 
						|
    end;
 | 
						|
  finally
 | 
						|
    Img.Free;
 | 
						|
  end;
 | 
						|
  Result := True;
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
{ TCairoPrinterCanvas }
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetPenProperties;
 | 
						|
  procedure SetDash(d: array of double);
 | 
						|
  begin
 | 
						|
    cairo_set_dash(cr, @d, High(d)+1, 0);
 | 
						|
  end;
 | 
						|
var
 | 
						|
  cap: cairo_line_cap_t;
 | 
						|
   w: double;
 | 
						|
begin
 | 
						|
  SetSourceColor(Pen.Color);
 | 
						|
  case Pen.Mode of
 | 
						|
    pmBlack: begin
 | 
						|
      SetSourceColor(clBlack);
 | 
						|
      cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | 
						|
    end;
 | 
						|
    pmWhite: begin
 | 
						|
      SetSourceColor(clWhite);
 | 
						|
      cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | 
						|
    end;
 | 
						|
    pmCopy: cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | 
						|
    pmXor: cairo_set_operator(cr, CAIRO_OPERATOR_XOR);
 | 
						|
    pmNotXor: cairo_set_operator(cr, CAIRO_OPERATOR_XOR);
 | 
						|
{    pmNop,
 | 
						|
    pmNot,
 | 
						|
    pmCopy,
 | 
						|
    pmNotCopy,
 | 
						|
    pmMergePenNot,
 | 
						|
    pmMaskPenNot,
 | 
						|
    pmMergeNotPen,
 | 
						|
    pmMaskNotPen,
 | 
						|
    pmMerge,
 | 
						|
    pmNotMerge,
 | 
						|
    pmMask,
 | 
						|
    pmNotMask,}
 | 
						|
  else
 | 
						|
    cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | 
						|
  end;
 | 
						|
  w := Pen.Width;
 | 
						|
  if w = 0 then
 | 
						|
    w := 0.5;
 | 
						|
  w := w * ScaleY;
 | 
						|
  cairo_set_line_width(cr, w); //line_width is diameter of the pen circle
 | 
						|
 | 
						|
  case Pen.Style of
 | 
						|
    psSolid:      cairo_set_dash(cr, nil, 0, 0);
 | 
						|
    psDash:       SetDash(Dash_Dash);
 | 
						|
    psDot:        SetDash(Dash_Dot);
 | 
						|
    psDashDot:    SetDash(Dash_DashDot);
 | 
						|
    psDashDotDot: SetDash(Dash_DashDotDot);
 | 
						|
  else
 | 
						|
    cairo_set_dash(cr, nil, 0, 0);
 | 
						|
  end;
 | 
						|
 | 
						|
  case Pen.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 Pen.Style of
 | 
						|
    psDash, psDot, psDashDot, psDashDotDot:
 | 
						|
      cap := CAIRO_LINE_CAP_BUTT
 | 
						|
  end;
 | 
						|
  cairo_set_line_cap(cr, cap);
 | 
						|
 | 
						|
  case Pen.JoinStyle of
 | 
						|
    pjsRound: cairo_set_line_join(cr, CAIRO_LINE_JOIN_ROUND);
 | 
						|
    pjsBevel: cairo_set_line_join(cr, CAIRO_LINE_JOIN_BEVEL);
 | 
						|
    pjsMiter: cairo_set_line_join(cr, CAIRO_LINE_JOIN_MITER);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetBrushProperties;
 | 
						|
begin
 | 
						|
  SetSourceColor(Brush.Color);
 | 
						|
{  case Brush.Style of
 | 
						|
    bsSolid
 | 
						|
    bsClear
 | 
						|
    bsHorizontal
 | 
						|
    bsVertical
 | 
						|
    bsFDiagonal
 | 
						|
    bsBDiagonal
 | 
						|
    bsCross
 | 
						|
    bsDiagCross
 | 
						|
    bsImage
 | 
						|
    bsPattern
 | 
						|
  end;}
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DoLineTo(X1, Y1: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  SetPenProperties;
 | 
						|
  cairo_move_to(cr, SX(PenPos.X), SY(PenPos.Y));
 | 
						|
  cairo_line_to(cr, SX(X1), SY(Y1));
 | 
						|
  SetInternalPenPos(Point(X1,Y1));
 | 
						|
  StrokeOnly;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DoMoveTo(x, y: integer);
 | 
						|
begin
 | 
						|
  // should not call inherited DoMoveTo which would end calling
 | 
						|
  // interface MoveToEx which breaks things for Qt
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DestroyCairoHandle;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetHandle(NewHandle: HDC);
 | 
						|
begin
 | 
						|
  if NewHandle = {%H-}HDC(cr) then
 | 
						|
    exit;
 | 
						|
 | 
						|
  if (NewHandle=0) and (cr<>nil) then
 | 
						|
    DestroyHandle;
 | 
						|
 | 
						|
  cr := {%H-}Pcairo_t(NewHandle);
 | 
						|
 | 
						|
  // update state
 | 
						|
  inherited SetHandle(NewHandle);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.BeginDoc;
 | 
						|
begin
 | 
						|
  inherited BeginDoc;
 | 
						|
  if assigned(printer) then
 | 
						|
    FLazClipRect:=printer.PaperSize.PaperRect.WorkRect;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.EndDoc;
 | 
						|
begin
 | 
						|
  inherited EndDoc;
 | 
						|
  cairo_show_page(cr);
 | 
						|
  FLazClipRect := Rect(0, 0, 0, 0);
 | 
						|
  //if caller is printer, then at the end destroy cairo handles (flush output)
 | 
						|
  //and establishes CreateCairoHandle call on the next print
 | 
						|
  Handle := 0;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.NewPage;
 | 
						|
begin
 | 
						|
  inherited NewPage;
 | 
						|
  cairo_show_page(cr);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.CreateBrush;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.CreateFont;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.CreateHandle;
 | 
						|
begin
 | 
						|
  ScaleX := SurfaceXDPI/XDPI;
 | 
						|
  ScaleY := SurfaceYDPI/YDPI;
 | 
						|
  Handle := CreateCairoHandle;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.CreatePen;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.CreateRegion;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.RealizeAntialiasing;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DestroyHandle;
 | 
						|
begin
 | 
						|
  cairo_destroy(cr);
 | 
						|
  cr := nil;
 | 
						|
  DestroyCairoHandle;
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.GetClipRect: TRect;
 | 
						|
var
 | 
						|
  x1,y1,x2,y2: double;
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
 | 
						|
  // it doesn't matter what the clip is in use, default or user
 | 
						|
  // this returns always the current clip
 | 
						|
 | 
						|
  cairo_clip_extents(cr, @x1, @y1, @x2, @y2);
 | 
						|
  result.Left:=round(x1/ScaleX);
 | 
						|
  result.Top:=round(y1/ScaleY);
 | 
						|
  result.Right:=round(x2/ScaleX);
 | 
						|
  result.Bottom:=round(y2/ScaleY);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetClipRect(const ARect: TRect);
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
  if FUserClipRect=nil then
 | 
						|
    New(FUserClipRect);
 | 
						|
 | 
						|
  fUserClipRect^.x := SX(ARect.Left);
 | 
						|
  fUserClipRect^.y := SY(ARect.Top);
 | 
						|
  fUserClipRect^.width := SX2(ARect.Right-ARect.Left);
 | 
						|
  fUserClipRect^.height:= SY2(ARect.Bottom-ARect.Top);
 | 
						|
 | 
						|
  cairo_reset_clip(cr);
 | 
						|
 | 
						|
  {$ifdef DebugClip}
 | 
						|
  with fUserClipRect^ do begin
 | 
						|
    DrawPoint(x, y, clRed);
 | 
						|
    DrawPoint(x+Width, y+Height, clBlue);
 | 
						|
    DrawRefRect(x, y, width, height, clAqua);
 | 
						|
  end;
 | 
						|
  {$endif}
 | 
						|
 | 
						|
  with fUserClipRect^ do
 | 
						|
    cairo_rectangle(cr, x, y, width, Height);
 | 
						|
 | 
						|
  cairo_Clip(cr);
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.GetClipping: Boolean;
 | 
						|
begin
 | 
						|
  result := (fUserClipRect<>nil);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetClipping(const AValue: boolean);
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
  cairo_reset_clip(cr);
 | 
						|
 | 
						|
  if not AValue then begin
 | 
						|
    // free user cliprect if exists
 | 
						|
    if fUserClipRect<>nil then
 | 
						|
      Dispose(fUserClipRect);
 | 
						|
    fUserClipRect := nil;
 | 
						|
  end
 | 
						|
  else begin
 | 
						|
    if fUserClipRect<>nil then
 | 
						|
    begin
 | 
						|
      with fUserClipRect^ do
 | 
						|
      begin
 | 
						|
        cairo_rectangle(cr, x, y, width, height);
 | 
						|
        cairo_clip(cr);
 | 
						|
      end;
 | 
						|
    end else
 | 
						|
      ; // cairo_reset_clip always clip
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DrawPoint(x, y: double; color: TColor);
 | 
						|
var
 | 
						|
  r,g,b: Double;
 | 
						|
begin
 | 
						|
  TColorToRGB(color, r, g, b);
 | 
						|
  cairo_set_source_rgb(cr, r, g, b);
 | 
						|
  cairo_rectangle(cr, x-2, y-2, 4, 4);
 | 
						|
  cairo_fill(cr);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DrawRefRect(x, y, awidth, aheight: double;
 | 
						|
  color: TColor);
 | 
						|
var
 | 
						|
  r,g,b: double;
 | 
						|
begin
 | 
						|
  TColorToRGB(color, r, g, b);
 | 
						|
  cairo_set_source_rgb(cr, r, g, b);
 | 
						|
  cairo_rectangle(cr, x, y, awidth, aheight);
 | 
						|
  cairo_move_to(cr, x, y);
 | 
						|
  cairo_line_to(cr, x+awidth, y+aheight);
 | 
						|
  cairo_move_to(cr, x+awidth, y);
 | 
						|
  cairo_line_to(cr, x, y+aheight);
 | 
						|
  cairo_stroke(cr);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DebugSys;
 | 
						|
var
 | 
						|
  x,y: double;
 | 
						|
  matrix: cairo_matrix_t;
 | 
						|
begin
 | 
						|
  cairo_get_current_point(cr, @x, @y);
 | 
						|
  cairo_get_matrix(cr, @matrix);
 | 
						|
  DebugLn('CurPoint:  x=%f y=%f',[x, y]);
 | 
						|
  with matrix do
 | 
						|
    DebugLn('CurMatrix: xx=%f yx=%f xy=%f yy=%f x0=%f y0=%f',[xx,yx,xy,yy,x0,y0]);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetLazClipRect(r: TRect);
 | 
						|
begin
 | 
						|
  FLazClipRect := r;
 | 
						|
end;
 | 
						|
 | 
						|
constructor TCairoPrinterCanvas.Create(APrinter: TPrinter);
 | 
						|
begin
 | 
						|
  inherited Create(APrinter);
 | 
						|
  ScaleX := 1;
 | 
						|
  ScaleY := 1;
 | 
						|
  FontScale := 1;
 | 
						|
  SurfaceXDPI := 72;
 | 
						|
  SurfaceYDPI := 72;
 | 
						|
  XDPI := SurfaceXDPI;
 | 
						|
  YDPI := SurfaceXDPI;
 | 
						|
end;
 | 
						|
 | 
						|
constructor TCairoPrinterCanvas.Create;
 | 
						|
begin
 | 
						|
  Create(nil);
 | 
						|
end;
 | 
						|
 | 
						|
destructor TCairoPrinterCanvas.Destroy;
 | 
						|
begin
 | 
						|
  if fUserClipRect<>nil then
 | 
						|
    Dispose(fUserClipRect);
 | 
						|
  fUserClipRect := nil;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  if fFontDesc<>nil then
 | 
						|
    pango_font_description_free(fFontDesc);
 | 
						|
  {$endif}
 | 
						|
  inherited Destroy;
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.SX(x: double): double;
 | 
						|
begin
 | 
						|
  Result := ScaleX*(x+FLazClipRect.Left);
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.SY(y: double): double;
 | 
						|
begin
 | 
						|
  Result := ScaleY*(y+FLazClipRect.Top);
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.SX2(x: double): double;
 | 
						|
begin
 | 
						|
  Result := ScaleX*x;
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.SY2(y: double): double;
 | 
						|
begin
 | 
						|
  Result := ScaleY*y;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetSourceColor(Color: TColor);
 | 
						|
var
 | 
						|
  R, G, B: double;
 | 
						|
begin
 | 
						|
  //TColor je ve formatu BGR
 | 
						|
  TColorToRGB(Color, R, G, B);
 | 
						|
  cairo_set_source_rgb(cr, R, G, B);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Rectangle(X1, Y1, X2, Y2: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csBrushValid, csPenValid]);
 | 
						|
  SetPenProperties;
 | 
						|
  cairo_rectangle(cr, SX(X1), SY(Y1), SX2(X2-X1), SY2(Y2-Y1));
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
//1 point rectangle in _Brush_ color
 | 
						|
procedure TCairoPrinterCanvas.FrameRect(const ARect: TRect);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csBrushValid]);
 | 
						|
  cairo_rectangle(cr, SX(ARect.Left), SY(ARect.Top), SX2(ARect.Right-ARect.Left), SY2(ARect.Bottom-ARect.Top));
 | 
						|
  SetSourceColor(Brush.Color);
 | 
						|
  cairo_set_line_width(cr, 1);
 | 
						|
  cairo_stroke(cr); //Don't touch
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Frame(const ARect: TRect);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  cairo_rectangle(cr, SX(ARect.Left), SY(ARect.Top), SX2(ARect.Right-ARect.Left), SY2(ARect.Bottom-ARect.Top));
 | 
						|
  cairo_set_line_width(cr, 1);
 | 
						|
  SetSourceColor(Pen.Color);
 | 
						|
  cairo_stroke(cr); //Don't touch
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
//C* - center, R* - halfaxis
 | 
						|
procedure TCairoPrinterCanvas.EllipseArcPath(CX, CY, RX, RY: Double; Angle1, Angle2: Double; Clockwise, Continuous: Boolean);
 | 
						|
begin
 | 
						|
  if (RX=0) or (RY=0) then //cairo_scale do not likes zero params
 | 
						|
    Exit;
 | 
						|
  cairo_save(cr);
 | 
						|
  try
 | 
						|
    cairo_translate(cr, SX(CX), SY(CY));
 | 
						|
    cairo_scale(cr, SX2(RX), SY2(RY));
 | 
						|
    if not Continuous then
 | 
						|
      cairo_move_to(cr, cos(Angle1), sin(Angle1)); //Move to arcs starting point
 | 
						|
    if Clockwise then
 | 
						|
      cairo_arc(cr, 0, 0, 1, Angle1, Angle2)
 | 
						|
    else
 | 
						|
    cairo_arc_negative(cr, 0, 0, 1, Angle1, Angle2);
 | 
						|
  finally
 | 
						|
    cairo_restore(cr);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.FillOnly;
 | 
						|
begin
 | 
						|
  if Brush.Style <> bsClear then begin
 | 
						|
    SetBrushProperties;
 | 
						|
    cairo_fill(cr);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.StrokeOnly;
 | 
						|
begin
 | 
						|
  if Pen.Style <> psClear then begin
 | 
						|
    SetPenProperties;
 | 
						|
    cairo_stroke(cr);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.TColorToRGB(Color: TColor; out R, G, B: double);
 | 
						|
begin
 | 
						|
  R := (Color and $FF) / 255;
 | 
						|
  G := ((Color shr 8) and $FF) / 255;
 | 
						|
  B := ((Color shr 16) and $FF) / 255;
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
{$ifdef pangocairo}
 | 
						|
function TCairoPrinterCanvas.StylesToStr(Styles: TFontStyles): string;
 | 
						|
begin
 | 
						|
  Result := '';
 | 
						|
  if fsBold in Styles then
 | 
						|
    Result := Result + 'bold ';
 | 
						|
  if fsItalic in Styles then
 | 
						|
    Result := Result + 'italic ';
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.UpdatePangoLayout(Layout: PPangoLayout);
 | 
						|
var
 | 
						|
  AttrListTemporary: Boolean;
 | 
						|
  AttrList: PPangoAttrList;
 | 
						|
  Attr: PPangoAttribute;
 | 
						|
begin
 | 
						|
  if Font.Underline or Font.StrikeThrough then begin
 | 
						|
 | 
						|
    AttrListTemporary := false;
 | 
						|
    AttrList := pango_layout_get_attributes(Layout);
 | 
						|
    if (AttrList = nil) then
 | 
						|
    begin
 | 
						|
      AttrList := pango_attr_list_new();
 | 
						|
      AttrListTemporary := True;
 | 
						|
    end;
 | 
						|
    if Font.Underline 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(Font.StrikeThrough);
 | 
						|
    pango_attr_list_change(AttrList, Attr);
 | 
						|
 | 
						|
    pango_layout_set_attributes(Layout, AttrList);
 | 
						|
 | 
						|
    pango_cairo_update_layout(cr, Layout);
 | 
						|
 | 
						|
    if AttrListTemporary then
 | 
						|
      pango_attr_list_unref(AttrList);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
{$endif}
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.FillAndStroke;
 | 
						|
begin
 | 
						|
  if Brush.Style <> bsClear then begin
 | 
						|
    SetBrushProperties;
 | 
						|
    if Pen.Style = psClear then
 | 
						|
      cairo_fill(cr)
 | 
						|
    else
 | 
						|
      cairo_fill_preserve(cr);
 | 
						|
  end;
 | 
						|
  if Pen.Style <> psClear then begin
 | 
						|
    SetPenProperties;
 | 
						|
    cairo_stroke(cr);
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.RoundRect(X1, Y1, X2, Y2: Integer; RX, RY: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  cairo_move_to(cr, SX(X1+RX), SY(Y1));
 | 
						|
  cairo_line_to(cr, SX(X2-RX), SY(Y1));
 | 
						|
  EllipseArcPath(X2-RX, Y1+RY, RX, RY, -PI/2, 0, True, True);
 | 
						|
  cairo_line_to(cr, SX(X2), SY(Y2-RY));
 | 
						|
  EllipseArcPath(X2-RX, Y2-RY, RX, RY, 0, PI/2, True, True);
 | 
						|
  cairo_line_to(cr, SX(X1+RX), SY(Y2));
 | 
						|
  EllipseArcPath(X1+RX, Y2-RY, RX, RY, PI/2, PI, True, True);
 | 
						|
  cairo_line_to(cr, SX(X1), SY(Y1+RX));
 | 
						|
  EllipseArcPath(X1+RX, Y1+RY, RX, RY, PI, PI*1.5, True, True);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.MixedRoundRect(X1, Y1, X2, Y2: Integer; RX,
 | 
						|
  RY: Integer; SquaredCorners: TSquaredCorners);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
 | 
						|
  cairo_move_to(cr, SX(X1+RX), SY(Y1));
 | 
						|
  cairo_line_to(cr, SX(X2-RX), SY(Y1));
 | 
						|
 | 
						|
  if scTopRight in SquaredCorners then
 | 
						|
  begin
 | 
						|
    cairo_line_to(cr, SX(X2), SY(Y1));
 | 
						|
    cairo_line_to(cr, SX(X2), SY(Y1+RY));
 | 
						|
  end else
 | 
						|
    EllipseArcPath(X2-RX, Y1+RY, RX, RY, -PI/2, 0, True, True);
 | 
						|
 | 
						|
  cairo_line_to(cr, SX(X2), SY(Y2-RY));
 | 
						|
 | 
						|
  if scBottomRight in SquaredCorners then
 | 
						|
  begin
 | 
						|
    cairo_line_to(cr, SX(X2), SY(Y2));
 | 
						|
    cairo_line_to(cr, SX(X2-RX), SY(Y2));
 | 
						|
  end else
 | 
						|
    EllipseArcPath(X2-RX, Y2-RY, RX, RY, 0, PI/2, True, True);
 | 
						|
 | 
						|
  cairo_line_to(cr, SX(X1+RX), SY(Y2));
 | 
						|
 | 
						|
  if scBottomLeft in SquaredCorners then
 | 
						|
  begin
 | 
						|
    cairo_line_to(cr, SX(X1), SY(Y2));
 | 
						|
    cairo_line_to(cr, SX(X1), SY(Y2-RY));
 | 
						|
  end else
 | 
						|
    EllipseArcPath(X1+RX, Y2-RY, RX, RY, PI/2, PI, True, True);
 | 
						|
 | 
						|
  cairo_line_to(cr, SX(X1), SY(Y1+RX));
 | 
						|
 | 
						|
  if scTopLeft in SquaredCorners then
 | 
						|
  begin
 | 
						|
    cairo_line_to(cr, SX(X1), SY(Y1));
 | 
						|
    cairo_line_to(cr, SX(X1+RX), SY(Y1));
 | 
						|
  end else
 | 
						|
    EllipseArcPath(X1+RX, Y1+RY, RX, RY, PI, PI*1.5, True, True);
 | 
						|
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.DrawSurface(const SourceRect, DestRect: TRect;
 | 
						|
  surface: Pcairo_surface_t);
 | 
						|
var
 | 
						|
  SW, SH: Double;
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
 | 
						|
  cairo_save(cr);
 | 
						|
  cairo_translate(cr, SX(DestRect.Left), SY(DestRect.Top));
 | 
						|
  SW := (DestRect.Right - DestRect.Left)/(SourceRect.Right-SourceRect.Left);
 | 
						|
  SH := (DestRect.Bottom - DestRect.Top)/(SourceRect.Bottom-SourceRect.Top);
 | 
						|
  cairo_scale(cr, SX2(SW), SY2(SH));
 | 
						|
  cairo_set_source_surface(cr, surface, 0, 0);
 | 
						|
  cairo_paint(cr);
 | 
						|
  cairo_restore(cr);
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Ellipse(X1, Y1, X2, Y2: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  EllipseArcPath((X2+X1)/2, (Y2+Y1)/2, (X2-X1)/2, (Y2-Y1)/2, 0, 2*PI, True, False);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Arc(ALeft, ATop, ARight, ABottom, Angle16Deg, Angle16DegLength: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  ArcPath(ALeft, ATop, ARight, ABottom, Angle16Deg, Angle16DegLength);
 | 
						|
  StrokeOnly;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Chord(X1, Y1, X2, Y2, Angle1, Angle2: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  ArcPath(X1, Y1, X2, Y2, Angle1, Angle2);
 | 
						|
  cairo_close_path(cr);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.RadialPie(Left, Top, Right, Bottom, Angle1, Angle2: Integer);
 | 
						|
var
 | 
						|
  cx, cy: double;
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  ArcPath(Left, Top, Right, Bottom, Angle1, Angle2);
 | 
						|
  cx := (Right+Left)/2;
 | 
						|
  cy := (Bottom+Top)/2;
 | 
						|
  cairo_line_to(cr, SX(cx), SY(cy));
 | 
						|
  cairo_close_path(cr);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.ArcPath(ALeft, ATop, ARight, ABottom, Angle16Deg, Angle16DegLength: double);
 | 
						|
var
 | 
						|
  k: Double;
 | 
						|
begin
 | 
						|
  k := - 2*PI/(360*16);
 | 
						|
  EllipseArcPath((ARight+ALeft)/2, (ABottom+ATop)/2, (ARight-ALeft)/2, (ABottom-ATop)/2,
 | 
						|
    Angle16Deg*k, Angle16DegLength*k, False, False);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Arc(ALeft, ATop, ARight, ABottom, StX, StY, EX, EY: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  ArcPath(ALeft, ATop, ARight, ABottom, StX, StY, EX, EY);
 | 
						|
  StrokeOnly;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Chord(X1, Y1, X2, Y2, StX, StY, EX, EY: Integer);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  ArcPath(X1, Y1, X2, Y2, StX, StY, EX, EY);
 | 
						|
  cairo_close_path(cr);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Pie(EllipseX1, EllipseY1, EllipseX2, EllipseY2,
 | 
						|
  StartX, StartY, EndX, EndY: Integer);
 | 
						|
var
 | 
						|
  cx, cy: double;
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid, csBrushValid]);
 | 
						|
  ArcPath(EllipseX1, EllipseY1, EllipseX2, EllipseY2, StartX, StartY, EndX, EndY);
 | 
						|
  cx := (EllipseX2+EllipseX1)/2;
 | 
						|
  cy := (EllipseY2+EllipseY1)/2;
 | 
						|
  cairo_line_to(cr, SX(cx), SY(cy));
 | 
						|
  cairo_close_path(cr);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.ArcPath(ALeft, ATop, ARight, ABottom, StX, StY, EX, EY: double);
 | 
						|
 | 
						|
  function ATanInt(x, y: double): double;
 | 
						|
  begin
 | 
						|
    if x <> 0 then begin
 | 
						|
      result := ArcTan(y/x);
 | 
						|
      if x < 0 then
 | 
						|
        result := result + PI;
 | 
						|
    end else begin
 | 
						|
      if y > 0 then
 | 
						|
        result := PI/2
 | 
						|
      else
 | 
						|
        result := - PI/2;
 | 
						|
    end;
 | 
						|
  end;
 | 
						|
 | 
						|
var
 | 
						|
  Angle1, Angle2: double;
 | 
						|
  cx, cy: double;
 | 
						|
begin
 | 
						|
  cx := (ARight+ALeft)/2;
 | 
						|
  cy := (ABottom+ATop)/2;
 | 
						|
  Angle1 := ATanInt(StX-cx, StY-cy);
 | 
						|
  Angle2 := ATanInt(EX-cx, EY-cy);
 | 
						|
  EllipseArcPath(cx, cy, (ARight-ALeft)/2, (ABottom-ATop)/2, Angle1, Angle2, False, False);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.PolyBezier(Points: PPoint; NumPts: Integer; Filled: boolean; Continuous: boolean);
 | 
						|
var
 | 
						|
  p, ep: PPoint;
 | 
						|
begin
 | 
						|
  p := Points;
 | 
						|
  ep := Points + NumPts;
 | 
						|
  while p < ep do begin
 | 
						|
    if (p = Points) or not Continuous then begin //First or non cont.
 | 
						|
      cairo_move_to(cr, SX(p^.X), SY(p^.Y));
 | 
						|
      inc(p);
 | 
						|
    end;
 | 
						|
    cairo_curve_to(cr, SX(p^.X), SY(p^.Y), SX((p+1)^.X), SY((p+1)^.Y), SX((p+2)^.X), SY((p+2)^.Y));
 | 
						|
    inc(p, 3);
 | 
						|
  end;
 | 
						|
  if Filled then begin
 | 
						|
    cairo_close_path(cr);
 | 
						|
    FillAndStroke;
 | 
						|
  end else
 | 
						|
    StrokeOnly;
 | 
						|
end;
 | 
						|
 | 
						|
//Toy interface
 | 
						|
procedure TCairoPrinterCanvas.SelectFont;
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
  SelectFontEx(Font.Style, Font.Name, abs(Font.Size), Font.Pitch);
 | 
						|
  SetSourceColor(Font.Color);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SelectFontEx(AStyle: TFontStyles;
 | 
						|
  const AName: string; ASize: double; aPitch: TFontPitch);
 | 
						|
var
 | 
						|
  slant: cairo_font_slant_t;
 | 
						|
  weight: cairo_font_weight_t;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  S, aFontName: string;
 | 
						|
  {$endif}
 | 
						|
begin
 | 
						|
  if fsBold in Font.Style then
 | 
						|
    weight := CAIRO_FONT_WEIGHT_BOLD
 | 
						|
  else
 | 
						|
    weight := CAIRO_FONT_WEIGHT_NORMAL;
 | 
						|
  if fsItalic in Font.Style then
 | 
						|
    slant := CAIRO_FONT_SLANT_ITALIC
 | 
						|
  else
 | 
						|
    slant := CAIRO_FONT_SLANT_NORMAL;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  if ASize<0.001 then
 | 
						|
    ASize := 10.0;
 | 
						|
  aFontName := AName;
 | 
						|
  if (aFontName='') or SameText(aFontName, 'default') then begin
 | 
						|
    if aPitch=fpFixed then
 | 
						|
      aFontName := 'monospace'
 | 
						|
    else
 | 
						|
      aFontName := 'sans-serif';
 | 
						|
  end;
 | 
						|
  S := format('%s %s %dpx',[aFontName, StylesToStr(AStyle), round(ASize)]);
 | 
						|
  if (fFontDesc=nil) or (S<>fFontDescStr) then
 | 
						|
  begin
 | 
						|
    if fFontDesc<>nil then
 | 
						|
      pango_font_description_free(fFontDesc);
 | 
						|
    fFontDesc := pango_font_description_from_string(pchar(s));
 | 
						|
  end;
 | 
						|
  fFontDescStr := s;
 | 
						|
  {$endif}
 | 
						|
  cairo_select_font_face(cr, PChar(AName), slant, weight);
 | 
						|
  cairo_set_font_size(cr, ASize*FontScale)
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.TextOut(X, Y: Integer; const Text: String);
 | 
						|
var
 | 
						|
  e: cairo_font_extents_t;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  Layout: PPangoLayout;
 | 
						|
  {$endif}
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csFontValid, csBrushValid]);
 | 
						|
  SelectFont;
 | 
						|
  cairo_font_extents(cr, @e);
 | 
						|
  cairo_save(cr);
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  // use absolute font size sintax  (px)
 | 
						|
  Layout := Pango_Cairo_Create_Layout(cr);
 | 
						|
  pango_layout_set_font_description(layout, fFontDesc);
 | 
						|
  UpdatePangoLayout(Layout);
 | 
						|
  {$endif}
 | 
						|
  if Font.Orientation = 0 then
 | 
						|
  begin
 | 
						|
    cairo_move_to(cr, SX(X), SY(Y)+e.ascent);
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    //DebugLn('TextOut ',Text);
 | 
						|
    //DebugSys;
 | 
						|
    pango_layout_set_text(layout, PChar(Text), -1);
 | 
						|
    {$else}
 | 
						|
    cairo_show_text(cr, PChar(Text)); //Reference point is on the base line
 | 
						|
    {$endif}
 | 
						|
  end
 | 
						|
  else
 | 
						|
  begin
 | 
						|
    cairo_move_to(cr, SX(X)+e.ascent, SY(Y));
 | 
						|
    cairo_rotate(cr, -gradtorad(Font.Orientation));
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    pango_layout_set_text(layout, PChar(Text), -1);
 | 
						|
    {$else}
 | 
						|
    cairo_show_text(cr, PChar(Text)); //Reference point is on the base line
 | 
						|
    {$endif}
 | 
						|
  end;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  pango_cairo_update_layout(cr, layout);
 | 
						|
  // get the same text origin as cairo_show_text (baseline left, instead of Pango's top left)
 | 
						|
  pango_cairo_show_layout_line (cr, pango_layout_get_line (layout, 0));
 | 
						|
  g_object_unref(layout);
 | 
						|
  {$endif}
 | 
						|
  cairo_restore(cr);
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
{$ifdef breaklines}
 | 
						|
type
 | 
						|
  TLine = class
 | 
						|
    Start, EndL: Integer;
 | 
						|
    Width: Double;
 | 
						|
  end;
 | 
						|
{$endif}
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.TextRect(ARect: TRect; X1, Y1: integer; const Text: string; const Style: TTextStyle);
 | 
						|
var
 | 
						|
  s: string;
 | 
						|
{$ifdef breaklines}
 | 
						|
  te: cairo_text_extents_t;
 | 
						|
  Lines: TList;
 | 
						|
  CurLine: TLine;
 | 
						|
  len: integer;
 | 
						|
  LastBreakEndL: Integer;
 | 
						|
  LastBreakStart: Integer;
 | 
						|
 | 
						|
  procedure BreakLine(en, st: Integer);
 | 
						|
  var
 | 
						|
    s1: string;
 | 
						|
    te: cairo_text_extents_t;
 | 
						|
  begin
 | 
						|
    if en>=0 then begin
 | 
						|
      //if en>1 then begin
 | 
						|
        if en <= len then
 | 
						|
          CurLine.EndL := en
 | 
						|
        else
 | 
						|
          CurLine.EndL := len;
 | 
						|
      //end else
 | 
						|
        //CurLine.EndL := 1;
 | 
						|
      s1 := Copy(s, CurLine.Start, CurLine.EndL-CurLine.Start+1);
 | 
						|
      cairo_text_extents(cr, PChar(s1), @te);
 | 
						|
      CurLine.Width := te.width;
 | 
						|
    end;
 | 
						|
    if st > 0 then begin
 | 
						|
      CurLine := TLine.Create;
 | 
						|
      Lines.Add(CurLine);
 | 
						|
      //if st <= len then
 | 
						|
        CurLine.Start := st;
 | 
						|
      //else
 | 
						|
      //  CurLine.Start := len;
 | 
						|
      CurLine.EndL := 0;
 | 
						|
    end;
 | 
						|
    LastBreakEndL := 0;
 | 
						|
    LastBreakStart := 0;
 | 
						|
  end;
 | 
						|
{$endif}
 | 
						|
 | 
						|
var
 | 
						|
  fd: TFontData;
 | 
						|
  s1: string;
 | 
						|
  i: integer;
 | 
						|
  BoxLeft, BoxTop, BoxWidth, BoxHeight: Double;
 | 
						|
  StartLeft, StartTop: Double;
 | 
						|
  x, y: Double;
 | 
						|
  r,b: double;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  Layout: PPangoLayout;
 | 
						|
  ink,logical: TPangoRectangle;
 | 
						|
  {$endif}
 | 
						|
 | 
						|
  {$ifdef breaklines}
 | 
						|
  fe: cairo_font_extents_t;
 | 
						|
  BreakBoxWidth: Double;
 | 
						|
  j: integer;
 | 
						|
  ch: string;
 | 
						|
  {$else}
 | 
						|
  Lines: TStringList;
 | 
						|
  {$endif}
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csFontValid, csBrushValid]);
 | 
						|
  cairo_save(cr);
 | 
						|
  try
 | 
						|
    s := Text;
 | 
						|
    BoxWidth := SX2(ARect.Right-ARect.Left);
 | 
						|
    BoxHeight := SY2(ARect.Bottom-ARect.Top);
 | 
						|
    BoxLeft := SX(ARect.Left);
 | 
						|
    BoxTop := SY(ARect.Top);
 | 
						|
    StartLeft := SX(X1);
 | 
						|
    StartTop := SY(Y1);
 | 
						|
    //DebugLn('Box= l=%f t=%f',[BoxLeft,BoxTop]);
 | 
						|
    //DebugLn('     x=%f y=%f',[StartLeft,StartTop]);
 | 
						|
    {$ifdef breaklines}
 | 
						|
    if Style.Alignment = taLeftJustify then
 | 
						|
      BreakBoxWidth := SX(ARect.Right - X1)
 | 
						|
    else
 | 
						|
      BreakBoxWidth := BoxWidth;
 | 
						|
    {$endif}
 | 
						|
 | 
						|
    if Style.Clipping then begin
 | 
						|
      r := BoxWidth+Pen.Width;
 | 
						|
      b := BoxHeight+Pen.Width;
 | 
						|
 | 
						|
      {$ifdef DebugClip}
 | 
						|
      DrawPoint(boxLeft, boxTop, clRed);
 | 
						|
      DrawPoint(boxLeft+r, boxTop+b, clBlue);
 | 
						|
      DrawRefRect(boxLeft, boxTop, r, b, clGreen);
 | 
						|
      {$endif}
 | 
						|
 | 
						|
      cairo_rectangle(cr, BoxLeft, BoxTop, r, b);
 | 
						|
      cairo_clip(cr);
 | 
						|
    end;
 | 
						|
 | 
						|
    if (Font.Orientation=900) or (Font.Orientation=2700) then begin
 | 
						|
      x := BoxWidth;
 | 
						|
      BoxWidth := BoxHeight;
 | 
						|
      BoxHeight := x;
 | 
						|
    end;
 | 
						|
 | 
						|
    if Style.ExpandTabs then
 | 
						|
      s := StringReplace(s, #9, '        ', [rfReplaceAll])
 | 
						|
    else
 | 
						|
      s := StringReplace(s, #9, ' ', [rfReplaceAll]);
 | 
						|
 | 
						|
    if Style.SingleLine then begin
 | 
						|
      s := StringReplace(s, #13+#10, ' ', [rfReplaceAll]);
 | 
						|
      s := StringReplace(s, #13, ' ', [rfReplaceAll]);
 | 
						|
      s := StringReplace(s, #10, ' ', [rfReplaceAll]);
 | 
						|
    end;
 | 
						|
 | 
						|
    if Style.Opaque then begin
 | 
						|
      SetSourceColor(Brush.Color);
 | 
						|
      cairo_rectangle(cr, BoxLeft, BoxTop, BoxWidth, BoxHeight);
 | 
						|
      cairo_fill(cr)
 | 
						|
    end;
 | 
						|
 | 
						|
    if Style.SystemFont and Assigned(OnGetSystemFont) then begin
 | 
						|
      fd := GetFontData(OnGetSystemFont());
 | 
						|
      SelectFontEx(fd.Style, fd.Name, fd.Height, fd.Pitch);
 | 
						|
      SetSourceColor(clWindowText);
 | 
						|
    end else
 | 
						|
      SelectFont;
 | 
						|
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    Layout := Pango_Cairo_Create_Layout(cr);
 | 
						|
    pango_layout_set_font_description(layout, fFontDesc);
 | 
						|
    UpdatePangolayout(Layout);
 | 
						|
    {$else}
 | 
						|
    cairo_font_extents(cr, @fe);
 | 
						|
    {$endif}
 | 
						|
 | 
						|
    {$ifdef breaklines}
 | 
						|
    Lines := TList.Create;
 | 
						|
    //Break lines
 | 
						|
    len := Length(s);
 | 
						|
    BreakLine(-1, 1);
 | 
						|
    i := 1;
 | 
						|
    while i<=len+1 do begin
 | 
						|
      if i<=len then
 | 
						|
        ch := s[i]
 | 
						|
      else
 | 
						|
        ch := '';
 | 
						|
      //CR LF breaking
 | 
						|
      if ch = #13 then begin
 | 
						|
        if (i < len) and (s[i+1] = #10) then begin
 | 
						|
          BreakLine(i-1, i+2);
 | 
						|
          inc(i, 2);
 | 
						|
          Continue;
 | 
						|
        end else begin
 | 
						|
          BreakLine(i-1, i+1);
 | 
						|
          inc(i, 1);
 | 
						|
          Continue;
 | 
						|
        end;
 | 
						|
      end;
 | 
						|
      if ch = #10 then begin
 | 
						|
        BreakLine(i-1, i+1);
 | 
						|
        inc(i, 1);
 | 
						|
        Continue;
 | 
						|
      end;
 | 
						|
 | 
						|
      //Word breaking
 | 
						|
      if Style.Wordbreak then begin
 | 
						|
        if (ch = '') or (ch = ' ') then begin //'' last char
 | 
						|
          s1 := Copy(s, CurLine.Start, i-CurLine.Start);
 | 
						|
          {$ifdef pangocairo}
 | 
						|
          {$else}
 | 
						|
          cairo_text_extents(cr, PChar(s1), @te);
 | 
						|
          {$endif}
 | 
						|
          //skip following break chars
 | 
						|
          j := i+1;
 | 
						|
          while (j<=len) and (s[j] = ' ') do
 | 
						|
            inc(j);
 | 
						|
          if (te.width+te.x_bearing) <= BreakBoxWidth then begin
 | 
						|
            LastBreakEndL := i-1;
 | 
						|
            LastBreakStart := j;
 | 
						|
          end else begin //overflow
 | 
						|
            if LastBreakEndL<=0 then begin //cannot break
 | 
						|
              BreakLine(i-1, j);
 | 
						|
              inc(i);
 | 
						|
              Continue;
 | 
						|
            end else begin
 | 
						|
              i := LastBreakStart; //before BreakLine where is LastBreakStart changed
 | 
						|
              BreakLine(LastBreakEndL, LastBreakStart);
 | 
						|
              Continue;
 | 
						|
            end;
 | 
						|
          end;
 | 
						|
        end;
 | 
						|
      end;
 | 
						|
 | 
						|
      //next char
 | 
						|
      inc(i);
 | 
						|
    end;
 | 
						|
    //Close last CurLine
 | 
						|
    BreakLine(Len, -1);
 | 
						|
 | 
						|
    {$else breaklines}
 | 
						|
 | 
						|
    Lines := TStringList.Create;
 | 
						|
    Lines.Text := s;
 | 
						|
 | 
						|
    {$endif}
 | 
						|
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    if Style.Wordbreak then begin
 | 
						|
      pango_layout_set_width(layout, Round(BoxWidth*PANGO_SCALE));
 | 
						|
      pango_layout_set_wrap(layout, PANGO_WRAP_WORD);
 | 
						|
      case Style.Alignment of //Works only with pango_layout_set_width
 | 
						|
        taLeftJustify:  pango_layout_set_alignment(layout, PANGO_ALIGN_LEFT);
 | 
						|
        taCenter:       pango_layout_set_alignment(layout, PANGO_ALIGN_CENTER);
 | 
						|
        taRightJustify: pango_layout_set_alignment(layout, PANGO_ALIGN_RIGHT);
 | 
						|
      end;
 | 
						|
    end;
 | 
						|
 | 
						|
    pango_layout_set_text(layout, pchar(s), -1);
 | 
						|
    pango_layout_get_extents(Layout, @ink, @logical);
 | 
						|
    //Calc start 'box' relative positions
 | 
						|
    case Style.Layout of
 | 
						|
      tlTop:    y := 0;
 | 
						|
      tlCenter: y := BoxHeight/2 - logical.Height/PANGO_SCALE/2;
 | 
						|
      tlBottom: y := BoxHeight - logical.height/PANGO_SCALE;
 | 
						|
    end;
 | 
						|
    {$else}
 | 
						|
    //Calc start positions
 | 
						|
    case Style.Layout of
 | 
						|
      tlTop:    y := 0;
 | 
						|
      tlCenter: y := BoxHeight/2 - fe.height*Lines.Count/2;
 | 
						|
      tlBottom: y := BoxHeight - fe.height*Lines.Count;
 | 
						|
    end;
 | 
						|
    {$endif}
 | 
						|
 | 
						|
    // translate origin
 | 
						|
    cairo_translate(cr, StartLeft, StartTop);
 | 
						|
    // rotate
 | 
						|
    cairo_rotate(cr, -DegToRad(Font.Orientation/10));
 | 
						|
 | 
						|
    //Text output
 | 
						|
    for i := 0 to Lines.Count-1 do begin
 | 
						|
 | 
						|
      {$ifdef breaklines}
 | 
						|
      CurLine := TLine(Lines.Items[i]);
 | 
						|
      s1 := Copy(s, CurLine.Start, CurLine.EndL-CurLine.Start+1);
 | 
						|
      {$else}
 | 
						|
      s1 := Lines[i];
 | 
						|
      {$endif}
 | 
						|
 | 
						|
      //DebugLn('i=%i y=%f s1=%s',[i,y,s1]);
 | 
						|
      {$ifdef pangocairo}
 | 
						|
      pango_layout_set_text(layout, pchar(s1), -1);
 | 
						|
      pango_layout_get_extents(Layout, @ink, @logical);
 | 
						|
      x := 0;
 | 
						|
      if not Style.Wordbreak then begin
 | 
						|
        case Style.Alignment of
 | 
						|
          taCenter:       x := BoxWidth/2 - logical.width/PANGO_SCALE/2;
 | 
						|
          taRightJustify: x := BoxWidth - logical.Width/PANGO_SCALE;
 | 
						|
        end;
 | 
						|
      end;
 | 
						|
      cairo_move_to(cr, x, y);
 | 
						|
      //DebugLn('TextRect ',S1);
 | 
						|
      //DebugSys;
 | 
						|
      pango_cairo_show_layout(cr, layout);
 | 
						|
      y := y + logical.height/PANGO_SCALE;
 | 
						|
      {$else}
 | 
						|
      case Style.Alignment of
 | 
						|
        taLeftJustify: x := StartLeft;
 | 
						|
        taCenter: x := BoxLeft + BoxWidth/2 - CurLine.Width/2;
 | 
						|
        taRightJustify: x := BoxLeft+BoxWidth - CurLine.Width;
 | 
						|
      end;
 | 
						|
      cairo_move_to(cr, x, y+fe.ascent);
 | 
						|
      cairo_show_text(cr, PChar(s1)); //Reference point is on the base line
 | 
						|
      y := y + fe.height;
 | 
						|
      {$endif}
 | 
						|
    end;
 | 
						|
    {$ifdef pangocairo}
 | 
						|
    g_object_unref(layout);
 | 
						|
    {$endif}
 | 
						|
 | 
						|
  finally
 | 
						|
    cairo_restore(cr);
 | 
						|
    {$ifdef breaklines}
 | 
						|
    for i := 0 to Lines.Count-1 do
 | 
						|
      TLine(Lines.Items[i]).Free;
 | 
						|
    {$endif}
 | 
						|
    Lines.Free;
 | 
						|
  end;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.TextExtent(const Text: string): TSize;
 | 
						|
var
 | 
						|
  extents: cairo_text_extents_t;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  Layout: PPangoLayout;
 | 
						|
  theRect: TPangoRectangle;
 | 
						|
  {$endif}
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid, csFontValid]);
 | 
						|
  SelectFont;
 | 
						|
  {$ifdef pangocairo}
 | 
						|
  Layout := Pango_Cairo_Create_Layout(cr);
 | 
						|
  pango_layout_set_font_description(Layout, fFontDesc);
 | 
						|
  cairo_text_extents(cr, PChar(Text), @extents);
 | 
						|
  pango_layout_set_text(Layout, pchar(Text), -1);
 | 
						|
  pango_layout_get_extents(Layout, nil, @theRect);
 | 
						|
  Result.cx := Round((theRect.width/PANGO_SCALE)/ScaleX);
 | 
						|
  Result.cy := Round((theRect.height/PANGO_SCALE)/ScaleY);
 | 
						|
  g_object_unref(Layout);
 | 
						|
  {$else}
 | 
						|
  cairo_text_extents(cr, PChar(Text), @extents); //transformation matrix is here ignored
 | 
						|
  Result.cx := Round((extents.width)/ScaleX+extents.x_bearing);
 | 
						|
  Result.cy := Round((extents.height)/ScaleY-extents.y_bearing);
 | 
						|
  {$endif}
 | 
						|
end;
 | 
						|
 | 
						|
function TCairoPrinterCanvas.GetTextMetrics(out M: TLCLTextMetric): boolean;
 | 
						|
var
 | 
						|
  e: cairo_font_extents_t;
 | 
						|
begin
 | 
						|
  RequiredState([csHandleValid, csFontValid]);
 | 
						|
  SelectFont;
 | 
						|
  cairo_font_extents(cr, @e); //transformation matrix is here ignored
 | 
						|
  FillChar(M{%H-}, SizeOf(M), 0);
 | 
						|
  M.Ascender := Round(e.ascent/ScaleY);
 | 
						|
  M.Descender := Round(e.descent/ScaleY);
 | 
						|
  M.Height := Round(e.height/ScaleY);
 | 
						|
  Result := True;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.StretchDraw(const DestRect: TRect; SrcGraphic: TGraphic);
 | 
						|
var
 | 
						|
  sf: Pcairo_surface_t;
 | 
						|
  buf: PByte;
 | 
						|
  W, H: Integer;
 | 
						|
  SW, SH: Double;
 | 
						|
begin
 | 
						|
  if not (SrcGraphic is TRasterImage) then begin
 | 
						|
    inherited StretchDraw(DestRect, SrcGraphic);
 | 
						|
    Exit;
 | 
						|
  end;
 | 
						|
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid]);
 | 
						|
  W := SrcGraphic.Width;
 | 
						|
  H := SrcGraphic.Height;
 | 
						|
 | 
						|
  buf := GetMem(W*H*4);
 | 
						|
  try
 | 
						|
    cairo_save(cr);
 | 
						|
    //FillDWord(buf^, W*H, $00000000);
 | 
						|
    if not GraphicToARGB32(SrcGraphic, buf) then
 | 
						|
      Exit;
 | 
						|
 | 
						|
    sf := cairo_image_surface_create_for_data(buf, CAIRO_FORMAT_ARGB32, W, H, W*4);
 | 
						|
    cairo_translate(cr, SX(DestRect.Left), SY(DestRect.Top));
 | 
						|
    SW := (DestRect.Right - DestRect.Left)/W;
 | 
						|
    SH := (DestRect.Bottom - DestRect.Top)/H;
 | 
						|
    cairo_scale(cr, SX2(SW), SY2(SH));
 | 
						|
    cairo_set_source_surface(cr, sf, 0, 0);
 | 
						|
    cairo_paint(cr);
 | 
						|
    cairo_surface_destroy(sf);
 | 
						|
    cairo_restore(cr);
 | 
						|
  finally
 | 
						|
    FreeMem(buf);
 | 
						|
  end;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.SetPixel(X, Y: Integer; Value: TColor);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  SetSourceColor(Value);
 | 
						|
  cairo_rectangle(cr, SX(X), SY(Y), 1, 1);
 | 
						|
  cairo_fill(cr);
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.PolylinePath(Points: PPoint; NumPts: Integer);
 | 
						|
var
 | 
						|
  p: PPoint;
 | 
						|
  i: integer;
 | 
						|
begin
 | 
						|
  p := Points;
 | 
						|
  cairo_move_to(cr, SX(p^.X), SY(p^.Y));
 | 
						|
  for i := 0 to NumPts-2 do begin
 | 
						|
    inc(p);
 | 
						|
    cairo_line_to(cr, SX(p^.X), SY(p^.Y));
 | 
						|
  end;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Polyline(Points: PPoint; NumPts: Integer);
 | 
						|
begin
 | 
						|
  if NumPts <= 0 then
 | 
						|
    Exit;
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csPenValid]);
 | 
						|
  PolylinePath(Points, NumPts);
 | 
						|
  StrokeOnly;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.Polygon(Points: PPoint; NumPts: Integer; Winding: boolean);
 | 
						|
begin
 | 
						|
  if NumPts <= 0 then
 | 
						|
    Exit;
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csBrushValid, csPenValid]);
 | 
						|
  PolylinePath(Points, NumPts);
 | 
						|
  cairo_close_path(cr);
 | 
						|
  FillAndStroke;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPrinterCanvas.FillRect(const ARect: TRect);
 | 
						|
begin
 | 
						|
  Changing;
 | 
						|
  RequiredState([csHandleValid, csBrushValid]);
 | 
						|
  cairo_rectangle(cr, SX(ARect.Left), SY(ARect.Top), SX2(ARect.Right-ARect.Left), SY2(ARect.Bottom-ARect.Top));
 | 
						|
  FillOnly;
 | 
						|
  Changed;
 | 
						|
end;
 | 
						|
 | 
						|
{ TCairoFileCanvas }
 | 
						|
 | 
						|
procedure TCairoFileCanvas.DestroyCairoHandle;
 | 
						|
begin
 | 
						|
  cairo_surface_finish(sf);
 | 
						|
  cairo_surface_destroy(sf);
 | 
						|
  sf := nil;
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoFileCanvas.UpdatePageSize;
 | 
						|
begin
 | 
						|
end;
 | 
						|
 | 
						|
{ TCairoPdfCanvas }
 | 
						|
 | 
						|
function TCairoPdfCanvas.CreateCairoHandle: HDC;
 | 
						|
begin
 | 
						|
  //Sizes are in Points, 72DPI (1pt = 1/72")
 | 
						|
  if fStream<>nil then
 | 
						|
    sf := cairo_pdf_surface_create_for_stream(@WriteToStream, fStream, PaperWidth*ScaleX, PaperHeight*ScaleY)
 | 
						|
  else
 | 
						|
    sf := cairo_pdf_surface_create(PChar(FOutputFileName), PaperWidth*ScaleX, PaperHeight*ScaleY);
 | 
						|
  result := {%H-}HDC(cairo_create(sf));
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPdfCanvas.UpdatePageSize;
 | 
						|
begin
 | 
						|
  cairo_pdf_surface_set_size(sf, PaperWidth*ScaleX, PaperHeight*ScaleY);
 | 
						|
end;
 | 
						|
 | 
						|
{ TCairoPsCanvas }
 | 
						|
 | 
						|
function TCairoPsCanvas.CreateCairoHandle: HDC;
 | 
						|
var
 | 
						|
  s: string;
 | 
						|
  W, H: Double;
 | 
						|
  acr: Pcairo_t;
 | 
						|
begin
 | 
						|
  if Orientation in [poLandscape, poReverseLandscape] then begin
 | 
						|
    s := '%%PageOrientation: Landscape';
 | 
						|
    W := PaperHeight*ScaleY; //switch H, W
 | 
						|
    H := PaperWidth*ScaleX;
 | 
						|
  end else begin
 | 
						|
    s := '%%PageOrientation: Portait';
 | 
						|
    W := PaperWidth*ScaleX;
 | 
						|
    H := PaperHeight*ScaleY;
 | 
						|
  end;
 | 
						|
 | 
						|
  //Sizes are in Points, 72DPI (1pt = 1/72")
 | 
						|
  if fStream<>nil then
 | 
						|
    sf := cairo_ps_surface_create_for_stream(@WriteToStream, fStream, W, H)
 | 
						|
  else
 | 
						|
    sf := cairo_ps_surface_create(PChar(FOutputFileName), W, H);
 | 
						|
  acr := cairo_create(sf);
 | 
						|
 | 
						|
  cairo_ps_surface_dsc_begin_setup(sf);
 | 
						|
  cairo_ps_surface_dsc_comment(sf, PChar(s));
 | 
						|
 | 
						|
  if (fStream=nil) and not FileExists(FOutputFileName) then
 | 
						|
  begin
 | 
						|
    DebugLn('Error: unable to write cairo ps to "'+FOutputFileName+'"');
 | 
						|
    DestroyCairoHandle;
 | 
						|
    exit(0);
 | 
						|
  end;
 | 
						|
 | 
						|
  //rotate and move
 | 
						|
  case Orientation of
 | 
						|
    poLandscape: begin
 | 
						|
      cairo_translate(acr, 0, H);
 | 
						|
      cairo_rotate(acr, -PI/2);
 | 
						|
    end;
 | 
						|
    poReverseLandscape: begin
 | 
						|
      cairo_translate(acr, W, 0);
 | 
						|
      cairo_rotate(acr, PI/2);
 | 
						|
    end;
 | 
						|
    poReversePortrait: begin
 | 
						|
      cairo_translate(acr, W, H);
 | 
						|
      cairo_rotate(acr, PI);
 | 
						|
    end;
 | 
						|
  end;
 | 
						|
  result := {%H-}HDC(acr);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPsCanvas.UpdatePageSize;
 | 
						|
begin
 | 
						|
  cairo_ps_surface_set_size(sf, PaperWidth*ScaleX, PaperHeight*ScaleY);
 | 
						|
end;
 | 
						|
 | 
						|
constructor TCairoPngCanvas.Create(APrinter: TPrinter);
 | 
						|
begin
 | 
						|
  inherited Create(APrinter);
 | 
						|
end;
 | 
						|
 | 
						|
{ TCairoSvgCanvas }
 | 
						|
 | 
						|
function TCairoSvgCanvas.CreateCairoHandle: HDC;
 | 
						|
begin
 | 
						|
  //Sizes are in Points, 72DPI (1pt = 1/72")
 | 
						|
  sf := cairo_svg_surface_create(PChar(FOutputFileName), PaperWidth*ScaleX, PaperHeight*ScaleY);
 | 
						|
  result := {%H-}HDC(cairo_create(sf));
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
{ TCairoPngCanvas }
 | 
						|
 | 
						|
function TCairoPngCanvas.CreateCairoHandle: HDC;
 | 
						|
var
 | 
						|
  acr: Pcairo_t;
 | 
						|
begin
 | 
						|
  //I do not know how to retrieve DPI of cairo_image_surface
 | 
						|
  //It looks like that Cairo uses same DPI as Screen, but how much is it in case of console app???
 | 
						|
  //You must set Surface?DPI externally. For example:
 | 
						|
  //c := TCairoPngCanvas.Create;
 | 
						|
  //c.SurfaceXDPI := GetDeviceCaps(DC, LOGPIXELSX);
 | 
						|
  //c.SurfaceYDPI := GetDeviceCaps(DC, LOGPIXELSY);
 | 
						|
  sf := cairo_image_surface_create(CAIRO_FORMAT_ARGB32, PaperWidth, PaperHeight);
 | 
						|
  acr := cairo_create(sf);
 | 
						|
  cairo_scale(acr, 1/ScaleX, 1/ScaleY);
 | 
						|
  result := {%H-}HDC(acr);
 | 
						|
end;
 | 
						|
 | 
						|
procedure TCairoPngCanvas.DestroyCairoHandle;
 | 
						|
begin
 | 
						|
  cairo_surface_write_to_png(sf, PChar(FOutputFileName));
 | 
						|
  inherited DestroyCairoHandle;
 | 
						|
end;
 | 
						|
 | 
						|
end.
 | 
						|
 |