LazMapViewer: Fix drawing of missing tiles. Issue #39111.

git-svn-id: https://svn.code.sf.net/p/lazarus-ccr/svn@9682 8e941d3f-bd1b-0410-a28a-d453659cc2b4
This commit is contained in:
wp_xxyyzz 2025-03-14 19:20:10 +00:00
parent a49a182aa2
commit 2eb97a3732
4 changed files with 94 additions and 32 deletions

View File

@ -234,7 +234,8 @@ end;
procedure TMvBGRADrawingEngine.FillRect(X1, Y1, X2, Y2: Integer); procedure TMvBGRADrawingEngine.FillRect(X1, Y1, X2, Y2: Integer);
begin begin
FBuffer.CanvasBGRA.FillRect(X1, Y1, X2, Y2); // FBuffer.CanvasBGRA.FillRect(X1, Y1, X2, Y2);
FBuffer.FillRect(X1, Y1, X2, Y2, BrushColor);
end; end;
function TMvBGRADrawingEngine.GetPenStyle: TPenStyle; function TMvBGRADrawingEngine.GetPenStyle: TPenStyle;

View File

@ -51,7 +51,7 @@ type
procedure SetPosition(AValue: TTileInfoPosition); procedure SetPosition(AValue: TTileInfoPosition);
protected protected
procedure AfterDrawTile(AMapView: TMapView; ADrawingEngine: TMvCustomDrawingEngine; procedure AfterDrawTile(AMapView: TMapView; ADrawingEngine: TMvCustomDrawingEngine;
ATileID: TTileID; ARect: TRect; var Handled: Boolean); override; ATileID: TTileID; ARect: TRect; var {%H-}Handled: Boolean); override;
public public
constructor Create(AOwner: TComponent); override; constructor Create(AOwner: TComponent); override;
published published

View File

