diff --git a/components/lazmapviewer/source/addons/bgra_drawingengine/mvde_bgra.pas b/components/lazmapviewer/source/addons/bgra_drawingengine/mvde_bgra.pas index 134cdf94a..911bd719f 100644 --- a/components/lazmapviewer/source/addons/bgra_drawingengine/mvde_bgra.pas +++ b/components/lazmapviewer/source/addons/bgra_drawingengine/mvde_bgra.pas @@ -13,7 +13,7 @@ unit mvDE_BGRA; {$mode objfpc}{$H+} - +{$WARN 6058 off : Call to subroutine "$1" marked as inline is not inlined} interface uses @@ -93,7 +93,7 @@ type procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override; procedure Rectangle(X1, Y1, X2, Y2: Integer); 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; function GetCacheItemClass: TPictureCacheItemClass; override; end; @@ -113,6 +113,13 @@ begin RegisterComponents(PALETTE_PAGE, [TMvBGRADrawingEngine]); 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 } function TBGRABitmapCacheItem.GetImageObject: TObject; @@ -415,12 +422,75 @@ begin FBuffer.CanvasBGRA.Font.Antialiasing := true; 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 ApplyFont; 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; +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); begin if (AText <> '') then @@ -429,7 +499,7 @@ begin FBuffer.CanvasBGRA.TextOut(X, Y, AText); end; end; - + } function TMvBGRADrawingEngine.GetCacheItemClass: TPictureCacheItemClass; begin Result := TBGRABitmapCacheItem; diff --git a/components/lazmapviewer/source/addons/rgbgraphics_drawingengine/mvde_rgbgraphics.pas b/components/lazmapviewer/source/addons/rgbgraphics_drawingengine/mvde_rgbgraphics.pas index ba82b33e4..b7081fec3 100644 --- a/components/lazmapviewer/source/addons/rgbgraphics_drawingengine/mvde_rgbgraphics.pas +++ b/components/lazmapviewer/source/addons/rgbgraphics_drawingengine/mvde_rgbgraphics.pas @@ -105,7 +105,7 @@ type procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override; procedure Rectangle(X1, Y1, X2, Y2: Integer); 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; function GetCacheItemClass: TPictureCacheItemClass; override; end; @@ -710,7 +710,8 @@ begin Opacity := 1.0; end; -function TMvRGBGraphicsDrawingEngine.TextExtent(const AText: String): TSize; +function TMvRGBGraphicsDrawingEngine.TextExtent(const AText: String; + ARotated: Boolean = false): TSize; var bmp: TBitmap; pts: TPointArray; @@ -721,7 +722,10 @@ begin bmp.Canvas.Font.Size := FFontSize; bmp.Canvas.Font.Style := FFontStyle; bmp.Canvas.Font.Orientation := FFontOrientation; - Result := MeasureTextSize(bmp.Canvas, AText, pts); + if (FFontOrientation = 0) or not ARotated then + Result := bmp.Canvas.TextExtent(AText) + else + Result := MeasureTextSize(bmp.Canvas, AText, pts); finally bmp.Free; end; diff --git a/components/lazmapviewer/source/mvde_intfgraphics.pas b/components/lazmapviewer/source/mvde_intfgraphics.pas index 5f0a4de5a..ac28a5545 100644 --- a/components/lazmapviewer/source/mvde_intfgraphics.pas +++ b/components/lazmapviewer/source/mvde_intfgraphics.pas @@ -96,7 +96,7 @@ type procedure PaintToCanvas(ACanvas: TCanvas; Origin: TPoint); override; procedure Rectangle(X1, Y1, X2, Y2: Integer); 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; function GetCacheItemClass: TPictureCacheItemClass; override; end; @@ -746,11 +746,13 @@ begin end; { Returns the size of the given text. - NOTE: Text rotation is taken into account. } -function TMvIntfGraphicsDrawingEngine.TextExtent(const AText: String): TSize; + NOTE: Text rotation is taken into account if the Rotated argument is true. } +function TMvIntfGraphicsDrawingEngine.TextExtent(const AText: String; + ARotated: Boolean = false): TSize; var bmp: TBitmap; pts: TPointArray; + R: TRect; begin bmp := TBitmap.Create; try @@ -758,7 +760,13 @@ begin bmp.Canvas.Font.Size := FFontSize; bmp.Canvas.Font.Style := FFontStyle; bmp.Canvas.Font.Orientation := FFontOrientation; - Result := MeasureTextSize(bmp.Canvas, AText, pts); + 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); finally bmp.Free; end; diff --git a/components/lazmapviewer/source/mvdrawingengine.pas b/components/lazmapviewer/source/mvdrawingengine.pas index 635fecd5b..98802ce77 100644 --- a/components/lazmapviewer/source/mvdrawingengine.pas +++ b/components/lazmapviewer/source/mvdrawingengine.pas @@ -102,7 +102,7 @@ type AFontStyle: TFontStyles; AFontColor: TColor; AFontOrientation: Single = 0.0); procedure SetPen(APen: TMvPen); 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; procedure TextOut(X, Y: Integer; const AText: String); virtual; abstract; function TextWidth(const AText: String): Integer; @@ -147,8 +147,8 @@ function ClipPolyToRect(constref ARect: TRect; var APoly: TPointArray; procedure PolyBounds(APoly: array of TPoint; out ABounds: TRect); // Measures the size of the given text and returns the vertices of the corners -// of the rectangle enclosing the text. Text rotation is supported. -// BUT: in case of rotated text, word-wrapping is not allowed. +// of the rectangle enclosing the text. Text rotation is supported. +// BUT: in case of rotated text, word-wrapping is not allowed. function MeasureTextSize(ACanvas: TCanvas; AText: String; out ACorners: TPointArray): TSize; diff --git a/components/lazmapviewer/source/mvmapviewer.pas b/components/lazmapviewer/source/mvmapviewer.pas index f60eb4ee8..4f0ba688c 100644 --- a/components/lazmapviewer/source/mvmapviewer.pas +++ b/components/lazmapviewer/source/mvmapviewer.pas @@ -3508,14 +3508,12 @@ var end; // Draw the point text + DrawingEngine.Opacity := FPOIOpacity; + DrawingEngine.BrushColor := FPOITextBgColor; if FPOITextBgColor = clNone then DrawingEngine.BrushStyle := bsClear else - begin DrawingEngine.BrushStyle := bsSolid; - DrawingEngine.BrushColor := FPOITextBgColor; - end; - DrawingEngine.Opacity := FPOIOpacity; // Text is at the left/centered/right of the GPS point... case txtPosHor of @@ -3588,7 +3586,7 @@ begin txt := APt.Name; if FPOITextBgColor <> clNone then 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. pt := Engine.LatLonToScreen(APt.RealPoint);