LazMapViewer: Improved output of rotated text. DrawingEngine.TextExtent now can return both unrotated and rotated text extents.
git-svn-id: https://svn.code.sf.net/p/lazarus-ccr/svn@9709 8e941d3f-bd1b-0410-a28a-d453659cc2b4
This commit is contained in:
parent
c5dd3f0862
commit
3a4e228b66
@ -13,7 +13,7 @@
|
|||||||
unit mvDE_BGRA;
|
unit mvDE_BGRA;
|
||||||
|
|
||||||
{$mode objfpc}{$H+}
|
{$mode objfpc}{$H+}
|
||||||
|
{$WARN 6058 off : Call to subroutine "$1" marked as inline is not inlined}
|
||||||
interface
|
interface
|
||||||
|
|
||||||
uses
|
uses
|
||||||
@ -93,7 +93,7 @@ type
|
|||||||
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
||||||
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
||||||
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
||||||
function TextExtent(const AText: String): TSize; override;
|
function TextExtent(const AText: String; ARotated: Boolean = false): TSize; override;
|
||||||
procedure TextOut(X, Y: Integer; const AText: String); override;
|
procedure TextOut(X, Y: Integer; const AText: String); override;
|
||||||
function GetCacheItemClass: TPictureCacheItemClass; override;
|
function GetCacheItemClass: TPictureCacheItemClass; override;
|
||||||
end;
|
end;
|
||||||
@ -113,6 +113,13 @@ begin
|
|||||||
RegisterComponents(PALETTE_PAGE, [TMvBGRADrawingEngine]);
|
RegisterComponents(PALETTE_PAGE, [TMvBGRADrawingEngine]);
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
function RotatePointF(P: TPointF; sinPhi, cosPhi: Double): TPointF;
|
||||||
|
begin
|
||||||
|
Result.X := cosPhi * P.X + sinPhi * P.Y;
|
||||||
|
Result.Y := -sinPhi * P.X + cosPhi * P.Y;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
{ TBGRABitmapCacheItem }
|
{ TBGRABitmapCacheItem }
|
||||||
|
|
||||||
function TBGRABitmapCacheItem.GetImageObject: TObject;
|
function TBGRABitmapCacheItem.GetImageObject: TObject;
|
||||||
@ -415,12 +422,75 @@ begin
|
|||||||
FBuffer.CanvasBGRA.Font.Antialiasing := true;
|
FBuffer.CanvasBGRA.Font.Antialiasing := true;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function TMvBGRADrawingEngine.TextExtent(const AText: String): TSize;
|
function TMvBGRADrawingEngine.TextExtent(const AText: String;
|
||||||
|
ARotated: Boolean = false): TSize;
|
||||||
|
var
|
||||||
|
s, c: Double;
|
||||||
|
pts: Array[0..3] of TPointF;
|
||||||
begin
|
begin
|
||||||
ApplyFont;
|
ApplyFont;
|
||||||
Result := FBuffer.CanvasBGRA.TextExtent(AText);
|
Result := FBuffer.CanvasBGRA.TextExtent(AText);
|
||||||
|
if (FFontOrientation <> 0) and ARotated then
|
||||||
|
begin
|
||||||
|
SinCos(FFontOrientation * pi / 1800, s, c);
|
||||||
|
pts[0] := PointF(0, 0);
|
||||||
|
pts[1] := RotatePointF(PointF(Result.CX, 0), s, c);
|
||||||
|
pts[2] := RotatePointF(PointF(Result.CX, Result.CY), s, c);
|
||||||
|
pts[3] := RotatePointF(PointF(0, Result.CY), s, c);
|
||||||
|
Result.CX := round(
|
||||||
|
MaxValue([pts[0].X, pts[1].X, pts[2].X, pts[3].X]) -
|
||||||
|
MinValue([pts[0].X, pts[1].X, pts[2].X, pts[3].X])
|
||||||
|
);
|
||||||
|
Result.CY := round(
|
||||||
|
MaxValue([pts[0].Y, pts[1].Y, pts[2].Y, pts[3].Y]) -
|
||||||
|
MinValue([pts[0].Y, pts[1].Y, pts[2].Y, pts[3].Y])
|
||||||
|
);
|
||||||
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
procedure TMvBGRADrawingEngine.TextOut(X, Y: Integer; const AText: String);
|
||||||
|
var
|
||||||
|
ext: TSize;
|
||||||
|
dx, dy: Single;
|
||||||
|
ctr: TPointF;
|
||||||
|
R: TRectF;
|
||||||
|
Pts: Array[0..3] of TPointF;
|
||||||
|
s, c: Double;
|
||||||
|
begin
|
||||||
|
if (AText <> '') then
|
||||||
|
begin
|
||||||
|
ApplyFont;
|
||||||
|
if FFontOrientation = 0 then
|
||||||
|
begin
|
||||||
|
FBuffer.FontVerticalAnchor := fvaTop;
|
||||||
|
FBuffer.CanvasBGRA.TextOut(X, Y, AText);
|
||||||
|
end else
|
||||||
|
begin
|
||||||
|
ext := FBuffer.CanvasBGRA.TextExtent(AText);
|
||||||
|
dx := ext.CX/2;
|
||||||
|
dy := ext.CY/2;
|
||||||
|
SinCos(FFontOrientation * pi / 1800, s, c);
|
||||||
|
Pts[0] := RotatePointF(PointF(-dx, -dy), s, c);
|
||||||
|
Pts[1] := RotatePointF(PointF(+dx, -dy), s, c);
|
||||||
|
Pts[2] := RotatePointF(PointF(+dx, +dy), s, c);
|
||||||
|
Pts[3] := RotatePointF(PointF(-dx, +dy), s, c);
|
||||||
|
R := RectF(
|
||||||
|
MinValue([Pts[0].X, Pts[1].X, Pts[2].X, Pts[3].X]),
|
||||||
|
MinValue([Pts[0].Y, Pts[1].Y, Pts[2].Y, Pts[3].Y]),
|
||||||
|
MaxValue([Pts[0].X, Pts[1].X, Pts[2].X, Pts[3].X]),
|
||||||
|
MaxValue([Pts[0].Y, Pts[1].Y, Pts[2].Y, Pts[3].Y])
|
||||||
|
);
|
||||||
|
|
||||||
|
dx := R.Width/2;
|
||||||
|
dy := R.Height/2;
|
||||||
|
ctr := PointF(X + dx, Y + dy);
|
||||||
|
FBuffer.CanvasBGRA.PolygonF([Pts[0] + ctr, Pts[1] + ctr, Pts[2] + ctr, Pts[3] + ctr], False, True);
|
||||||
|
FBuffer.FontVerticalAnchor := fvaXCenter;
|
||||||
|
FBuffer.TextOut(ctr.X, ctr.Y, AText, FontColor, taCenter);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
{
|
||||||
procedure TMvBGRADrawingEngine.TextOut(X, Y: Integer; const AText: String);
|
procedure TMvBGRADrawingEngine.TextOut(X, Y: Integer; const AText: String);
|
||||||
begin
|
begin
|
||||||
if (AText <> '') then
|
if (AText <> '') then
|
||||||
@ -429,7 +499,7 @@ begin
|
|||||||
FBuffer.CanvasBGRA.TextOut(X, Y, AText);
|
FBuffer.CanvasBGRA.TextOut(X, Y, AText);
|
||||||
end;
|
end;
|
||||||
end;
|
end;
|
||||||
|
}
|
||||||
function TMvBGRADrawingEngine.GetCacheItemClass: TPictureCacheItemClass;
|
function TMvBGRADrawingEngine.GetCacheItemClass: TPictureCacheItemClass;
|
||||||
begin
|
begin
|
||||||
Result := TBGRABitmapCacheItem;
|
Result := TBGRABitmapCacheItem;
|
||||||
|
@ -105,7 +105,7 @@ type
|
|||||||
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
||||||
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
||||||
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
||||||
function TextExtent(const AText: String): TSize; override;
|
function TextExtent(const AText: String; ARotated: Boolean = false): TSize; override;
|
||||||
procedure TextOut(X, Y: Integer; const AText: String); override;
|
procedure TextOut(X, Y: Integer; const AText: String); override;
|
||||||
function GetCacheItemClass: TPictureCacheItemClass; override;
|
function GetCacheItemClass: TPictureCacheItemClass; override;
|
||||||
end;
|
end;
|
||||||
@ -710,7 +710,8 @@ begin
|
|||||||
Opacity := 1.0;
|
Opacity := 1.0;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
function TMvRGBGraphicsDrawingEngine.TextExtent(const AText: String): TSize;
|
function TMvRGBGraphicsDrawingEngine.TextExtent(const AText: String;
|
||||||
|
ARotated: Boolean = false): TSize;
|
||||||
var
|
var
|
||||||
bmp: TBitmap;
|
bmp: TBitmap;
|
||||||
pts: TPointArray;
|
pts: TPointArray;
|
||||||
@ -721,6 +722,9 @@ begin
|
|||||||
bmp.Canvas.Font.Size := FFontSize;
|
bmp.Canvas.Font.Size := FFontSize;
|
||||||
bmp.Canvas.Font.Style := FFontStyle;
|
bmp.Canvas.Font.Style := FFontStyle;
|
||||||
bmp.Canvas.Font.Orientation := FFontOrientation;
|
bmp.Canvas.Font.Orientation := FFontOrientation;
|
||||||
|
if (FFontOrientation = 0) or not ARotated then
|
||||||
|
Result := bmp.Canvas.TextExtent(AText)
|
||||||
|
else
|
||||||
Result := MeasureTextSize(bmp.Canvas, AText, pts);
|
Result := MeasureTextSize(bmp.Canvas, AText, pts);
|
||||||
finally
|
finally
|
||||||
bmp.Free;
|
bmp.Free;
|
||||||
|
@ -96,7 +96,7 @@ type
|
|||||||
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override;
|
||||||
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
procedure Rectangle(X1, Y1, X2, Y2: Integer); override;
|
||||||
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
function SaveToImage(AClass: TRasterImageClass): TRasterImage; override;
|
||||||
function TextExtent(const AText: String): TSize; override;
|
function TextExtent(const AText: String; ARotated: Boolean = false): TSize; override;
|
||||||
procedure TextOut(X, Y: Integer; const AText: String); override;
|
procedure TextOut(X, Y: Integer; const AText: String); override;
|
||||||
function GetCacheItemClass: TPictureCacheItemClass; override;
|
function GetCacheItemClass: TPictureCacheItemClass; override;
|
||||||
end;
|
end;
|
||||||
@ -746,11 +746,13 @@ begin
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
{ Returns the size of the given text.
|
{ Returns the size of the given text.
|
||||||
NOTE: Text rotation is taken into account. }
|
NOTE: Text rotation is taken into account if the Rotated argument is true. }
|
||||||
function TMvIntfGraphicsDrawingEngine.TextExtent(const AText: String): TSize;
|
function TMvIntfGraphicsDrawingEngine.TextExtent(const AText: String;
|
||||||
|
ARotated: Boolean = false): TSize;
|
||||||
var
|
var
|
||||||
bmp: TBitmap;
|
bmp: TBitmap;
|
||||||
pts: TPointArray;
|
pts: TPointArray;
|
||||||
|
R: TRect;
|
||||||
begin
|
begin
|
||||||
bmp := TBitmap.Create;
|
bmp := TBitmap.Create;
|
||||||
try
|
try
|
||||||
@ -758,6 +760,12 @@ begin
|
|||||||
bmp.Canvas.Font.Size := FFontSize;
|
bmp.Canvas.Font.Size := FFontSize;
|
||||||
bmp.Canvas.Font.Style := FFontStyle;
|
bmp.Canvas.Font.Style := FFontStyle;
|
||||||
bmp.Canvas.Font.Orientation := FFontOrientation;
|
bmp.Canvas.Font.Orientation := FFontOrientation;
|
||||||
|
if (FFontOrientation = 0) or (not ARotated) then
|
||||||
|
begin
|
||||||
|
R := Rect(0, 0, 10000, 10000);
|
||||||
|
DrawText(bmp.Canvas.Handle, PChar(AText), Length(AText), R, DT_CALCRECT or DT_WORDBREAK);
|
||||||
|
Result := TSize(R.BottomRight);
|
||||||
|
end else
|
||||||
Result := MeasureTextSize(bmp.Canvas, AText, pts);
|
Result := MeasureTextSize(bmp.Canvas, AText, pts);
|
||||||
finally
|
finally
|
||||||
bmp.Free;
|
bmp.Free;
|
||||||
|
@ -102,7 +102,7 @@ type
|
|||||||
AFontStyle: TFontStyles; AFontColor: TColor; AFontOrientation: Single = 0.0);
|
AFontStyle: TFontStyles; AFontColor: TColor; AFontOrientation: Single = 0.0);
|
||||||
procedure SetPen(APen: TMvPen);
|
procedure SetPen(APen: TMvPen);
|
||||||
procedure SetPen(APenStyle: TPenStyle; APenWidth: Integer; APenColor: TColor);
|
procedure SetPen(APenStyle: TPenStyle; APenWidth: Integer; APenColor: TColor);
|
||||||
function TextExtent(const AText: String): TSize; virtual; abstract;
|
function TextExtent(const AText: String; ARotated: Boolean = false): TSize; virtual; abstract;
|
||||||
function TextHeight(const AText: String): Integer;
|
function TextHeight(const AText: String): Integer;
|
||||||
procedure TextOut(X, Y: Integer; const AText: String); virtual; abstract;
|
procedure TextOut(X, Y: Integer; const AText: String); virtual; abstract;
|
||||||
function TextWidth(const AText: String): Integer;
|
function TextWidth(const AText: String): Integer;
|
||||||
|
@ -3508,14 +3508,12 @@ var
|
|||||||
end;
|
end;
|
||||||
|
|
||||||
// Draw the point text
|
// Draw the point text
|
||||||
|
DrawingEngine.Opacity := FPOIOpacity;
|
||||||
|
DrawingEngine.BrushColor := FPOITextBgColor;
|
||||||
if FPOITextBgColor = clNone then
|
if FPOITextBgColor = clNone then
|
||||||
DrawingEngine.BrushStyle := bsClear
|
DrawingEngine.BrushStyle := bsClear
|
||||||
else
|
else
|
||||||
begin
|
|
||||||
DrawingEngine.BrushStyle := bsSolid;
|
DrawingEngine.BrushStyle := bsSolid;
|
||||||
DrawingEngine.BrushColor := FPOITextBgColor;
|
|
||||||
end;
|
|
||||||
DrawingEngine.Opacity := FPOIOpacity;
|
|
||||||
|
|
||||||
// Text is at the left/centered/right of the GPS point...
|
// Text is at the left/centered/right of the GPS point...
|
||||||
case txtPosHor of
|
case txtPosHor of
|
||||||
@ -3588,7 +3586,7 @@ begin
|
|||||||
txt := APt.Name;
|
txt := APt.Name;
|
||||||
if FPOITextBgColor <> clNone then
|
if FPOITextBgColor <> clNone then
|
||||||
txt := ' ' + txt + ' '; // add some margin
|
txt := ' ' + txt + ' '; // add some margin
|
||||||
txtExtent := DrawingEngine.TextExtent(txt);
|
txtExtent := DrawingEngine.TextExtent(txt, DrawingEngine.FontOrientation <> 0);
|
||||||
|
|
||||||
// Draw point, in case of cyclic points multiple times.
|
// Draw point, in case of cyclic points multiple times.
|
||||||
pt := Engine.LatLonToScreen(APt.RealPoint);
|
pt := Engine.LatLonToScreen(APt.RealPoint);
|
||||||
|
Loading…
Reference in New Issue
Block a user