LazMapViewer: Avoid crash of TMarkerClickPlugin when a click occurs between points. Support cursor change on points. Add OnCanClick event to disallow clicking on specific points.

git-svn-id: https://svn.code.sf.net/p/lazarus-ccr/svn@9686 8e941d3f-bd1b-0410-a28a-d453659cc2b4
This commit is contained in:
wp_xxyyzz 2025-03-18 11:24:45 +00:00
parent a1e91b1045
commit cc5dd7eb5e
5 changed files with 154 additions and 24 deletions

View File

@ -5,7 +5,7 @@
<PathDelim Value="\"/> <PathDelim Value="\"/>
<General> <General>
<SessionStorage Value="InProjectDir"/> <SessionStorage Value="InProjectDir"/>
<Title Value="MarkerClick_Demo"/> <Title Value="Marker Click Demo"/>
<Scaled Value="True"/> <Scaled Value="True"/>
<ResourceType Value="res"/> <ResourceType Value="res"/>
<UseXPManifest Value="True"/> <UseXPManifest Value="True"/>

View File

@ -17,6 +17,7 @@ uses
begin begin
RequireDerivedFormResource := True; RequireDerivedFormResource := True;
Application.Title:='Marker Click Demo';
Application.Scaled:=True; Application.Scaled:=True;
{$PUSH}{$WARN 5044 OFF} {$PUSH}{$WARN 5044 OFF}
Application.MainFormOnTaskbar := True; Application.MainFormOnTaskbar := True;

View File

@ -1,18 +1,18 @@
object MainForm: TMainForm object MainForm: TMainForm
Left = 576 Left = 576
Height = 426 Height = 606
Top = 248 Top = 248
Width = 653 Width = 806
Caption = 'Marker Click Demo' Caption = 'Marker Click Demo'
ClientHeight = 426 ClientHeight = 606
ClientWidth = 653 ClientWidth = 806
LCLVersion = '4.99.0.0' LCLVersion = '4.99.0.0'
OnCreate = FormCreate OnCreate = FormCreate
object MapView: TMapView object MapView: TMapView
Left = 0 Left = 0
Height = 340 Height = 497
Top = 0 Top = 0
Width = 653 Width = 806
Align = alClient Align = alClient
Cyclic = True Cyclic = True
DownloadEngine = MapView.BuiltInDLE DownloadEngine = MapView.BuiltInDLE
@ -22,11 +22,28 @@ object MainForm: TMainForm
MapProvider = 'Open Topo Map' MapProvider = 'Open Topo Map'
PluginManager = PluginManager PluginManager = PluginManager
POIImages = POI_Images POIImages = POI_Images
object Memo: TMemo
Left = 603
Height = 497
Top = 0
Width = 203
Align = alRight
ScrollBars = ssAutoBoth
TabOrder = 0
end
object Splitter1: TSplitter
Left = 598
Height = 497
Top = 0
Width = 5
Align = alRight
ResizeAnchor = akRight
end
end end
object Bevel1: TBevel object Bevel1: TBevel
AnchorSideLeft.Control = Owner AnchorSideLeft.Control = Owner
AnchorSideLeft.Side = asrCenter AnchorSideLeft.Side = asrCenter
Left = 323 Left = 400
Height = 50 Height = 50
Top = 108 Top = 108
Width = 6 Width = 6
@ -34,14 +51,14 @@ object MainForm: TMainForm
end end
object Panel1: TPanel object Panel1: TPanel
Left = 0 Left = 0
Height = 86 Height = 109
Top = 340 Top = 497
Width = 653 Width = 806
Align = alBottom Align = alBottom
AutoSize = True AutoSize = True
BevelOuter = bvNone BevelOuter = bvNone
ClientHeight = 86 ClientHeight = 109
ClientWidth = 653 ClientWidth = 806
TabOrder = 1 TabOrder = 1
object cgPointTypes: TCheckGroup object cgPointTypes: TCheckGroup
AnchorSideLeft.Control = Panel1 AnchorSideLeft.Control = Panel1
@ -81,13 +98,12 @@ object MainForm: TMainForm
} }
end end
object Label2: TLabel object Label2: TLabel
AnchorSideLeft.Control = cgPointTypes AnchorSideLeft.Control = Label1
AnchorSideLeft.Side = asrBottom AnchorSideLeft.Side = asrBottom
AnchorSideTop.Control = Label1 AnchorSideTop.Control = Label1
AnchorSideTop.Side = asrBottom Left = 104
Left = 379
Height = 15 Height = 15
Top = 23 Top = 86
Width = 91 Width = 91
BorderSpacing.Left = 16 BorderSpacing.Left = 16
Caption = 'blue: MAP points' Caption = 'blue: MAP points'
@ -98,18 +114,51 @@ object MainForm: TMainForm
end end
object Label1: TLabel object Label1: TLabel
AnchorSideLeft.Control = cgPointTypes AnchorSideLeft.Control = cgPointTypes
AnchorSideLeft.Side = asrBottom
AnchorSideTop.Control = cgPointTypes AnchorSideTop.Control = cgPointTypes
Left = 379 AnchorSideTop.Side = asrBottom
Left = 8
Height = 15 Height = 15
Top = 8 Top = 86
Width = 80 Width = 80
BorderSpacing.Left = 16 BorderSpacing.Bottom = 8
Caption = 'red: GPS points' Caption = 'red: GPS points'
Font.Color = clRed Font.Color = clRed
ParentColor = False ParentColor = False
ParentFont = False ParentFont = False
end end
object rgClickBehaviour: TRadioGroup
AnchorSideLeft.Control = cgPointTypes
AnchorSideLeft.Side = asrBottom
AnchorSideTop.Control = Panel1
AnchorSideBottom.Control = cgPointTypes
AnchorSideBottom.Side = asrBottom
Left = 379
Height = 70
Top = 8
Width = 125
Anchors = [akTop, akLeft, akBottom]
AutoFill = True
AutoSize = True
BorderSpacing.Left = 16
BorderSpacing.Top = 8
Caption = 'Click behaviour'
ChildSizing.LeftRightSpacing = 10
ChildSizing.HorizontalSpacing = 10
ChildSizing.EnlargeHorizontal = crsHomogenousChildResize
ChildSizing.EnlargeVertical = crsHomogenousChildResize
ChildSizing.ShrinkHorizontal = crsScaleChilds
ChildSizing.ShrinkVertical = crsScaleChilds
ChildSizing.Layout = cclLeftToRightThenTopToBottom
ChildSizing.ControlsPerLine = 1
ClientHeight = 50
ClientWidth = 121
ItemIndex = 0
Items.Strings = (
'Show dialog'
'Log click points'
)
TabOrder = 1
end
end end
object PluginManager: TMvPluginManager object PluginManager: TMvPluginManager
Left = 401 Left = 401

