mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-27 17:53:57 +02:00
439 lines
11 KiB
ObjectPascal
439 lines
11 KiB
ObjectPascal
{
|
|
*****************************************************************************
|
|
See the file COPYING.modifiedLGPL.txt, included in this distribution,
|
|
for details about the license.
|
|
*****************************************************************************
|
|
|
|
Authors: Alexander Klenin
|
|
|
|
}
|
|
unit TADrawerFPCanvas;
|
|
|
|
{$MODE ObjFPC}{$H+}
|
|
|
|
interface
|
|
|
|
{$DEFINE USE_FTFONT}
|
|
{$IF (FPC_VERSION = 2) and (FPC_RELEASE <= 4) and defined(WIN64)}
|
|
{$UNDEF USE_FTFONT}
|
|
{$ENDIF}
|
|
|
|
uses
|
|
Classes, FPCanvas, FPImage, {$IFDEF USE_FTFONT}FTFont,{$ENDIF}
|
|
TAChartUtils, TADrawUtils;
|
|
|
|
type
|
|
|
|
{ TFPCanvasDrawer }
|
|
|
|
TFPCanvasDrawer = class(TBasicDrawer, IChartDrawer)
|
|
strict private
|
|
FCanvas: TFPCustomCanvas;
|
|
{$IFDEF USE_FTFONT}
|
|
FFont: TFreeTypeFont;
|
|
FMeasureFont: TFreeTypeFont;
|
|
// FreeType measures text size in rotated orientation. But we need it
|
|
// in horizontal horientation. FMeasureFont is always horizontal.
|
|
{$ENDIF}
|
|
procedure EnsureFont;
|
|
procedure SetBrush(ABrush: TFPCustomBrush);
|
|
procedure SetFont(AFont: TFPCustomFont);
|
|
procedure SetPen(APen: TFPCustomPen);
|
|
strict protected
|
|
function SimpleTextExtent(const AText: String): TPoint; override;
|
|
procedure SimpleTextOut(AX, AY: Integer; const AText: String); override;
|
|
public
|
|
constructor Create(ACanvas: TFPCustomCanvas);
|
|
destructor Destroy; override;
|
|
public
|
|
procedure AddToFontOrientation(ADelta: Integer);
|
|
procedure ClippingStart;
|
|
procedure ClippingStart(const AClipRect: TRect);
|
|
procedure ClippingStop;
|
|
procedure Ellipse(AX1, AY1, AX2, AY2: Integer);
|
|
procedure FillRect(AX1, AY1, AX2, AY2: Integer);
|
|
function GetBrushColor: TChartColor;
|
|
function GetFontAngle: Double; override;
|
|
function GetFontColor: TFPColor; override;
|
|
function GetFontName: String; override;
|
|
function GetFontSize: Integer; override;
|
|
function GetFontStyle: TChartFontStyles; override;
|
|
function GetPenColor: TChartColor;
|
|
procedure Line(AX1, AY1, AX2, AY2: Integer);
|
|
procedure Line(const AP1, AP2: TPoint);
|
|
procedure LineTo(AX, AY: Integer); override;
|
|
procedure MoveTo(AX, AY: Integer); override;
|
|
procedure Polygon(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer); override;
|
|
procedure Polyline(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer);
|
|
procedure PrepareSimplePen(AColor: TChartColor);
|
|
procedure PutPixel(AX, AY: Integer; AColor: TChartColor); override;
|
|
procedure RadialPie(
|
|
AX1, AY1, AX2, AY2: Integer;
|
|
AStartAngle16Deg, AAngleLength16Deg: Integer);
|
|
procedure Rectangle(const ARect: TRect);
|
|
procedure Rectangle(AX1, AY1, AX2, AY2: Integer);
|
|
procedure ResetFont;
|
|
procedure SetBrushColor(AColor: TChartColor);
|
|
procedure SetBrushParams(AStyle: TFPBrushStyle; AColor: TChartColor);
|
|
procedure SetPenColor(AColor: TChartColor);
|
|
procedure SetPenParams(AStyle: TFPPenStyle; AColor: TChartColor; AWidth: Integer = 1);
|
|
procedure SetPenWidth(AWidth: Integer);
|
|
end;
|
|
|
|
procedure InitFonts;
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
SysUtils, TAGeometry;
|
|
|
|
type
|
|
TFPCanvasHelperCrack = class(TFPCanvasHelper);
|
|
|
|
procedure AssignFPCanvasHelper(ADest, ASrc: TFPCanvasHelper);
|
|
var
|
|
d: TFPCanvasHelperCrack absolute ADest;
|
|
begin
|
|
d.DoCopyProps(ASrc);
|
|
end;
|
|
|
|
{ TFPCanvasDrawer }
|
|
|
|
procedure TFPCanvasDrawer.AddToFontOrientation(ADelta: Integer);
|
|
begin
|
|
EnsureFont;
|
|
{$IFDEF USE_FTFONT}
|
|
FFont.Angle := FFont.Angle + OrientToRad(ADelta);
|
|
{$ELSE}
|
|
Unused(ADelta);
|
|
{$ENDIF}
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.ClippingStart(const AClipRect: TRect);
|
|
begin
|
|
Unused(AClipRect);
|
|
FCanvas.ClipRect := AClipRect;
|
|
ClippingStart;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.ClippingStart;
|
|
begin
|
|
// FIXME: FPCanvas.Clipping is broken
|
|
// FCanvas.Clipping := true;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.ClippingStop;
|
|
begin
|
|
FCanvas.Clipping := false;
|
|
end;
|
|
|
|
constructor TFPCanvasDrawer.Create(ACanvas: TFPCustomCanvas);
|
|
begin
|
|
inherited Create;
|
|
FCanvas := ACanvas;
|
|
end;
|
|
|
|
destructor TFPCanvasDrawer.Destroy;
|
|
begin
|
|
{$IFDEF USE_FTFONT}
|
|
FreeAndNil(FFont);
|
|
FreeAndNil(FMeasureFont);
|
|
{$ENDIF}
|
|
inherited Destroy;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Ellipse(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
FCanvas.Ellipse(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.EnsureFont;
|
|
begin
|
|
{$IFDEF USE_FTFONT}
|
|
if FFont = nil then begin
|
|
FFont := TFreeTypeFont.Create;
|
|
FFont.Size := DEFAULT_FONT_SIZE;
|
|
FFont.Resolution := 96;
|
|
FFont.AntiAliased := true; //false;
|
|
FCanvas.Font := FFont;
|
|
end;
|
|
|
|
if FMeasureFont = nil then begin
|
|
FMeasureFont := TFreeTypeFont.Create;
|
|
FMeasureFont.Size := DEFAULT_FONT_SIZE;
|
|
FMeasureFont.Resolution := 96;
|
|
FMeasureFont.AntiAliased := false;
|
|
end;
|
|
{$ENDIF}
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.FillRect(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
// FIXME
|
|
FCanvas.Rectangle(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetBrushColor: TChartColor;
|
|
begin
|
|
Result := FPColorToChartColor(FCanvas.Brush.FPColor);
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetFontAngle: Double;
|
|
begin
|
|
{$IFDEF USE_FTFONT}
|
|
Result := FFont.Angle; // Freetype font angle is in rad.
|
|
{$ELSE}
|
|
Result := 0;
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetFontColor: TFPColor;
|
|
begin
|
|
Result := FCanvas.Font.FPColor;
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetFontName: String;
|
|
begin
|
|
Result := FCanvas.Font.Name;
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetFontSize: Integer;
|
|
begin
|
|
if FCanvas.Font.Size = 0 then
|
|
Result := DEFAULT_FONT_SIZE
|
|
else
|
|
Result := FCanvas.Font.Size;
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetFontStyle: TChartFontStyles;
|
|
begin
|
|
Result := [];
|
|
if FCanvas.Font.Bold then Include(Result, cfsBold);
|
|
if FCanvas.Font.Italic then Include(Result, cfsItalic);
|
|
if FCanvas.Font.Underline then Include(Result, cfsUnderline);
|
|
if FCanvas.Font.Strikethrough then Include(Result, cfsStrikeout);
|
|
end;
|
|
|
|
function TFPCanvasDrawer.GetPenColor: TChartColor;
|
|
begin
|
|
Result := FPColorToChartColor(FCanvas.Pen.FPColor);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Line(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
FCanvas.Line(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Line(const AP1, AP2: TPoint);
|
|
begin
|
|
FCanvas.Line(AP1, AP2);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.LineTo(AX, AY: Integer);
|
|
begin
|
|
FCanvas.LineTo(AX, AY);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.MoveTo(AX, AY: Integer);
|
|
begin
|
|
FCanvas.MoveTo(AX, AY);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Polygon(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer);
|
|
begin
|
|
if (ANumPts = Length(APoints)) and (AStartIndex = 0) then
|
|
FCanvas.Polygon(APoints)
|
|
else
|
|
FCanvas.Polygon(CopyPoints(APoints, AStartIndex, ANumPts));
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Polyline(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer);
|
|
begin
|
|
if (ANumPts = Length(APoints)) and (AStartIndex = 0) then
|
|
FCanvas.Polyline(APoints)
|
|
else
|
|
FCanvas.Polyline(CopyPoints(APoints, AStartIndex, ANumPts));
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.PrepareSimplePen(AColor: TChartColor);
|
|
begin
|
|
FCanvas.Pen.FPColor := FChartColorToFPColorFunc(AColor);
|
|
FCanvas.Pen.Width := 1;
|
|
FCanvas.Pen.Style := psSolid;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.PutPixel(AX, AY: Integer; AColor: TChartColor);
|
|
begin
|
|
FCanvas.Colors[AX, AY] := FChartColorToFPColorFunc(AColor);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.RadialPie(
|
|
AX1, AY1, AX2, AY2: Integer; AStartAngle16Deg, AAngleLength16Deg: Integer);
|
|
var
|
|
e: TEllipse;
|
|
p: TPointArray;
|
|
begin
|
|
e.InitBoundingBox(AX1, AY1, AX2, AY2);
|
|
p := e.TesselateRadialPie(
|
|
Deg16ToRad(AStartAngle16Deg), Deg16ToRad(AAngleLength16Deg), 4);
|
|
Polygon(p, 0, Length(p));
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Rectangle(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
FCanvas.Rectangle(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.Rectangle(const ARect: TRect);
|
|
begin
|
|
FCanvas.Rectangle(ARect);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.ResetFont;
|
|
begin
|
|
FCanvas.Font.Orientation := 0;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetBrush(ABrush: TFPCustomBrush);
|
|
begin
|
|
AssignFPCanvasHelper(FCanvas.Brush, ABrush);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetBrushColor(AColor: TChartColor);
|
|
begin
|
|
FCanvas.Brush.FPColor := FChartColorToFPColorFunc(AColor);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetBrushParams(
|
|
AStyle: TFPBrushStyle; AColor: TChartColor);
|
|
begin
|
|
FCanvas.Brush.FPColor := FChartColorToFPColorFunc(AColor);
|
|
FCanvas.Brush.Style := AStyle;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetFont(AFont: TFPCustomFont);
|
|
begin
|
|
EnsureFont;
|
|
{$IFDEF USE_FTFONT}
|
|
AssignFPCanvasHelper(FFont, AFont);
|
|
AssignFPCanvasHelper(FMeasureFont, AFont);
|
|
// DoCopyProps performs direct variable assignment, so call SetName by hand.
|
|
FFont.Name := AFont.Name;
|
|
if AFont.Size = 0 then
|
|
FFont.Size := DEFAULT_FONT_SIZE
|
|
else
|
|
FFont.Size := AFont.Size;
|
|
FFont.Angle := OrientToRad(FGetFontOrientationFunc(AFont));
|
|
FMeasureFont.Name := FFont.Name;
|
|
FMeasureFont.Size := FFont.Size;
|
|
FMeasureFont.Angle := 0;
|
|
{$ELSE}
|
|
Unused(AFont);
|
|
{$ENDIF}
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetPen(APen: TFPCustomPen);
|
|
begin
|
|
AssignFPCanvasHelper(FCanvas.Pen, APen);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetPenColor(AColor: TChartColor);
|
|
begin
|
|
FCanvas.Pen.FPColor := FChartColorToFPColorFunc(AColor);
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetPenParams(AStyle: TFPPenStyle; AColor: TChartColor;
|
|
AWidth: Integer = 1);
|
|
begin
|
|
FCanvas.Pen.Style := AStyle;
|
|
FCanvas.Pen.FPColor := FChartColorToFPColorFunc(AColor);
|
|
FCanvas.Pen.Width := AWidth;
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SetPenWidth(AWidth: Integer);
|
|
begin
|
|
FCanvas.Pen.Width := AWidth;
|
|
end;
|
|
|
|
function TFPCanvasDrawer.SimpleTextExtent(const AText: String): TPoint;
|
|
begin
|
|
Result := Point(0, 0);
|
|
EnsureFont;
|
|
{$IFDEF USE_FTFONT}
|
|
FCanvas.Font := FMeasureFont;
|
|
FMeasureFont.GetTextSize(AText, Result.X, Result.Y);
|
|
FCanvas.Font := FFont;
|
|
// FreeType measures the exact pixel height of characters. But the LCL font
|
|
// has some space above and below. --> increase height bei 25%
|
|
Result.Y := Result.Y * 5 div 4;
|
|
{$ELSE}
|
|
FCanvas.GetTextSize(AText, Result.X, Result.Y);
|
|
{$ENDIF}
|
|
end;
|
|
|
|
procedure TFPCanvasDrawer.SimpleTextOut(AX, AY: Integer; const AText: String);
|
|
{$IFDEF USE_FTFONT}
|
|
var
|
|
p: TPoint;
|
|
h: Integer;
|
|
{$ENDIF}
|
|
begin
|
|
EnsureFont;
|
|
{$IFDEF USE_FTFONT}
|
|
// FreeType uses lower-left instead of upper-left corner as starting position.
|
|
// --> we must correct for text height to find correct text starting point
|
|
h := SimpleTextExtent('Tg').y;
|
|
// Approximately correct for the difference in text height in LCL
|
|
// (incl enpty space below character) and freetype (exact text height).
|
|
h := h * 3 div 4;
|
|
// Rotate text height according to font direction
|
|
p := RotatePoint(Point(0, h), -FFont.Angle);
|
|
FCanvas.TextOut(p.X + AX, p.Y + AY, AText);
|
|
{$ELSE}
|
|
Unused(AX, AY);
|
|
Unused(AText); // wp: why not call FCanvas.TextOut ???
|
|
{$ENDIF}
|
|
end;
|
|
|
|
procedure InitFonts;
|
|
begin
|
|
{$IFDEF USE_FTFONT}
|
|
InitEngine;
|
|
|
|
// Windows: Search path is set automatically
|
|
|
|
{$IFDEF LINUX}
|
|
FontMgr.SearchPath :=
|
|
'/usr/share/cups/fonts/;' +
|
|
'/usr/share/fonts/truetype/;' +
|
|
'/usr/share/fonts/truetype/liberation/;' +
|
|
'/usr/local/lib/X11/fonts/;' +
|
|
GetUserDir + '.fonts/';
|
|
{$ENDIF}
|
|
|
|
{$IFDEF LCLCarbon}
|
|
FontMgr.SearchPath :=
|
|
'/Library/Fonts/;' +
|
|
'/System/Library/Fonts/;' +
|
|
'/Network/Library/Fonts/;' +
|
|
'~/Library/Fonts/';
|
|
{$ENDIF}
|
|
|
|
{$IFDEF LCLCocoa}
|
|
FontMgr.SearchPath :=
|
|
'/Library/Fonts/;' +
|
|
'/System/Library/Fonts/;' +
|
|
'/Network/Library/Fonts/;' +
|
|
'~/Library/Fonts/';
|
|
{$ENDIF}
|
|
{$ENDIF}
|
|
end;
|
|
|
|
end.
|
|
|