mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-27 00:03:48 +02:00
573 lines
15 KiB
ObjectPascal
573 lines
15 KiB
ObjectPascal
{
|
|
*****************************************************************************
|
|
See the file COPYING.modifiedLGPL.txt, included in this distribution,
|
|
for details about the license.
|
|
*****************************************************************************
|
|
|
|
Authors: Alexander Klenin
|
|
|
|
}
|
|
|
|
unit TADrawerCanvas;
|
|
|
|
{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, FPCanvas, FPImage, Graphics, SysUtils, TAChartUtils, TADrawUtils;
|
|
|
|
type
|
|
IChartTCanvasDrawer = interface
|
|
['{6D8E5591-6788-4D2D-9FE6-596D5157C3C2}']
|
|
function GetCanvas: TCanvas;
|
|
property Canvas: TCanvas read GetCanvas;
|
|
end;
|
|
|
|
{ TCanvasDrawer }
|
|
|
|
TCanvasDrawer = class(
|
|
TBasicDrawer, IChartDrawer, IChartTCanvasDrawer)
|
|
strict private
|
|
procedure SetBrush(ABrush: TFPCustomBrush);
|
|
procedure SetFont(AFont: TFPCustomFont);
|
|
procedure SetPen(APen: TFPCustomPen);
|
|
strict protected
|
|
FCanvas: TCanvas;
|
|
FBuffer: TBitmap;
|
|
// function GetFontAngle: Double; override;
|
|
function SimpleTextExtent(const AText: String): TPoint; override;
|
|
procedure SimpleTextOut(AX, AY: Integer; const AText: String); override;
|
|
public
|
|
procedure AddToFontOrientation(ADelta: Integer);
|
|
procedure ClippingStart;
|
|
procedure ClippingStart(const AClipRect: TRect);
|
|
procedure ClippingStop;
|
|
constructor Create(ACanvas: TCanvas);
|
|
destructor Destroy; override;
|
|
procedure Ellipse(AX1, AY1, AX2, AY2: Integer);
|
|
procedure FillRect(AX1, AY1, AX2, AY2: Integer);
|
|
function GetBrushColor: TChartColor;
|
|
function GetCanvas: TCanvas; virtual;
|
|
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 PutImage(AX, AY: Integer; AImage: TFPCustomImage); override;
|
|
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 SetAntialiasingMode(AValue: TChartAntialiasingMode);
|
|
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);
|
|
procedure SetTransparency(ATransparency: TChartTransparency);
|
|
end;
|
|
|
|
TScaledCanvasDrawer = class(TCanvasDrawer)
|
|
protected
|
|
FCoeff: Double;
|
|
public
|
|
constructor Create(ACanvas: TCanvas; ACoeff: Double; AScaleItems: TScaleItems);
|
|
function Scale(ADistance: Integer): Integer; override;
|
|
end;
|
|
|
|
function CanvasGetFontOrientationFunc(AFont: TFPCustomFont): Integer;
|
|
function ChartColorSysToFPColor(AChartColor: TChartColor): TFPColor;
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
GraphType, LCLIntf, LCLType, IntfGraphics,
|
|
TAGeometry;
|
|
|
|
function CanvasGetFontOrientationFunc(AFont: TFPCustomFont): Integer;
|
|
begin
|
|
if AFont is TFont then
|
|
Result := TFont(AFont).Orientation
|
|
else
|
|
Result := AFont.Orientation; //0;
|
|
end;
|
|
|
|
function ChartColorSysToFPColor(AChartColor: TChartColor): TFPColor;
|
|
begin
|
|
Result := ChartColorToFPColor(ColorToRGB(AChartColor));
|
|
end;
|
|
|
|
{ TCanvasDrawer }
|
|
|
|
procedure TCanvasDrawer.AddToFontOrientation(ADelta: Integer);
|
|
begin
|
|
with GetCanvas.Font do
|
|
Orientation := Orientation + ADelta;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.ClippingStart(const AClipRect: TRect);
|
|
begin
|
|
FCanvas.ClipRect := AClipRect;
|
|
FBuffer.Canvas.ClipRect := AClipRect;
|
|
ClippingStart;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.ClippingStart;
|
|
begin
|
|
FCanvas.Clipping := true;
|
|
FBuffer.Canvas.Clipping := true;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.ClippingStop;
|
|
begin
|
|
FCanvas.Clipping := false;
|
|
FBuffer.Canvas.Clipping := false;
|
|
end;
|
|
|
|
constructor TCanvasDrawer.Create(ACanvas: TCanvas);
|
|
begin
|
|
inherited Create;
|
|
FCanvas := ACanvas;
|
|
FBuffer := TBitmap.Create;
|
|
FBuffer.PixelFormat := pf32bit;
|
|
end;
|
|
|
|
destructor TCanvasDrawer.Destroy;
|
|
begin
|
|
FreeAndNil(FBuffer);
|
|
inherited;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Ellipse(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
GetCanvas.Ellipse(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.FillRect(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
GetCanvas.FillRect(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
function TCanvasDrawer.GetBrushColor: TChartColor;
|
|
begin
|
|
Result := GetCanvas.Brush.Color;
|
|
end;
|
|
|
|
function TCanvasDrawer.GetCanvas: TCanvas;
|
|
begin
|
|
// When transparency is off, draw directly on canvas for better speed.
|
|
if FTransparency > 0 then
|
|
Result := FBuffer.Canvas
|
|
else
|
|
Result := FCanvas;
|
|
end;
|
|
|
|
function TCanvasDrawer.GetFontAngle: Double;
|
|
begin
|
|
Result := OrientToRad(GetCanvas.Font.Orientation);
|
|
end;
|
|
|
|
function TCanvasDrawer.GetFontColor: TFPColor;
|
|
begin
|
|
Result := TColorToFPColor(GetCanvas.Font.Color);
|
|
end;
|
|
|
|
function TCanvasDrawer.GetFontName: String;
|
|
begin
|
|
Result := GetCanvas.Font.Name;
|
|
end;
|
|
|
|
function TCanvasDrawer.GetFontSize: Integer;
|
|
var
|
|
h: Integer;
|
|
begin
|
|
Result := GetCanvas.Font.Size;
|
|
if Result = 0 then begin
|
|
h := GetFontData(GetCanvas.Font.Reference.Handle).Height;
|
|
Result := round(abs(h) * 72 / ScreenInfo.PixelsPerInchY);
|
|
end;
|
|
end;
|
|
|
|
function TCanvasDrawer.GetFontStyle: TChartFontStyles;
|
|
begin
|
|
Result := TChartFontStyles(GetCanvas.Font.Style);
|
|
end;
|
|
|
|
function TCanvasDrawer.GetPenColor: TChartColor;
|
|
begin
|
|
Result := GetCanvas.Pen.Color;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Line(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
GetCanvas.Line(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Line(const AP1, AP2: TPoint);
|
|
begin
|
|
GetCanvas.Line(AP1, AP2);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.LineTo(AX, AY: Integer);
|
|
begin
|
|
GetCanvas.LineTo(AX, AY);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.MoveTo(AX, AY: Integer);
|
|
begin
|
|
GetCanvas.MoveTo(AX, AY);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Polygon(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer);
|
|
begin
|
|
GetCanvas.Polygon(APoints, false, AStartIndex, ANumPts);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Polyline(
|
|
const APoints: array of TPoint; AStartIndex, ANumPts: Integer);
|
|
begin
|
|
if ANumPts <= 0 then exit;
|
|
GetCanvas.Polyline(APoints, AStartIndex, ANumPts);
|
|
// TCanvas.Polyline does not draw the end point.
|
|
with APoints[AStartIndex + ANumPts - 1] do
|
|
GetCanvas.Pixels[X, Y] := GetCanvas.Pen.Color;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.PrepareSimplePen(AColor: TChartColor);
|
|
begin
|
|
with GetCanvas.Pen do begin
|
|
if FXor then
|
|
Color := clWhite
|
|
else
|
|
Color := ColorOrMono(AColor);
|
|
Style := psSolid;
|
|
if FXor then
|
|
Mode := pmXor
|
|
else
|
|
Mode := pmCopy;
|
|
if (scalePen in FScaleItems) then
|
|
Width := Scale(1) else
|
|
Width := 1;
|
|
end;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.PutImage(AX, AY: Integer; AImage: TFPCustomImage);
|
|
var
|
|
x, y: Integer;
|
|
bmp: TBitmap;
|
|
begin
|
|
bmp := TBitmap.Create;
|
|
try
|
|
if AImage is TLazIntfImage then
|
|
bmp.LoadFromIntfImage(TLazIntfImage(AImage))
|
|
else begin
|
|
bmp.SetSize(AImage.Width, AImage.Height);
|
|
bmp.Transparent := true;
|
|
bmp.TransparentMode := tmFixed;
|
|
bmp.TransparentColor := bmp.Canvas.Pixels[0, 0];
|
|
for y := 0 to AImage.Height - 1 do
|
|
for x := 0 to AImage.Width - 1 do
|
|
if AImage[x, y].alpha > 0 then
|
|
bmp.Canvas.Colors[x, y] := AImage[x, y];
|
|
end;
|
|
GetCanvas.Draw(AX, AY, bmp);
|
|
finally
|
|
bmp.Free;
|
|
end;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.PutPixel(AX, AY: Integer; AColor: TChartColor);
|
|
begin
|
|
GetCanvas.Pixels[AX, AY] := AColor;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.RadialPie(
|
|
AX1, AY1, AX2, AY2: Integer;
|
|
AStartAngle16Deg, AAngleLength16Deg: Integer);
|
|
begin
|
|
GetCanvas.RadialPie(
|
|
AX1, AY1, AX2, AY2, AStartAngle16Deg, AAngleLength16Deg);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Rectangle(AX1, AY1, AX2, AY2: Integer);
|
|
begin
|
|
GetCanvas.Rectangle(AX1, AY1, AX2, AY2);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.Rectangle(const ARect: TRect);
|
|
begin
|
|
GetCanvas.Rectangle(ARect);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.ResetFont;
|
|
begin
|
|
GetCanvas.Font.Orientation := 0;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetAntialiasingMode(AValue: TChartAntialiasingMode);
|
|
begin
|
|
GetCanvas.AntialiasingMode := TAntialiasingMode(AValue);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetBrush(ABrush: TFPCustomBrush);
|
|
begin
|
|
with GetCanvas.Brush do begin
|
|
if ABrush is TBrush then
|
|
Assign(ABrush)
|
|
else begin
|
|
FPColor := ABrush.FPColor;
|
|
Pattern := ABrush.Pattern;
|
|
Style := ABrush.Style;
|
|
end;
|
|
if FXor then
|
|
Style := bsClear
|
|
else if FMonochromeColor <> clTAColor then
|
|
Color := FMonochromeColor;
|
|
end;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetBrushColor(AColor: TChartColor);
|
|
begin
|
|
GetCanvas.Brush.Color := ColorOrMono(AColor);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetBrushParams(
|
|
AStyle: TFPBrushStyle; AColor: TChartColor);
|
|
begin
|
|
GetCanvas.Brush.Color := ColorOrMono(AColor);
|
|
GetCanvas.Brush.Style := AStyle;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetFont(AFont: TFPCustomFont);
|
|
var
|
|
st: TFontStyles = [];
|
|
begin
|
|
with GetCanvas.Font do begin
|
|
if AFont is TFont then
|
|
Assign(AFont)
|
|
else begin
|
|
BeginUpdate;
|
|
FPColor := AFont.FPColor;
|
|
Name := AFont.Name;
|
|
Size := AFont.Size;
|
|
Orientation := AFont.Orientation;
|
|
if AFont.Italic then
|
|
Include(st, fsItalic);
|
|
if AFont.Bold then
|
|
Include(st, fsBold);
|
|
if AFont.Underline then
|
|
Include(st, fsUnderline);
|
|
if AFont.StrikeThrough then
|
|
Include(st, fsStrikeOut);
|
|
Style := st;
|
|
EndUpdate;
|
|
end;
|
|
if FMonochromeColor <> clTAColor then
|
|
Color := FMonochromeColor;
|
|
if scaleFont in FScaleItems then
|
|
Size := Scale(GetFontSize)
|
|
else
|
|
Size := GetFontSize;
|
|
end;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetPen(APen: TFPCustomPen);
|
|
begin
|
|
with GetCanvas do begin
|
|
if FXor then begin
|
|
Brush.Style := bsClear;
|
|
if APen = nil then
|
|
Pen.Style := psSolid
|
|
else
|
|
Pen.Style := APen.Style;
|
|
Pen.Mode := pmXor;
|
|
Pen.Color := clWhite;
|
|
if APen = nil then
|
|
Pen.Width := 1
|
|
else
|
|
Pen.Width := APen.Width;
|
|
end
|
|
else begin
|
|
if APen is TPen then
|
|
Pen.Assign(APen)
|
|
else begin
|
|
Pen.Color := FPColorToChartColor(APen.FPColor);
|
|
Pen.Style := APen.Style;
|
|
Pen.Width := APen.Width;
|
|
Pen.Mode := APen.Mode;
|
|
Pen.Pattern := APen.Pattern;
|
|
end;
|
|
if FMonochromeColor <> clTAColor then
|
|
Pen.Color := FMonochromeColor;
|
|
end;
|
|
if scalePen in FScaleItems then
|
|
Pen.Width := Scale(Pen.Width);
|
|
end;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetPenColor(AColor: TChartColor);
|
|
begin
|
|
if not FXor then
|
|
GetCanvas.Pen.Color := ColorOrMono(AColor);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetPenParams(AStyle: TFPPenStyle; AColor: TChartColor;
|
|
AWidth: Integer = 1);
|
|
begin
|
|
GetCanvas.Pen.Style := AStyle;
|
|
GetCanvas.Pen.Width := AWidth;
|
|
if not FXor then
|
|
GetCanvas.Pen.Color := ColorOrMono(AColor);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetPenWidth(AWidth: Integer);
|
|
begin
|
|
GetCanvas.Pen.Width := AWidth;
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SetTransparency(ATransparency: TChartTransparency);
|
|
|
|
function FillAlpha(AAlpha: Byte): Byte;
|
|
var
|
|
img: TRawImage;
|
|
p, pEnd: PCardinal;
|
|
x: Cardinal = 0;
|
|
r: Cardinal = 0;
|
|
begin
|
|
FBuffer.BeginUpdate;
|
|
img := FBuffer.RawImage;
|
|
p := PCardinal(img.Data);
|
|
TRGBAQuad(x).Alpha := AAlpha;
|
|
pEnd := PCardinal(img.Data + img.DataSize);
|
|
// This loop is time-critical, so: avoid conditionals inside,
|
|
// use dword-sized instead of byte-sized access.
|
|
while p < pEnd do begin
|
|
// On the first pass, set all alpha values to AAlpha.
|
|
// Drawing will reset alpha of changed pixels to zero.
|
|
// On the second pass, flip unchanged pixels back to zero alpha,
|
|
// and changed ones to the desired alpha level.
|
|
p^ := p^ xor x;
|
|
r := r or p^;
|
|
Inc(p);
|
|
end;
|
|
FBuffer.EndUpdate;
|
|
Result := TRGBAQuad(r).Alpha;
|
|
end;
|
|
|
|
begin
|
|
if FTransparency = ATransparency then exit;
|
|
// For each transparency change, create a buffer bitmap, draw on that,
|
|
// then alpha-blend the bitmap to the canvas.
|
|
// This is slow, but currently seems the only way.
|
|
if FTransparency > 0 then begin
|
|
// StretchMaskBlt performs alpha blending only if the image contains
|
|
// at least one non-zero alpha value, so fully transparent image
|
|
// becomes black box. Workround: do not call StretchMaskBlt in this case.
|
|
if FillAlpha(255 - FTransparency) > 0 then
|
|
StretchMaskBlt(
|
|
FCanvas.Handle, 0, 0, FCanvas.Width, FCanvas.Height,
|
|
FBuffer.Canvas.Handle, 0, 0, FCanvas.Width, FCanvas.Height,
|
|
0, 0, 0, SRCCOPY);
|
|
end;
|
|
inherited;
|
|
if FTransparency > 0 then begin
|
|
FBuffer.SetSize(0, 0);
|
|
FBuffer.SetSize(FCanvas.Width, FCanvas.Height);
|
|
FillAlpha(255 - FTransparency);
|
|
end;
|
|
end;
|
|
|
|
function TCanvasDrawer.SimpleTextExtent(const AText: String): TPoint;
|
|
begin
|
|
Result := GetCanvas.TextExtent(AText);
|
|
end;
|
|
|
|
procedure TCanvasDrawer.SimpleTextOut(AX, AY: Integer; const AText: String);
|
|
|
|
procedure DrawSimpleText(ACanvas: TCanvas; x, y: Integer; const txt: String);
|
|
// add right-to-left mode. Cannot use TextOut since it does not respect TextStyle
|
|
var
|
|
r: TRect;
|
|
ts: TTextStyle;
|
|
begin
|
|
ts := ACanvas.TextStyle;
|
|
ts.RightToLeft := FRightToLeft;
|
|
ts.WordBreak := false; // added to disable erroneous workbreaks in Linux printing
|
|
ts.Clipping := false;
|
|
r := Bounds(x, y, 1, 1);
|
|
ACanvas.TextRect(r, x, y, txt, ts);
|
|
end;
|
|
|
|
procedure DrawXorText;
|
|
var
|
|
bmp: TBitmap;
|
|
p, ext, bmpSize: TPoint;
|
|
a: Double;
|
|
begin
|
|
ext := GetCanvas.TextExtent(AText);
|
|
a := OrientToRad(GetCanvas.Font.Orientation);
|
|
bmpSize := MeasureRotatedRect(ext, a);
|
|
p := bmpSize div 2 - RotatePoint(ext div 2, -a);
|
|
|
|
bmp := TBitmap.Create;
|
|
try
|
|
bmp.SetSize(bmpSize.X, bmpSize.Y);
|
|
bmp.Canvas.Brush.Style := bsClear;
|
|
bmp.Canvas.Font := GetCanvas.Font;
|
|
bmp.Canvas.Font.Color := clWhite;
|
|
DrawSimpleText(bmp.Canvas, p.X, p.Y, AText);
|
|
bmp.Canvas.Pen.Color := clWhite;
|
|
BitBlt(
|
|
GetCanvas.Handle, AX - p.X, AY - p.Y, bmpSize.X, bmpSize.Y,
|
|
bmp.Canvas.Handle, 0, 0, SRCINVERT);
|
|
finally
|
|
bmp.Free;
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
if FXor then
|
|
DrawXorText
|
|
else
|
|
DrawSimpleText(GetCanvas, AX, AY, AText);
|
|
end;
|
|
|
|
|
|
{ TScaledCanvasDrawer }
|
|
|
|
constructor TScaledCanvasDrawer.Create(ACanvas: TCanvas; ACoeff: Double;
|
|
AScaleItems: TScaleItems);
|
|
begin
|
|
inherited Create(ACanvas);
|
|
FCoeff := ACoeff;
|
|
FScaleItems := AScaleItems;
|
|
end;
|
|
|
|
function TScaledCanvasDrawer.Scale(ADistance: Integer): Integer;
|
|
begin
|
|
Result := Round(FCoeff * ADistance);
|
|
end;
|
|
|
|
initialization
|
|
// Suppress incorrect "TAGeometry is unused" hint
|
|
Unused(DoublePoint(0, 0));
|
|
|
|
end.
|
|
|