View File

@ -18,14 +18,18 @@ type
cgPointTypes: TCheckGroup; cgPointTypes: TCheckGroup;
Label1: TLabel; Label1: TLabel;
Label2: TLabel; Label2: TLabel;
Memo: TMemo;
Panel1: TPanel; Panel1: TPanel;
POI_Images: TImageList; POI_Images: TImageList;
MapView: TMapView; MapView: TMapView;
PluginManager: TMvPluginManager; PluginManager: TMvPluginManager;
rgClickBehaviour: TRadioGroup;
Splitter1: TSplitter;
procedure cgPointTypesItemClick(Sender: TObject; Index: integer); procedure cgPointTypesItemClick(Sender: TObject; Index: integer);
procedure FormCreate(Sender: TObject); procedure FormCreate(Sender: TObject);
private private
Plugin: TMarkerClickPlugin; Plugin: TMarkerClickPlugin;
procedure MarkerCanClickHandler(AMapView: TMapView; APoint: TGPSPoint; var CanClick: Boolean);
procedure MarkerClickHandler({%H-}AMapView: TMapView; APoint: TGPSPoint); procedure MarkerClickHandler({%H-}AMapView: TMapView; APoint: TGPSPoint);
public public
@ -124,6 +128,8 @@ begin
Plugin := TMarkerClickPlugin.Create(PluginManager); Plugin := TMarkerClickPlugin.Create(PluginManager);
Plugin.OnMarkerClick := @MarkerClickHandler; Plugin.OnMarkerClick := @MarkerClickHandler;
Plugin.OnCanClick := @MarkerCanClickHandler;
// Plugin.MapView := MapView; // Required if the MapView has a non-default cursor.
for i := 0 to cgPointTypes.Items.Count-1 do for i := 0 to cgPointTypes.Items.Count-1 do
cgPointTypes.Checked[i] := true; cgPointTypes.Checked[i] := true;
@ -141,6 +147,21 @@ begin
Plugin.PointTypes := pointTypes; Plugin.PointTypes := pointTypes;
end; end;
procedure TMainForm.MarkerCanClickHandler(AMapView: TMapView; APoint: TGPSPoint;
var CanClick: Boolean);
var
mapTrack: TMapTrack;
i: Integer;
begin
// Disallow clicking on the points of the (blue) map track
mapTrack := MapView.Layers[0].Tracks[0];
for i := 0 to mapTrack.Points.Count-1 do
begin
CanClick := not APoint.RealPoint.Equal(mapTrack.Points[i].RealPoint);
if not CanClick then exit;
end;
end;
procedure TMainForm.MarkerClickHandler(AMapView: TMapView; APoint: TGPSPoint); procedure TMainForm.MarkerClickHandler(AMapView: TMapView; APoint: TGPSPoint);
var var
s, sName, sLat, sLon: String; s, sName, sLat, sLon: String;
@ -148,10 +169,13 @@ begin
sName := APoint.Name; sName := APoint.Name;
sLat := ' Latitude ' + LatToStr(APoint.Lat, true); sLat := ' Latitude ' + LatToStr(APoint.Lat, true);
sLon := ' Longitude ' + LonToStr(APoint.Lon, true); sLon := ' Longitude ' + LonToStr(APoint.Lon, true);
s := sLat + Lineending + sLon; s := sLat + LineEnding + sLon;
if sName <> '' then if sName <> '' then
s := sName + LineEnding + s; s := sName + LineEnding + s;
ShowMessage(s); case rgClickBehaviour.ItemIndex of
0: ShowMessage(s);
1: Memo.Lines.Add(s + LineEnding);
end;
end; end;
end. end.