@ -123,7 +123,7 @@ type
function DegreesToPixelsEPSG3395(const AWin: TMapWindow; APt: TRealPoint): TPoint; function DegreesToPixelsEPSG3395(const AWin: TMapWindow; APt: TRealPoint): TPoint;
function DegreesToPixelsEPSG3857(const AWin: TMapWindow; APt: TRealPoint): TPoint; function DegreesToPixelsEPSG3857(const AWin: TMapWindow; APt: TRealPoint): TPoint;
procedure Redraw(const aWin: TMapWindow; const PaintOnly: Boolean = False); procedure Redraw(const aWin: TMapWindow; const PaintOnly: Boolean = False);
function CalculateVisibleTiles(const aWin: TMapWindow; out Area: TArea): Boolean; function CalculateVisibleTiles(const aWin: TMapWindow; out AFullyCovered: Boolean): TArea;
function IsCurrentWin(const aWin: TMapWindow) : boolean; function IsCurrentWin(const aWin: TMapWindow) : boolean;
protected protected
procedure AdjustZoomCenter(var AWin: TMapWindow); procedure AdjustZoomCenter(var AWin: TMapWindow);
@ -355,36 +355,58 @@ begin
CalculateWin(AWin); CalculateWin(AWin);
end; end;
// Returns whether the view area is not covered fully with a tiles { Calculates the range of tile ids needed to cover the window with a map.
function TMapViewerEngine.CalculateVisibleTiles(const aWin: TMapWindow; out Returns AFullyCovered=false when not the entire windows is covered by the
Area: TArea): Boolean; map tiles. }
function TMapViewerEngine.CalculateVisibleTiles(const AWin: TMapWindow;
out AFullyCovered: Boolean): TArea;
var var
MaxX, MaxY, startX, startY: int64; maxX, maxY, startX, startY: Int64;
WorldMax: Int64; worldMax: Int64;
begin begin
Area := Default(TArea); Result := Default(TArea);
Result := (aWin.X <= 0) and (aWin.Y <= 0);
WorldMax := Int64(1) shl AWin.Zoom - 1; AFullyCovered := true;
MaxX := Int64(aWin.Width) div TileSize.CX + 1; if (AWin.X > 0) or (AWin.Y > 0) then
MaxY := Int64(aWin.Height) div TileSize.CY + 1; AFullyCovered := false;
if (MaxX > WorldMax) or (MaxY > WorldMax) then
worldMax := Int64(1) shl AWin.Zoom - 1;
maxX := Int64(AWin.Width) div TileSize.CX + 1;
maxY := Int64(AWin.Height) div TileSize.CY + 1;
if maxY > worldMax then
begin begin
Result := False; maxY := worldMax;
MaxX := Min(WorldMax, MaxX); AFullyCovered := false;
MaxY := Min(WorldMax, MaxY);
end; end;
startX := -aWin.X div TileSize.CX; if (not Cyclic) and (maxX > worldMax) then
startY := -aWin.Y div TileSize.CY;
if (startX < 0) or (startY < 0) then
begin begin
startX := Max(0, startX); maxX := worldMax;
startY := Max(0, startY); AFullyCovered := false;
Result := False; end;
startX := Max(0, -AWin.X div TileSize.CX);
startY := Max(0, -AWin.Y div TileSize.CY);
Result.Left := startX;
if Cyclic then
Result.Right := Result.Left + maxX
else
if startX + maxX < worldMax then
Result.Right := startX + maxX
else
begin
Result.Right := worldMax;
AFullyCovered := false;
end;
Result.Top := startY;
if startY + maxY < worldMax then
Result.Bottom := startY + maxY
else
begin
Result.Bottom := worldMax;
AFullyCovered := false;
end; end;
Area.Left := startX;
Area.Right := startX + MaxX;
Area.Top := startY;
Area.Bottom := startY + MaxY;
end; end;
procedure TMapViewerEngine.CalculateWin(var AWin: TMapWindow); procedure TMapViewerEngine.CalculateWin(var AWin: TMapWindow);
@ -1096,6 +1118,7 @@ var
iTile, numTiles, XShift: Integer; iTile, numTiles, XShift: Integer;
Tiles: TTileIdArray = nil; Tiles: TTileIdArray = nil;
foundInCache: Boolean; foundInCache: Boolean;
fullyCovered: Boolean;
procedure AddJob; procedure AddJob;
var var
@ -1112,6 +1135,34 @@ var
end; end;
end; end;
procedure EraseAround;
var
x, y: Integer;
begin
// Erase part above the map
y := AWin.Y;
if y > 0 then
EraseBackground(Rect(0, 0, AWin.Width, y));
// Erase part below the map
y := AWin.Y + (numTiles - 1 - TilesVis.Top) * TileSize.CX;
if y < AWin.Height then
EraseBackground(Rect(0, y, AWin.Width, AWin.Height));
if not Cyclic then
begin
// Erase part at the left of the map
x := AWin.X;
if x > 0 then
EraseBackground(Rect(0, 0, x, AWin.Height));
// Erase part at the right of the map
x := AWin.X + (numTiles - 1 - TilesVis.Left) * TileSize.CX;
if x < AWin.Width then
EraseBackground(Rect(x, 0, AWin.Width, AWin.Height));
end;
end;
(*
procedure EraseAround; procedure EraseAround;
var var
T, L, B, R: Integer; T, L, B, R: Integer;
@ -1144,6 +1195,7 @@ var
AWin.Height) AWin.Height)
); );
end; end;
*)
begin begin
if not(Active) then if not(Active) then
@ -1154,12 +1206,15 @@ begin
Exit; Exit;
end; end;
if not CalculateVisibleTiles(AWin, TilesVis) then numTiles := 1 shl AWin.Zoom;
TilesVis := CalculateVisibleTiles(AWin, fullyCovered);
if not fullyCovered then
EraseAround; EraseAround;
// if not CalculateVisibleTiles(AWin, TilesVis) then
// EraseAround;
SetLength(Tiles, (TilesVis.Bottom - TilesVis.Top + 1) * (TilesVis.Right - TilesVis.Left + 1)); SetLength(Tiles, (TilesVis.Bottom - TilesVis.Top + 1) * (TilesVis.Right - TilesVis.Left + 1));
iTile := Low(Tiles); iTile := Low(Tiles);
numTiles := 1 shl AWin.Zoom;
XShift := IfThen(aWin.X > 0, numTiles - aWin.X div TileSize.CX - 1, 0); XShift := IfThen(aWin.X > 0, numTiles - aWin.X div TileSize.CX - 1, 0);
for Y := TilesVis.Top to TilesVis.Bottom do for Y := TilesVis.Top to TilesVis.Bottom do
for X := TilesVis.Left to TilesVis.Right do for X := TilesVis.Left to TilesVis.Right do

View File

@ -3116,11 +3116,10 @@ const
Canvas.FillRect(0, 0, ClientWidth, ClientHeight); Canvas.FillRect(0, 0, ClientWidth, ClientHeight);
end; end;
procedure FullRedraw; procedure ObjectsDraw;
var var
W: Integer; W: Integer;
begin begin
Engine.Redraw;
W := ClientWidth; W := ClientWidth;
if Cyclic then if Cyclic then
W := Min(1 shl Zoom * TileSize.CX, W); W := Min(1 shl Zoom * TileSize.CX, W);
@ -3129,13 +3128,20 @@ const
if Assigned(FBeforeDrawObjectsEvent) then if Assigned(FBeforeDrawObjectsEvent) then
FBeforeDrawObjectsEvent(Self); FBeforeDrawObjectsEvent(Self);
DrawObjects(Default(TTileId), 0, 0, W - 1, ClientHeight); DrawObjects(Default(TTileID), 0, 0, W - 1, ClientHeight);
GetPluginManager.AfterDrawObjects(Self); GetPluginManager.AfterDrawObjects(Self);
if Assigned(FAfterDrawObjectsEvent) then if Assigned(FAfterDrawObjectsEvent) then
FAfterDrawObjectsEvent(Self); FAfterDrawObjectsEvent(Self);
end;
procedure FullRedraw;
begin
Engine.Redraw;
ObjectsDraw;
DrawingEngine.PaintToCanvas(Canvas); DrawingEngine.PaintToCanvas(Canvas);
if DebugTiles then // DebugTiles is deprecated if DebugTiles then // DebugTiles is deprecated
DrawCenter; DrawCenter;