mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-08-09 00:56:02 +02:00
TAChart: Refactor TADrawUtils unit
git-svn-id: trunk@29532 -
This commit is contained in:
parent
829ea72a9d
commit
2d3a007277
@ -89,12 +89,16 @@ type
|
|||||||
{ TChartTextOut }
|
{ TChartTextOut }
|
||||||
|
|
||||||
TChartTextOut = class
|
TChartTextOut = class
|
||||||
|
strict private
|
||||||
FAlignment: TAlignment;
|
FAlignment: TAlignment;
|
||||||
FDrawer: IChartDrawer;
|
FDrawer: IChartDrawer;
|
||||||
FPos: TPoint;
|
FPos: TPoint;
|
||||||
FText1: String;
|
FText1: String;
|
||||||
FText2: TStrings;
|
FText2: TStrings;
|
||||||
FWidth: Integer;
|
FWidth: Integer;
|
||||||
|
|
||||||
|
procedure DoTextOutString;
|
||||||
|
procedure DoTextOutList;
|
||||||
public
|
public
|
||||||
constructor Create(ADrawer: IChartDrawer);
|
constructor Create(ADrawer: IChartDrawer);
|
||||||
public
|
public
|
||||||
@ -143,25 +147,16 @@ type
|
|||||||
procedure DrawLineDepth(ACanvas: TCanvas; AX1, AY1, AX2, AY2, ADepth: Integer);
|
procedure DrawLineDepth(ACanvas: TCanvas; AX1, AY1, AX2, AY2, ADepth: Integer);
|
||||||
procedure DrawLineDepth(ACanvas: TCanvas; const AP1, AP2: TPoint; ADepth: Integer);
|
procedure DrawLineDepth(ACanvas: TCanvas; const AP1, AP2: TPoint; ADepth: Integer);
|
||||||
|
|
||||||
function MultiLineTextExtent(ACanvas: TCanvas; const AText: String): TPoint;
|
|
||||||
function MultiLineTextExtent(ACanvas: TCanvas; AText: TStrings): TPoint;
|
|
||||||
procedure MultiLineTextOut(
|
|
||||||
ACanvas: TCanvas; APos: TPoint; const AText: String;
|
|
||||||
AAlignment: TAlignment; AWidth: Integer);
|
|
||||||
procedure MultiLineTextOut(
|
|
||||||
ACanvas: TCanvas; APos: TPoint; AText: TStrings;
|
|
||||||
AAlignment: TAlignment; AWidth: Integer);
|
|
||||||
|
|
||||||
procedure PrepareSimplePen(ACanvas: TCanvas; AColor: TColor);
|
|
||||||
procedure PrepareXorPen(ACanvas: TCanvas);
|
procedure PrepareXorPen(ACanvas: TCanvas);
|
||||||
|
|
||||||
function TypicalTextHeight(ACanvas: TCanvas): Integer;
|
|
||||||
|
|
||||||
implementation
|
implementation
|
||||||
|
|
||||||
uses
|
uses
|
||||||
Math, TAChartUtils;
|
Math, TAChartUtils;
|
||||||
|
|
||||||
|
const
|
||||||
|
LINE_INTERVAL = 2;
|
||||||
|
|
||||||
procedure DrawLineDepth(ACanvas: TCanvas; AX1, AY1, AX2, AY2, ADepth: Integer);
|
procedure DrawLineDepth(ACanvas: TCanvas; AX1, AY1, AX2, AY2, ADepth: Integer);
|
||||||
begin
|
begin
|
||||||
DrawLineDepth(ACanvas, Point(AX1, AY1), Point(AX2, AY2), ADepth);
|
DrawLineDepth(ACanvas, Point(AX1, AY1), Point(AX2, AY2), ADepth);
|
||||||
@ -176,86 +171,6 @@ begin
|
|||||||
ACanvas.Polygon([AP1, AP1 + d, AP2 + d, AP2]);
|
ACanvas.Polygon([AP1, AP1 + d, AP2 + d, AP2]);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
const
|
|
||||||
LINE_INTERVAL = 2;
|
|
||||||
|
|
||||||
function MultiLineTextExtent(ACanvas: TCanvas; const AText: String): TPoint;
|
|
||||||
var
|
|
||||||
sl: TStrings;
|
|
||||||
begin
|
|
||||||
if Pos(LineEnding, AText) = 0 then
|
|
||||||
exit(ACanvas.TextExtent(AText));
|
|
||||||
sl := TStringList.Create;
|
|
||||||
try
|
|
||||||
sl.Text := AText;
|
|
||||||
Result := MultiLineTextExtent(ACanvas, sl);
|
|
||||||
finally
|
|
||||||
sl.Free;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
function MultiLineTextExtent(ACanvas: TCanvas; AText: TStrings): TPoint;
|
|
||||||
var
|
|
||||||
i: Integer;
|
|
||||||
begin
|
|
||||||
Result := Size(0, -LINE_INTERVAL);
|
|
||||||
for i := 0 to AText.Count - 1 do
|
|
||||||
with ACanvas.TextExtent(AText[i]) do begin
|
|
||||||
Result.X := Max(Result.X, cx);
|
|
||||||
Result.Y += cy + LINE_INTERVAL;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
procedure MultiLineTextOut(
|
|
||||||
ACanvas: TCanvas; APos: TPoint; const AText: String; AAlignment: TAlignment;
|
|
||||||
AWidth: Integer);
|
|
||||||
var
|
|
||||||
sl: TStrings;
|
|
||||||
begin
|
|
||||||
if Pos(LineEnding, AText) = 0 then begin
|
|
||||||
ACanvas.TextOut(APos.X, APos.Y, AText);
|
|
||||||
exit;
|
|
||||||
end;
|
|
||||||
sl := TStringList.Create;
|
|
||||||
try
|
|
||||||
sl.Text := AText;
|
|
||||||
MultiLineTextOut(ACanvas, APos, sl, AAlignment, AWidth);
|
|
||||||
finally
|
|
||||||
sl.Free;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
procedure MultiLineTextOut(
|
|
||||||
ACanvas: TCanvas; APos: TPoint; AText: TStrings; AAlignment: TAlignment;
|
|
||||||
AWidth: Integer);
|
|
||||||
var
|
|
||||||
i: Integer;
|
|
||||||
a: Double;
|
|
||||||
lineExtent, p: TPoint;
|
|
||||||
begin
|
|
||||||
a := -OrientToRad(ACanvas.Font.Orientation);
|
|
||||||
for i := 0 to AText.Count - 1 do begin
|
|
||||||
lineExtent := ACanvas.TextExtent(AText[i]);
|
|
||||||
p := APos;
|
|
||||||
case AAlignment of
|
|
||||||
taCenter: p += RotatePoint(Point((AWidth - lineExtent.X) div 2, 0), a);
|
|
||||||
taRightJustify: p += RotatePoint(Point(AWidth - lineExtent.X, 0), a);
|
|
||||||
end;
|
|
||||||
ACanvas.TextOut(p.X, p.Y, AText[i]);
|
|
||||||
APos += RotatePoint(Point(0, lineExtent.Y + LINE_INTERVAL), a);
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
procedure PrepareSimplePen(ACanvas: TCanvas; AColor: TColor);
|
|
||||||
begin
|
|
||||||
with ACanvas.Pen do begin
|
|
||||||
Color := AColor;
|
|
||||||
Style := psSolid;
|
|
||||||
Mode := pmCopy;
|
|
||||||
Width := 1;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
procedure PrepareXorPen(ACanvas: TCanvas);
|
procedure PrepareXorPen(ACanvas: TCanvas);
|
||||||
begin
|
begin
|
||||||
with ACanvas do begin
|
with ACanvas do begin
|
||||||
@ -267,13 +182,6 @@ begin
|
|||||||
end;
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function TypicalTextHeight(ACanvas: TCanvas): Integer;
|
|
||||||
const
|
|
||||||
TYPICAL_TEXT = 'Iy';
|
|
||||||
begin
|
|
||||||
Result := ACanvas.TextHeight(TYPICAL_TEXT);
|
|
||||||
end;
|
|
||||||
|
|
||||||
{ TChartTextOut }
|
{ TChartTextOut }
|
||||||
|
|
||||||
function TChartTextOut.Alignment(AAlignment: TAlignment): TChartTextOut;
|
function TChartTextOut.Alignment(AAlignment: TAlignment): TChartTextOut;
|
||||||
@ -291,12 +199,46 @@ end;
|
|||||||
procedure TChartTextOut.Done;
|
procedure TChartTextOut.Done;
|
||||||
begin
|
begin
|
||||||
if FText2 = nil then
|
if FText2 = nil then
|
||||||
MultiLineTextOut(FDrawer.Canvas, FPos, FText1, FAlignment, FWidth)
|
DoTextOutString
|
||||||
else
|
else
|
||||||
MultiLineTextOut(FDrawer.Canvas, FPos, FText2, FAlignment, FWidth);
|
DoTextOutList;
|
||||||
Free;
|
Free;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
procedure TChartTextOut.DoTextOutList;
|
||||||
|
var
|
||||||
|
i: Integer;
|
||||||
|
a: Double;
|
||||||
|
lineExtent, p: TPoint;
|
||||||
|
begin
|
||||||
|
a := -OrientToRad(FDrawer.Canvas.Font.Orientation);
|
||||||
|
for i := 0 to FText2.Count - 1 do begin
|
||||||
|
lineExtent := FDrawer.Canvas.TextExtent(FText2[i]);
|
||||||
|
p := FPos;
|
||||||
|
case FAlignment of
|
||||||
|
taCenter: p += RotatePoint(Point((FWidth - lineExtent.X) div 2, 0), a);
|
||||||
|
taRightJustify: p += RotatePoint(Point(FWidth - lineExtent.X, 0), a);
|
||||||
|
end;
|
||||||
|
FDrawer.Canvas.TextOut(p.X, p.Y, FText2[i]);
|
||||||
|
FPos += RotatePoint(Point(0, lineExtent.Y + LINE_INTERVAL), a);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure TChartTextOut.DoTextOutString;
|
||||||
|
begin
|
||||||
|
if System.Pos(LineEnding, FText1) = 0 then begin
|
||||||
|
FDrawer.Canvas.TextOut(FPos.X, FPos.Y, FText1);
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
FText2 := TStringList.Create;
|
||||||
|
try
|
||||||
|
FText2.Text := FText1;
|
||||||
|
DoTextOutList;
|
||||||
|
finally
|
||||||
|
FText2.Free;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
function TChartTextOut.Pos(AX, AY: Integer): TChartTextOut;
|
function TChartTextOut.Pos(AX, AY: Integer): TChartTextOut;
|
||||||
begin
|
begin
|
||||||
FPos := Point(AX, AY);
|
FPos := Point(AX, AY);
|
||||||
@ -388,7 +330,12 @@ end;
|
|||||||
|
|
||||||
procedure TCanvasDrawer.PrepareSimplePen(AColor: TChartColor);
|
procedure TCanvasDrawer.PrepareSimplePen(AColor: TChartColor);
|
||||||
begin
|
begin
|
||||||
TADrawUtils.PrepareSimplePen(FCanvas, AColor);
|
with FCanvas.Pen do begin
|
||||||
|
Color := AColor;
|
||||||
|
Style := psSolid;
|
||||||
|
Mode := pmCopy;
|
||||||
|
Width := 1;
|
||||||
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
procedure TCanvasDrawer.RadialPie(
|
procedure TCanvasDrawer.RadialPie(
|
||||||
@ -438,13 +385,30 @@ begin
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
function TCanvasDrawer.TextExtent(const AText: String): TPoint;
|
function TCanvasDrawer.TextExtent(const AText: String): TPoint;
|
||||||
|
var
|
||||||
|
sl: TStrings;
|
||||||
begin
|
begin
|
||||||
Result := MultiLineTextExtent(FCanvas, AText);
|
if Pos(LineEnding, AText) = 0 then
|
||||||
|
exit(FCanvas.TextExtent(AText));
|
||||||
|
sl := TStringList.Create;
|
||||||
|
try
|
||||||
|
sl.Text := AText;
|
||||||
|
Result := TextExtent(sl);
|
||||||
|
finally
|
||||||
|
sl.Free;
|
||||||
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function TCanvasDrawer.TextExtent(AText: TStrings): TPoint;
|
function TCanvasDrawer.TextExtent(AText: TStrings): TPoint;
|
||||||
|
var
|
||||||
|
i: Integer;
|
||||||
begin
|
begin
|
||||||
Result := MultiLineTextExtent(FCanvas, AText);
|
Result := Size(0, -LINE_INTERVAL);
|
||||||
|
for i := 0 to AText.Count - 1 do
|
||||||
|
with FCanvas.TextExtent(AText[i]) do begin
|
||||||
|
Result.X := Max(Result.X, cx);
|
||||||
|
Result.Y += cy + LINE_INTERVAL;
|
||||||
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function TCanvasDrawer.TextOut: TChartTextOut;
|
function TCanvasDrawer.TextOut: TChartTextOut;
|
||||||
|
@ -23,7 +23,7 @@ interface
|
|||||||
|
|
||||||
uses
|
uses
|
||||||
Classes, Graphics, SysUtils, Types,
|
Classes, Graphics, SysUtils, Types,
|
||||||
TACustomSeries, TALegend, TAChartUtils;
|
TAChartUtils, TACustomSeries, TALegend;
|
||||||
|
|
||||||
type
|
type
|
||||||
TLabelParams = record
|
TLabelParams = record
|
||||||
@ -146,6 +146,7 @@ var
|
|||||||
i: Integer;
|
i: Integer;
|
||||||
prevAngle: Double = 0;
|
prevAngle: Double = 0;
|
||||||
prevLabelPoly: TPointArray = nil;
|
prevLabelPoly: TPointArray = nil;
|
||||||
|
drawer: IChartDrawer;
|
||||||
begin
|
begin
|
||||||
if IsEmpty then exit;
|
if IsEmpty then exit;
|
||||||
|
|
||||||
@ -167,12 +168,13 @@ begin
|
|||||||
end;
|
end;
|
||||||
if not Marks.IsMarkLabelsVisible then exit;
|
if not Marks.IsMarkLabelsVisible then exit;
|
||||||
prevAngle := 0;
|
prevAngle := 0;
|
||||||
|
drawer := TCanvasDrawer.Create(ACanvas);
|
||||||
for i := 0 to Count - 1 do
|
for i := 0 to Count - 1 do
|
||||||
with FSlices[i].FLabel do begin
|
with FSlices[i].FLabel do begin
|
||||||
if FText <> '' then begin
|
if FText <> '' then begin
|
||||||
if RotateLabels then
|
if RotateLabels then
|
||||||
Marks.SetAdditionalAngle(prevAngle + FSlices[i].FAngle / 2);
|
Marks.SetAdditionalAngle(prevAngle + FSlices[i].FAngle / 2);
|
||||||
Marks.DrawLabel(ACanvas, FAttachment, FCenter, FText, prevLabelPoly);
|
Marks.DrawLabel(drawer, FAttachment, FCenter, FText, prevLabelPoly);
|
||||||
end;
|
end;
|
||||||
prevAngle += FSlices[i].FAngle;
|
prevAngle += FSlices[i].FAngle;
|
||||||
end;
|
end;
|
||||||
|
Loading…
Reference in New Issue
Block a user