View File

@ -138,19 +138,28 @@ type
{ TMarkerClickPlugin } { TMarkerClickPlugin }
TMarkerCanClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint; var CanClick: Boolean) of object;
TMarkerClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint) of object; TMarkerClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint) of object;
TMarkerClickPlugin = class(TMvMarkerPlugin) TMarkerClickPlugin = class(TMvMarkerPlugin)
private private
FCursor: TCursor;
FSavedCursor: TCursor;
FShift: TShiftState; FShift: TShiftState;
FOnCanClick: TMarkerCanClickEvent;
FOnMarkerClick: TMarkerClickEvent; FOnMarkerClick: TMarkerClickEvent;
protected protected
procedure MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton; procedure MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton;
AShift: TShiftState; X,Y: Integer; var Handled: Boolean); override; AShift: TShiftState; X,Y: Integer; var Handled: Boolean); override;
procedure MouseMove(AMapView: TMapView; {%H-}AShift: TShiftState;
X,Y: Integer; var Handled: Boolean); override;
procedure SetMapView(AValue: TMapView); override;
public public
constructor Create(AOwner: TComponent); override; constructor Create(AOwner: TComponent); override;
published published
property Cursor: TCursor read FCursor write FCursor default crHandPoint;
property Shift: TShiftState read FShift write FShift default [ssLeft]; property Shift: TShiftState read FShift write FShift default [ssLeft];
property OnCanClick: TMarkerCanClickEvent read FOnCanClick write FOnCanClick;
property OnMarkerClick: TMarkerClickEvent read FOnMarkerClick write FOnMarkerClick; property OnMarkerClick: TMarkerClickEvent read FOnMarkerClick write FOnMarkerClick;
end; end;
@ -351,6 +360,12 @@ implementation
uses uses
Types; Types;
function IfThen(AValue: Boolean; ACursor1, ACursor2: TCursor): TCursor;
begin
if AValue then Result := ACursor1 else Result := ACursor2;
end;
{ TCenterMargerPlugin } { TCenterMargerPlugin }
constructor TCenterMarkerPlugin.Create(AOwner: TComponent); constructor TCenterMarkerPlugin.Create(AOwner: TComponent);
@ -928,6 +943,8 @@ end;
constructor TMarkerClickPlugin.Create(AOwner: TComponent); constructor TMarkerClickPlugin.Create(AOwner: TComponent);
begin begin
inherited; inherited;
FCursor := crHandPoint;
FSavedCursor := crDefault;
FShift := [ssLeft]; FShift := [ssLeft];
end; end;
@ -935,18 +952,57 @@ procedure TMarkerClickPlugin.MouseDown(AMapView: TMapView; Button: TMouseButton;
AShift: TShiftState; X, Y: Integer; var Handled: Boolean); AShift: TShiftState; X, Y: Integer; var Handled: Boolean);
var var
gpsPoint: TGPSPoint; gpsPoint: TGPSPoint;
canClick: Boolean;
begin begin
if Handled then if Handled then
exit; exit;
gpsPoint := FindNearestMarker(AMapView, X, Y); gpsPoint := FindNearestMarker(AMapView, X, Y);
if Assigned(FOnMarkerClick) and (AShift = FShift) then if Assigned(gpsPoint) and Assigned(FOnMarkerClick) and (AShift = FShift) then
begin begin
if Assigned(FOnCanClick) then
begin
canClick := true;
FOnCanClick(AMapView, gpsPoint, canClick);
if not canClick then
exit;
end;
FOnMarkerClick(AMapView, gpsPoint); FOnMarkerClick(AMapView, gpsPoint);
Handled := true; Handled := true;
end; end;
end; end;
procedure TMarkerClickPlugin.MouseMove(AMapView: TMapView;
{%H-}AShift: TShiftState; X,Y: Integer; var Handled: Boolean);
var
gpsPoint: TGPSPoint;
canClick: Boolean;
begin
if Handled then
exit;
gpsPoint := FindNearestMarker(AMapView, X, Y);
if Assigned(gpsPoint) then
begin
canClick := true;
if Assigned(FOnCanClick) then
FOnCanClick(AMapView, gpsPoint, canClick);
end else
canClick := false;
AMapView.Cursor := IfThen(canClick, FCursor, FSavedCursor);
end;
{ Store the original MapView cursor. Is used when the mouse is not over a
clickable point. If no MapView is assigned to the plugin it is assumed that
the MapView has the default cursor. }
procedure TMarkerClickPlugin.SetMapView(AValue: TMapView);
begin
inherited;
if Assigned(MapView) then
FSavedCursor := MapView.Cursor
else
FSavedCursor := crDefault;
end;
{ TMarkerHintPlugin } { TMarkerHintPlugin }