lazarus/components/tachart/tadrawerfpcanvas.pas

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.