lazarus-ccr/components/lazmapviewer/source/addons/plugins/markers/mvmarkerplugins.pas

1364 lines
41 KiB
ObjectPascal

{-------------------------------------------------------------------------------
mvMarkerPlugins.pas
License: modified LGPL with linking exception (like RTL, FCL and LCL)
See the file COPYING.modifiedLGPL.txt, included in the Lazarus distribution,
for details about the license.
See also: https://wiki.lazarus.freepascal.org/FPC_modified_LGPL
--------------------------------------------------------------------------------
This unit collects some typical marker-related plugins:
- TMarkerHintPlugin
displays a hint when the mouse hovers over a GPSPoint or a MapPoint.
Handle the OnHint event to define the exact string to be displayed in the
popup hint window.
- TMarkerClickPlugin
Fires the OnMarkerClick event when the user clicks on a GPSPoint or a
MapPoint. Which mouse key and modification shift key (CTRL, SHIFT, ALT)
triggers the event, is defined by the Shift property.
- TMarkerEditorPlugin
Allows to add new GPSPoints or MapPoints, to select them by clicking and
to drag them to another location. There are also methods to convert a group
of selected markers to a GPSTrack/MapTrack or GPSArea/MapArea, or to
delete the points in this selection.
- TDraggableMarkerPlugin
Allows to drag the marker hit by pressing the mouse button to a new location.
Unlike the TMarkerEditorPlugin data are handled such that multiple maps
can be processed by the same plugin.
-------------------------------------------------------------------------------}
unit mvMarkerPlugins;
{$mode objfpc}{$H+}
interface
uses
Classes, SysUtils,
Graphics, Controls, Forms, LCLIntf,
mvMapViewer, mvDrawingEngine, mvPluginCommon, mvGPSObj, mvGeoMath, mvTypes;
type
{ TMarkerHintPlugin }
{ Event allowing to create a different hint window class for custom drawing
of the hint. }
TMarkerCreateHintWindowEvent = procedure(AMapView: TMapView;
out AHintWindow: THintWindow) of object;
{ Event to define the hint text for the marker at the given point.
Return an empty string when no hint should be displayed. }
TMarkerHintEvent = procedure (AMapView: TMapView; APoint: TGPSPoint;
var AHint: String) of object;
TMarkerHintPlugin = class(TMvMarkerPlugin)
private
const
DEFAULT_HINT_OFFSET_X = 0;
DEFAULT_HINT_OFFSET_Y = 15;
DEFAULT_HIDE_INTERVAL = 1000;
private
FAutoHideHint: Boolean;
FHideInterval: Integer;
FHintOffsetX: Integer;
FHintOffsetY: Integer;
FHintWindow: THintWindow;
FShowHint: Boolean;
FOnCreateHintWindow: TMarkerCreateHintWindowEvent;
FOnHint: TMarkerHintEvent;
protected
function CreateHintWindow(AMapView: TMapView): THintWindow; virtual;
procedure DisplayHint(AMapView: TMapView; APoint: TGPSPoint; X, Y: Integer); virtual;
procedure HideHint; virtual;
protected
procedure MouseMove(AMapView: TMapView; {%H-}AShift: TShiftState; X,Y: Integer;
var Handled: Boolean); override;
public
constructor Create(AOwner: TComponent); override;
published
property AutoHideHint: Boolean read FAutoHideHint write FAutoHideHint default false;
property HideInterval: Integer read FHideInterval write FHideInterval default 0;
property HintOffsetX: Integer read FHintOffsetX write FHintOffsetX default DEFAULT_HINT_OFFSET_X;
property HintOffsetY: Integer read FHintOffsetY write FHintOffsetY default DEFAULT_HINT_OFFSET_Y;
property ShowHint: Boolean read FShowHint write FShowHint default true;
property OnCreateHintWindow: TMarkerCreateHintWindowEvent read FOnCreateHintWindow write FOnCreateHintWindow;
property OnHint: TMarkerHintEvent read FOnHint write FOnHint;
end;
{ TCustomMarkerClickPlugin }
TMarkerCanClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint; var CanClick: Boolean) of object;
TMarkerClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint) of object;
TCustomMarkerClickPlugin = class(TMvMarkerPlugin)
private
FCursor: TCursor;
FOnCanClick: TMarkerCanClickEvent;
FOnMarkerClick: TMarkerClickEvent;
FShift: TShiftState;
protected
FMouseDownOnMarker: Boolean;
FMousePoint: TPoint;
FOrigGpsPoint: TGPSPoint;
FSavedCursor: TCursor;
procedure MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton;
AShift: TShiftState; X,Y: Integer; var Handled: Boolean); override;
procedure MouseMove(AMapView: TMapView; {%H-}AShift: TShiftState;
X,Y: Integer; var Handled: Boolean); override;
procedure MouseUp({%H-}AMapView: TMapView; {%H-}Button: TMouseButton;
{%H-}AShift: TShiftState; {%H-}X,{%H-}Y: Integer; var {%H-}Handled: Boolean); override;
procedure SetMapView(AValue: TMapView); override;
property Shift: TShiftState read FShift write FShift default [ssLeft];
public
constructor Create(AOwner: TComponent); override;
published
property Cursor: TCursor read FCursor write FCursor default crHandPoint;
property OnCanClick: TMarkerCanClickEvent read FOnCanClick write FOnCanClick;
property OnMarkerClick: TMarkerClickEvent read FOnMarkerClick write FOnMarkerClick;
end;
{ TMarkerClickPlugin }
TMarkerClickPlugin = class(TCustomMarkerClickPlugin)
published
property Shift;
end;
{ TMarkerEditorPlugin }
TMarkerDrawPointEvent = procedure (AMapView: TMapView;
ADrawingEngine: TMvCustomDrawingEngine; AGPSPoint: TGPSPoint;
AScreenPoint: TPoint; AMarkerSize: Integer) of object;
TMarkerNewPointEvent = procedure (AMapView: TMapView; APoint: TGPSPoint) of object;
TMarkerStartDragEvent = procedure (AMapView: TMapView; var CanDrag: Boolean) of object;
TMarkerClickMode = (cmNewPoint, cmSelectPoint, cmSelectShape,
cmTogglePoint, cmRubberband);
TMarkerNewPointType = (nptGPSPoint, nptMapPoint);
TMarkerEditorPlugin = class(TMarkerClickPlugin)
private
const
DEFAULT_CLICKMODE = cmSelectPoint;
DEFAULT_RUBBERBAND_BORDERCOLOR = clGray;
DEFAULT_RUBBERBAND_FILLCOLOR = clWhite;
DEFAULT_RUBBERBAND_OPACITY = 0.55;
private
FClickMode: TMarkerClickMode;
FDragCursor: TCursor;
FDragging: Boolean;
FExtendSelection: Boolean;
FNewPointType: TMarkerNewPointType;
FRubberbandBorderColor: TColor;
FRubberbandFillColor: TColor;
FRubberbandOpacity: Single;
FRubberbandMode: Boolean;
FRubberbandStartPt: TPoint;
FRubberbandEndPt: TPoint;
FSelection: TGPSPointList;
FOrigSelection: array of TRealPoint; // Selection before dragging starts
FOnDrawPoint: TMarkerDrawPointEvent;
FOnNewPoint: TMarkerNewPointEvent;
FOnSelectionChange: TNotifyEvent;
FOnStartDrag: TMarkerStartDragEvent;
FOnEndDrag: TNotifyEvent;
function IsOpacityStored: Boolean;
procedure SetExtendSelection(AValue: Boolean);
protected
procedure AddToSelection(AMapView: TMapView; APoint: TGPSPoint; AExtendSelection: Boolean);
procedure DeleteFromList(AMapView: TMapView; APoint: TGPSPoint);
procedure DoSelectionChange(AMapView: TMapView);
procedure DragStart(AMapView: TMapView);
procedure DragTo(AMapView: TMapView; X, Y: Integer);
procedure DragEnd(AMapView: TMapView);
procedure DrawPoint(AMapView: TMapView; ADrawingEngine: TMvCustomDrawingEngine;
AGpsPoint: TGPSPoint; AScreenPoint: TPoint; AMarkerSize: Integer);
procedure DrawRubberband(AMapView: TMapView);
procedure DrawSelection(AMapView: TMapView);
procedure FindContainerOfPoint(AMapView: TMapView; APoint: TGPSPoint; var AContainer: TGPSObj; var AIndex: Integer);
procedure FindMapCollection(AMapView: TMapView; APoint: TGPSPoint; var ACollection: TMapCollectionBase; var AIndex: Integer);
function RubberbandRect: TRect;
procedure RubberbandStart(AMapView: TMapView; X, Y: Integer);
procedure RubberbandTo(AMapView: TMapView; X, Y: Integer);
procedure RubberbandEnd(AMapView: TMapView; X, Y: Integer);
procedure ToggleSelected(AMapView: TMapView; APoint: TGPSPoint);
protected
procedure AfterDrawObjects(AMapView: TMapView; var {%H-}Handled: Boolean); override;
procedure MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton;
AShift: TShiftState; X,Y: Integer; var Handled: Boolean); override;
procedure MouseMove(AMapView: TMapView; {%H-}AShift: TShiftState;
X,Y: Integer; var Handled: Boolean); override;
procedure MouseUp(AMapView: TMapView; {%H-}Button: TMouseButton;
AShift: TShiftState; X,Y: Integer; var Handled: Boolean); override;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
function ConvertSelectedPointsToGPSArea(AMapView: TMapView; AreaID: Integer): TGPSArea;
function ConvertSelectedPointsToMapArea(AMapView: TMapView; ALayer: TMapLayer): TMapArea;
function ConvertSelectedPointsToGPSTrack(AMapView: TMapView; ATrackID: Integer): TGPSTrack;
function ConvertSelectedPointsToMapTrack(AMapView: TMapView; ALayer: TMapLayer): TMapTrack;
procedure DeleteSelectedPoints(AMapView: TMapView);
procedure MoveSelectionBy(AMapView: TMapView; dx, dy: Double);
procedure MoveSelectionBy(AMapView: TMapView; dx, dy: Integer);
function NewPoint(AMapView: TMapView; X, Y: Integer): TGPSPoint;
procedure SelectAllPointsOfShape(AMapView: TMapView; APoint: TGPSPoint; AExtendSelection: Boolean);
procedure SelectInRubberband(AMapView: TMapView);
property Selection: TGPSPointList read FSelection;
published
property ClickMode: TMarkerClickMode read FClickMode write FClickMode default DEFAULT_CLICKMODE;
property DragCursor: TCursor read FDragCursor write FDragCursor default crSizeAll;
property ExtendSelection: Boolean read FExtendSelection write SetExtendSelection default false;
property NewPointType: TMarkerNewPointType read FNewPointType write FNewPointType default nptGPSPoint;
property RubberbandBorderColor: TColor read FRubberbandBorderColor write FRubberbandBorderColor default DEFAULT_RUBBERBAND_BORDERCOLOR;
property RubberbandFillColor: TColor read FRubberbandFillColor write FRubberbandFillColor default DEFAULT_RUBBERBAND_FILLCOLOR;
property RubberbandOpacity: Single read FRubberbandOpacity write FRubberbandOpacity stored IsOpacityStored;
property Shift;
property OnDrawPoint: TMarkerDrawPointEvent read FOnDrawPoint write FOnDrawPoint;
property OnEndDrag: TNotifyEvent read FOnEndDrag write FOnEndDrag;
property OnNewPoint: TMarkerNewPointEvent read FOnNewPoint write FOnNewPoint;
property OnSelectionChange: TNotifyEvent read FOnSelectionChange write FOnSelectionChange;
property OnStartDrag: TMarkerStartDragEvent read FOnStartDrag write FOnStartDrag;
end;
{ TDraggableMarkerPlugin }
TDraggableMarkerPlugin = class;
TDraggableMarkerCanMoveEvent = function (Sender : TDraggableMarkerPlugin; AMarker : TGPSPoint) : Boolean of object;
TDraggableMarkerMovedEvent = procedure (Sender : TDraggableMarkerPlugin; AMarker : TGPSPoint; AOrgPosition : TRealPoint) of object;
{ TDraggableMarkerData }
PDraggableMarkerData = ^TDraggableMarkerData;
TDraggableMarkerData = record
FDraggedMarker : TGPSPoint;
FOrgPosition : TRealPoint;
end;
TDraggableMarkerPlugin = class(TMvMultiMapsPlugin)
private
const
DEFAULT_TOLERANCE = 5;
private
FDraggableMarkerCanMoveEvent : TDraggableMarkerCanMoveEvent;
FDraggableMarkerMovedEvent : TDraggableMarkerMovedEvent;
FDragMouseButton: TMouseButton;
FTolerance: Integer;
function GetFirstMarkerAtMousePos(const AMapView: TMapView; const AX, AY : Integer) : TGPSPoint;
function GetDraggedMarker(AMapView : TMapView) : TGPSPoint;
function GetOrgPosition(AMapView : TMapView): TRealPoint;
protected
procedure MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton; {%H-}Shift: TShiftState;
X, Y: Integer; var Handled: Boolean); override;
procedure MouseMove(AMapView: TMapView; {%H-}AShift: TShiftState; X,Y: Integer;
var Handled: Boolean); override;
procedure MouseUp(AMapView: TMapView; {%H-}Button: TMouseButton; {%H-}Shift: TShiftState;
{%H-}X, {%H-}Y: Integer; var Handled: Boolean); override;
public
constructor Create(AOwner: TComponent); override;
procedure Assign(Source: TPersistent); override;
property DraggedMarker[AMapView : TMapView] : TGPSPoint read GetDraggedMarker;
property OrgPosition[AMapView : TMapView] : TRealPoint read GetOrgPosition;
published
property DraggableMarkerCanMoveEvent : TDraggableMarkerCanMoveEvent read FDraggableMarkerCanMoveEvent write FDraggableMarkerCanMoveEvent;
property DraggableMarkerMovedEvent : TDraggableMarkerMovedEvent read FDraggableMarkerMovedEvent write FDraggableMarkerMovedEvent;
property DragMouseButton : TMouseButton read FDragMouseButton write FDragMouseButton default mbLeft;
property Tolerance: Integer read FTolerance write FTolerance default DEFAULT_TOLERANCE;
end;
implementation
uses
Types;
function IfThen(AValue: Boolean; ACursor1, ACursor2: TCursor): TCursor;
begin
if AValue then Result := ACursor1 else Result := ACursor2;
end;
{ TMarkerHintPlugin }
constructor TMarkerHintPlugin.Create(AOwner: TComponent);
begin
inherited;
FHintOffsetX := DEFAULT_HINT_OFFSET_X;
FHintOffsetY := DEFAULT_HINT_OFFSET_Y;
FHideInterval := DEFAULT_HIDE_INTERVAL;
FShowHint := true;
end;
function TMarkerHintPlugin.CreateHintWindow(AMapView: TMapView): THintWindow;
begin
if Assigned(FOnCreateHintWindow) then
FOnCreateHintWindow(AMapView, Result)
else
Result := THintWindow.Create(self);
end;
procedure TMarkerHintPlugin.DisplayHint(AMapView: TMapView; APoint: TGPSPoint;
X, Y: Integer);
var
hintTxt: String;
hintRct: TRect;
hintPt: TPoint;
dx, dy: Integer;
begin
if APoint.Name <> '' then
hintTxt := Format('%s' + LineEnding + '(%s / %s)', [
APoint.Name, LatToStr(APoint.Lat, true), LonToStr(APoint.Lon, true)
])
else
hintTxt := Format('(%s / %s)', [LatToStr(APoint.Lat, true), LonToStr(APoint.Lon, true)]);
if Assigned(FOnHint) then
FOnHint(AMapView, APoint, hintTxt);
if (hintTxt = '') or not FShowHint then
exit;
if not Assigned(FHintWindow) then
FHintWindow := CreateHintWindow(AMapView);
FHintWindow.AutoHide := FAutoHideHint;
FHintWindow.HideInterval := FHideInterval;
hintRct := FHintWindow.CalcHintRect(AMapView.Width, hintTxt, nil);
hintPt := AMapView.ClientToScreen(Point(X, Y));
if FHintOffsetX = -1 then
dx := - hintRct.Width div 2
else
dx := FHintOffsetX;
if FHintOffsetY = -1 then
dy := - hintRct.Height div 2
else
dy := FHintOffsetY;
OffsetRect(hintRct, hintPt.X + dx, hintPt.Y + dy);
FHintWindow.ActivateHint(hintRct, hintTxt);
end;
procedure TMarkerHintPlugin.HideHint;
begin
FreeAndNil(FHintWindow);
end;
procedure TMarkerHintPlugin.MouseMove(AMapView: TMapView; AShift: TShiftState;
X,Y: Integer; var Handled: Boolean);
var
gpsPoint: TGPSPoint;
begin
if Handled then
exit;
gpsPoint := FindNearestMarker(AMapView, X, Y);
if gpsPoint = nil then
HideHint
else
DisplayHint(AMapView, gpsPoint, X, Y);
end;
{ TCustomMarkerClickPlugin }
constructor TCustomMarkerClickPlugin.Create(AOwner: TComponent);
begin
inherited;
FCursor := crHandPoint;
FSavedCursor := crDefault;
FShift := [ssLeft];
end;
procedure TCustomMarkerClickPlugin.MouseDown(AMapView: TMapView;
Button: TMouseButton; AShift: TShiftState; X, Y: Integer; var Handled: Boolean);
var
canClick: Boolean;
begin
if Handled then
exit;
FOrigGPSPoint := FindNearestMarker(AMapView, X, Y);
if Assigned(FOrigGPSPoint) and (AShift = FShift) then
begin
if Assigned(FOnCanClick) then
begin
canClick := true;
FOnCanClick(AMapView, FOrigGPSPoint, canClick);
if not canClick then
exit;
end;
if Assigned(FOnMarkerClick) then
FOnMarkerClick(AMapView, FOrigGPSPoint);
FMouseDownOnMarker := true;
FMousePoint := Point(X, Y);
Handled := true;
end;
end;
procedure TCustomMarkerClickPlugin.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;
if not FMouseDownOnMarker then
AMapView.Cursor := IfThen(canClick, FCursor, FSavedCursor);
end;
procedure TCustomMarkerClickPlugin.MouseUp(AMapView: TMapView; Button: TMouseButton;
AShift: TShiftState; X, Y: Integer; var Handled: Boolean);
begin
FMouseDownOnMarker := false;
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 TCustomMarkerClickPlugin.SetMapView(AValue: TMapView);
begin
inherited;
if Assigned(MapView) then
FSavedCursor := MapView.Cursor
else
FSavedCursor := crDefault;
end;
{ TMarkerEditorPlugin }
type
TMarkerData = record
Lat, Lon: Double;
Elevation: Double;
DateTime: TDateTime;
end;
function GPSPointToMarkerData(P: TGPSPoint): TMarkerData;
begin
Result.Lat := P.Lat;
Result.Lon := P.Lon;
Result.Elevation := P.Elevation;
Result.DateTime := P.DateTime;
end;
procedure MarkerDataToGPSPoint(M: TMarkerData; P: TGPSPoint);
begin
P.Lat := M.Lat;
P.Lon := M.Lon;
P.Elevation := M.Elevation;
P.DateTime := M.DateTime;
end;
constructor TMarkerEditorPlugin.Create(AOwner: TComponent);
begin
inherited;
FClickMode := DEFAULT_CLICKMODE;
FDragCursor := crSizeAll;
FRubberbandBorderColor := DEFAULT_RUBBERBAND_BORDERCOLOR;
FRubberbandFillColor := DEFAULT_RUBBERBAND_FILLCOLOR;
FRubberbandOpacity := DEFAULT_RUBBERBAND_OPACITY;
FSelection := TGPSPointList.Create(false); // false = do not free objects
end;
destructor TMarkerEditorPlugin.Destroy;
begin
FSelection.Free;
inherited;
end;
procedure TMarkerEditorPlugin.AddToSelection(AMapView: TMapView;
APoint: TGPSPoint; AExtendSelection: Boolean);
var
idx: Integer;
begin
if AExtendSelection then
begin
idx := FSelection.IndexOf(APoint);
if idx > -1 then
FSelection.Move(idx, FSelection.Count-1)
else
FSelection.Add(APoint);
end else
begin
FSelection.Clear;
FSelection.Add(APoint);
end;
DoSelectionChange(AMapView);
end;
procedure TMarkerEditorPlugin.AfterDrawObjects(AMapView: TMapView;
var {%H-}Handled: Boolean);
begin
inherited;
DrawSelection(AMapView);
if FRubberbandMode then
DrawRubberband(AMapView);
end;
function TMarkerEditorPlugin.ConvertSelectedPointsToGPSArea(
AMapView: TMapView; AreaID: Integer): TGPSArea;
var
M: TMarkerData;
begin
if FSelection.Count < 3 then
raise EMvPluginException.Create('For an area, the selection must contain at least 3 points.');
Result := TGPSArea.Create;
while FSelection.Count > 0 do
begin
M := GPSPointToMarkerData(FSelection[0]);
DeleteFromList(AMapView, FSelection[0]);
FSelection.Delete(0);
Result.Points.Add(TGPSPoint.Create(M.Lon, M.Lat, M.Elevation, M.DateTime));
end;
AMapView.GPSItems.Add(Result, AreaID);
Update;
DoSelectionChange(AMapView);
end;
function TMarkerEditorPlugin.ConvertSelectedPointsToMapArea(
AMapView: TMapView; ALayer: TMapLayer): TMapArea;
var
M: TMarkerData;
P: TMapPoint;
begin
if FSelection.Count < 3 then
raise EMvPluginException.Create('For an area, the selection must contain at least 3 points.');
Result := ALayer.Areas.Add as TMapArea;
while FSelection.Count > 0 do
begin
M := GPSPointToMarkerData(FSelection[0]);
DeleteFromList(AMapView, FSelection[0]);
FSelection.Delete(0);
P := Result.Points.Add as TMapPoint;
MarkerDataToGPSPoint(M, TGPSPoint(P.GPSObj));
end;
Update;
DoSelectionChange(AMapView);
end;
function TMarkerEditorPlugin.ConvertSelectedPointsToGPSTrack(
AMapView: TMapView; ATrackID: Integer): TGPSTrack;
var
M: TMarkerData;
begin
if FSelection.Count < 2 then
raise EMvPluginException.Create('For a track, the selection must contain at least 2 points.');
Result := TGPSTrack.Create;
while FSelection.Count > 0 do
begin
M := GPSPointToMarkerData(FSelection[0]);
DeleteFromList(AMapView, FSelection[0]);
FSelection.Delete(0);
Result.Points.Add(TGPSPoint.Create(M.Lon, M.Lat, M.Elevation, M.DateTime));
end;
AMapView.GPSItems.Add(Result, ATrackID);
Update;
DoSelectionChange(AMapView);
end;
function TMarkerEditorPlugin.ConvertSelectedPointsToMapTrack(
AMapView: TMapView; ALayer: TMapLayer): TMapTrack;
var
M: TMarkerData;
P: TMapPoint;
begin
if FSelection.Count < 2 then
raise EMvPluginException.Create('For a track, the selection must contain at least 2 points.');
Result := ALayer.Tracks.Add as TMapTrack;
while FSelection.Count > 0 do
begin
M := GPSPointToMarkerData(FSelection[0]);
DeleteFromList(AMapView, FSelection[0]);
FSelection.Delete(0);
P := Result.Points.Add as TMapPoint;
MarkerDataToGPSPoint(M, TGPSPoint(P.GPSObj));
end;
Update;
DoSelectionChange(AMapView);
end;
{ Searches for the given point in all the point lists of the mapviewer
(GPSItems, Layers, Tracks, Areas, POIs). If found, the point is removed
from the list and destroyed. }
procedure TMarkerEditorPlugin.DeleteFromList(AMapView: TMapView;
APoint: TGPSPoint);
var
gpsObj: TGPSObj = nil;
collection: TMapCollectionBase = nil;
idx: Integer = -1;
begin
// Find the point in the GPS lists
FindContainerOfPoint(AMapView, APoint, gpsObj, idx);
if (gpsObj is TGPSObjectList) then
begin
TGPSObjectList(gpsObj).Delete(APoint);
exit;
end else
if (gpsObj is TGPSPolyLine) then
begin
TGPSPolyLine(gpsObj).Points.Delete(idx);
exit;
end;
// Find the point in the map collections
FindMapCollection(AMapView, APoint, collection, idx);
if collection <> nil then
collection.Delete(idx);
end;
procedure TMarkerEditorPlugin.DeleteSelectedPoints(AMapView: TMapView);
var
i: Integer;
begin
for i := FSelection.Count-1 downto 0 do
begin
DeleteFromList(AMapView, FSelection[i]);
FSelection.Delete(i);
end;
Update;
end;
procedure TMarkerEditorPlugin.DoSelectionChange(AMapView: TMapView);
begin
if Assigned(FOnSelectionChange) then
FOnSelectionChange(AMapView);
end;
procedure TMarkerEditorPlugin.DragStart(AMapView: TMapView);
var
i: Integer;
canDrag: Boolean;
begin
if Assigned(FOnStartDrag) then
begin
canDrag := true;
FOnStartDrag(AMapView, canDrag);
if not canDrag then exit;
end;
AMapView.Cursor := DragCursor;
FDragging := true;
// Save original selection point coordinates in case they must be restored later.
SetLength(FOrigSelection, FSelection.Count);
for i := 0 to High(FOrigSelection) do
FOrigSelection[i] := FSelection[i].RealPoint;
end;
procedure TMarkerEditorPlugin.DragTo(AMapView: TMapView; X, Y: Integer);
var
dX, dY: Integer;
begin
if FDragging then
begin
dX := X - FMousePoint.X;
dY := Y - FMousePoint.Y;
MoveSelectionBy(AMapView, dX, dY);
Update;
FMousePoint := Point(X, Y);
end;
end;
procedure TMarkerEditorPlugin.DragEnd(AMapView: TMapView);
begin
FDragging := false;
AMapView.Cursor := FSavedCursor;
if Assigned(FOnEndDrag) then
FOnEndDrag(AMapView);
end;
{ Draw the selection marker for the given point. The drawing engine already
has been setup for the correct settings. }
procedure TMarkerEditorPlugin.DrawPoint(AMapView: TMapView;
ADrawingEngine: TMvCustomDrawingEngine; AGpsPoint: TGPSPoint;
AScreenPoint: TPoint; AMarkerSize: Integer);
begin
if Assigned(FOnDrawPoint) then
FOnDrawPoint(AMapView, ADrawingEngine, AGPSPoint, AScreenPoint, AMarkerSize)
else
ADrawingEngine.Rectangle(
AScreenPoint.X - AMarkerSize,
AScreenPoint.Y - AMarkerSize,
AScreenPoint.X + AMarkerSize,
AScreenPoint.Y + AMarkerSize
);
end;
procedure TMarkerEditorPlugin.DrawRubberband(AMapView: TMapView);
var
DE: TMvCustomDrawingEngine;
R: TRect;
begin
DE := AMapView.DrawingEngine;
if FRubberbandFillColor = clNone then
DE.BrushStyle := bsClear
else
begin
DE.BrushColor := FRubberbandFillColor;
DE.BrushStyle := bsSolid;
end;
if FRubberbandBorderColor = clNone then
DE.PenStyle := psClear
else
begin
DE.PenColor := FRubberbandBorderColor;
DE.PenStyle := psSolid;
DE.PenWidth := 1;
end;
DE.Opacity := FRubberbandOpacity;
R := RubberbandRect;
DE.Rectangle(R.Left, R.Top, R.Right, R.Bottom);
end;
procedure TMarkerEditorPlugin.DrawSelection(AMapView: TMapView);
const
MARKER_SIZE = 5;
var
i, j: Integer;
P: TPoint;
markerSize: Integer;
DE: TMvCustomDrawingEngine;
pts: TPointArray;
begin
if FSelection.Count = 0 then
exit;
DE := AMapView.DrawingEngine;
DE.PenColor := clRed;
DE.PenStyle := psSolid;
DE.PenWidth := 2;
DE.BrushColor := clBlack;
DE.BrushStyle := bsSolid;
markerSize := AMapView.Scale96ToFont(MARKER_SIZE);
for i := 0 to FSelection.Count - 1 do
begin
if i = FSelection.Count - 1 then
begin
// The last point is marked as being "focused"
DE.PenWidth := 3;
DE.BrushColor := clLime;
inc(markerSize, 1);
end;
P := AMapView.LatLonToScreen(FSelection[i].RealPoint);
pts := AMapView.CyclicPointsOf(P);
for j := 0 to High(pts) do
DrawPoint(AMapView, DE, FSelection[i], pts[j], markerSize);
end;
end;
function IsSamePoint(AItem: TGPSObj; APoint: TGPSPoint): Boolean;
begin
Result := (AItem is TGPSPoint) and TGPSPoint(AItem).RealPoint.Equal(APoint.RealPoint);
end;
{ Tries to find the point in one of the gps-type lists.
If found, returns the found object and the index of the point in the list.
Otherwise, nil and -1 are returned, respectively. }
procedure TMarkerEditorPlugin.FindContainerOfPoint(AMapView: TMapView;
APoint: TGPSPoint; var AContainer: TGPSObj; var AIndex: Integer);
var
i, j, k: Integer;
gpsLayer: TGPSObjectList;
gpsPolyline: TGPSPolyLine;
item: TGPSObj;
begin
// Iterate over the 10 layers of GPSItems
for i := 0 to 9 do
begin
gpsLayer := AMapView.GPSLayer[i];
for j := 0 to gpsLayer.Count-1 do
begin
item := gpsLayer[j];
if IsSamePoint(item, APoint) then
begin
AContainer := gpsLayer;
AIndex := j;
exit;
end;
if (item is TGPSPolyline) then
begin
gpsPolyLine := TGPSPolyLine(item);
for k := 0 to gpsPolyLine.Points.Count-1 do
begin
item := gpsPolyLine.Points[k];
if IsSamePoint(item, APoint) then
begin
AContainer := gpsPolyLine;
AIndex := k;
exit;
end;
end;
end;
end;
end;
// If we get here the point has not been found.
AContainer := nil;
AIndex := -1;
end;
{ Tries to find the point in one of the map-type collections of all layers.
If found, returns the collection and the index of the point in the collection.
Otherwise, nil and -1 are returned, respectively. }
procedure TMarkerEditorPlugin.FindMapCollection(AMapView: TMapView;
APoint: TGPSPoint; var ACollection: TMapCollectionBase; var AIndex: Integer);
var
i, j, k: Integer;
p: TMapPoint;
mapLayer: TMapLayer;
mapTrack: TMapTrack;
mapArea: TMapArea;
begin
// Iterate over all map-layers
for i := 0 to AMapView.Layers.Count-1 do
begin
mapLayer := AMapView.Layers[i];
// Points of interest?
for j := 0 to mapLayer.PointsOfInterest.Count-1 do
begin
p := mapLayer.PointsOfInterest[j];
if IsSamePoint(p.GPSObj, APoint) then
begin
ACollection := mapLayer.PointsOfInterest;
AIndex := j;
exit;
end;
end;
// Tracks?
for j := 0 to mapLayer.Tracks.Count-1 do
begin
mapTrack := mapLayer.Tracks[j];
for k := 0 to mapTrack.Points.Count-1 do
begin
p := mapTrack.Points[k];
if IsSamePoint(p.GPSObj, APoint) then
begin
ACollection := mapTrack.Points;
AIndex := k;
exit;
end;
end;
end;
// Areas?
for j := 0 to mapLayer.Areas.Count-1 do
begin
mapArea := mapLayer.Areas[j];
for k := 0 to mapArea.Points.Count-1 do
begin
p := mapArea.Points[k];
if IsSamePoint(p.GPSObj, APoint) then
begin
ACollection := mapArea.Points;
AIndex := k;
exit;
end;
end;
end;
end;
// When we get here, the point has not been found.
ACollection := nil;
AIndex := -1;
end;
function TMarkerEditorPlugin.IsOpacityStored: Boolean;
begin
Result := FRubberbandOpacity <> DEFAULT_RUBBERBAND_OPACITY;
end;
{ Moves the selection by the given amound of pixels in x and y direction. }
procedure TMarkerEditorPlugin.MoveSelectionBy(AMapView: TMapView; dx, dy: Integer);
var
i: Integer;
P: TPoint;
rPt: TRealPoint;
begin
for i := 0 to FSelection.Count-1 do
begin
P := AMapView.LatLonToScreen(FSelection[i].RealPoint);
P.X := P.X + dx;
P.Y := P.Y + dy;
rPt := AMapView.ScreenToLatLon(P);
FSelection[i].MoveTo(rPt.Lon, rPt.Lat);
end;
end;
{ Moves the selection by the given amound of degrees in x and y direction }
procedure TMarkerEditorPlugin.MoveSelectionBy(AMapView: TMapView; dx, dy: Double);
var
i: Integer;
P: TPoint;
rPt: TRealPoint;
begin
for i := 0 to FSelection.Count-1 do
begin
rPt := FSelection[i].RealPoint;
rPt.Lon := FSelection[i].Lon + dX;
rPt.Lat := FSelection[i].Lat + dY;
if rPt.Lat > 90 then
rPt.Lat := rPt.Lat - 180;
if rPt.Lon < -90 then
rPt.Lon := rPt.Lon + 180;
P := AMapView.LatLonToScreen(rPt);
rPt := AMapView.ScreenToLatLon(P);
FSelection[i].MoveTo(rPt.Lon, rPt.Lat);
end;
end;
procedure TMarkerEditorPlugin.MouseDown(AMapView: TMapView;
{%H-}Button: TMouseButton; {%H-}AShift: TShiftState;
X, Y: Integer; var Handled: Boolean);
begin
inherited;
if FMouseDownOnMarker then
begin
case FClickMode of
cmSelectPoint:
AddToSelection(AMapView, FOrigGPSPoint, FExtendSelection);
cmSelectShape:
SelectAllPointsOfShape(AMapView, FOrigGPSPoint, FExtendSelection);
cmTogglePoint:
ToggleSelected(AMapView, FOrigGPSPoint);
end;
Update;
Handled := true;
end else
begin
case FClickMode of
cmNewPoint:
begin
FOrigGPSPoint := NewPoint(AMapView, X, Y);
AddToSelection(AMapView, FOrigGPSPoint, FExtendSelection);
Handled := true;
end;
cmRubberband:
;
else
FSelection.Clear;
end;
Update;
end;
end;
procedure TMarkerEditorPlugin.MouseMove(AMapView: TMapView;
{%H-}AShift: TShiftState; X,Y: Integer; var Handled: Boolean);
const
SENSITIVITY = 5;
var
R: TRect;
begin
inherited;
if FMouseDownOnMarker then
begin
if not FDragging then
begin
// The mouse must be moved by more than SENSITIVITY pixels for dragging to
// start
R := Rect(X - SENSITIVITY, Y - SENSITIVITY, X + SENSITIVITY, Y + SENSITIVITY);
if not PtInRect(R, Point(X, Y)) then
begin
FDragging := false;
exit;
end;
DragStart(AMapView);
end;
DragTo(AMapView, X, Y);
Handled := true;
end else
if not FDragging and (FClickMode = cmRubberband) and (Shift = AShift) then
begin
if not FRubberbandMode then
RubberbandStart(AMapView, X, Y)
else
RubberbandTo(AMapView, X, Y);
Handled := true;
end;
end;
procedure TMarkerEditorPlugin.MouseUp(AMapView: TMapView;
{%H-}Button: TMouseButton; {%H-}AShift: TShiftState;
X, Y: Integer; var Handled: Boolean);
begin
inherited;
if FDragging then
DragEnd(AMapView);
if FRubberbandMode then
RubberbandEnd(AMapView, X, Y);
end;
function TMarkerEditorPlugin.NewPoint(AMapView: TMapView;
X, Y: Integer): TGPSPoint;
const
ID = 4000;
var
gpsPoint: TGPSPoint;
mapPoint: TMapPoint;
layer: TMapLayer;
rPt: TRealPoint;
begin
rPt := AMapView.ScreenToLatLon(Point(X, Y));
case FNewPointType of
nptGPSPoint:
begin
gpsPoint := TGPSPointOfInterest.CreateFrom(rPt);
AMapView.GPSItems.Add(gpsPoint, ID);
Result := gpsPoint;
end;
nptMapPoint:
begin
if AMapView.Layers.Count = 0 then
layer := AMapView.Layers.Add as TMapLayer
else
layer := AMapView.Layers[0];
mapPoint := layer.PointsOfInterest.Add as TMapPoint;
mapPoint.RealPoint := rPt;
Result := mapPoint.GPSObj as TGPSPoint;
end;
end;
if Assigned(FOnNewPoint) then
FOnNewPoint(AMapView, Result);
end;
procedure TMarkerEditorPlugin.RubberbandEnd(AMapView: TMapView; X, Y: Integer);
begin
FRubberbandMode := false;
FRubberbandEndPt := Point(X, Y);
SelectInRubberband(AMapview);
Update;
end;
function TMarkerEditorPlugin.RubberbandRect: TRect;
begin
Result.TopLeft := FRubberBandStartPt;
Result.BottomRight := FRubberbandEndPt;
Result.NormalizeRect;
end;
procedure TMarkerEditorPlugin.RubberbandStart(AMapView: TMapView; X, Y: Integer);
begin
FRubberbandMode := true;
FRubberbandStartPt := Point(X, Y);
FRubberbandEndPt := Point(X, Y);
end;
procedure TMarkerEditorPlugin.RubberbandTo(AMapView: TMapView; X, Y: Integer);
begin
FRubberbandEndPt := Point(X, Y);
Update;
end;
procedure TMarkerEditorPlugin.SelectAllPointsOfShape(AMapView: TMapView;
APoint: TGPSPoint; AExtendSelection: Boolean);
var
obj: TGPSObj = nil;
collection: TMapCollectionBase = nil;
idx: Integer = -1;
i: Integer;
gpsPolyline: TGPSPolyLine;
item: TGPSObj;
p: TMapPoint;
procedure Finished;
begin
AddToSelection(AMapView, APoint, true); // Mark APoint as being focused
Update;
DoSelectionChange(AMapView);
end;
begin
if not AExtendSelection then
FSelection.Clear;
// Find point in gpsObj-type of containers
FindContainerOfPoint(AMapView, APoint, obj, idx);
// Is is a point of interest?
if obj is TGPSObjectList then
begin
item := TGPSObjectList(obj).Items[idx];
AddToSelection(AMapView, TGPSPoint(item), true);
Finished;
exit;
end else
// ... or a track / layer?
if obj is TGPSPolyLine then
begin
gpsPolyLine := TGPSPolyLine(obj);
for i := 0 to gpsPolyLine.Points.Count-1 do
begin
item := TGPSPoint(gpsPolyLine.Points[i]);
AddToSelection(AMapView, TGPSPoint(item), true);
end;
Finished;
exit;
end;
// Find point in map-type collections
FindMapCollection(AMapView, APoint, collection, idx);
if collection is TMapPointsOfInterest then
begin
p := collection.Items[idx] as TMapPoint;
AddToSelection(AMapView, TGPSPoint(p.GPSObj), true);
end else
if collection <> nil then
begin
for i := 0 to collection.Count-1 do
begin
p := collection.Items[i] as TMapPoint;
AddToSelection(AMapView, TGPSPoint(p.GPSObj), true);
end;
end;
Finished;
end;
procedure TMarkerEditorPlugin.SelectInRubberband(AMapView: TMapView);
var
area: TRealArea;
R: TRect;
pts: TGPSObjArray;
i: Integer;
begin
if not FExtendSelection then
FSelection.Clear;
R := RubberbandRect;
area.TopLeft := AMapView.ScreenToLatLon(R.TopLeft);
area.BottomRight := AMapView.ScreenToLatLon(R.BottomRight);
pts := AMapView.VisiblePointsInArea(area, PointTypes);
for i := 0 to High(pts) do
AddToSelection(AMapView, TGPSPoint(pts[i]), true);
Update;
end;
procedure TMarkerEditorPlugin.SetExtendSelection(AValue: Boolean);
begin
if FExtendSelection = AValue then exit;
FExtendSelection := AValue;
if (not FExtendSelection) then
begin
FSelection.Clear;
if (FOrigGPSPoint <> nil) then
FSelection.Add(FOrigGPSPoint);
end;
Update;
end;
procedure TMarkerEditorPlugin.ToggleSelected(AMapView: TMapView;
APoint: TGPSPoint);
var
idx: Integer;
begin
idx := FSelection.IndexOf(APoint);
if idx = -1 then
begin
if not FExtendSelection then
FSelection.Clear;
FSelection.Add(APoint);
end else
FSelection.Delete(idx);
DoSelectionChange(AMapView);
end;
{ TDraggableMarkerPlugin }
constructor TDraggableMarkerPlugin.Create(AOwner: TComponent);
begin
inherited;
FTolerance := DEFAULT_TOLERANCE;
end;
procedure TDraggableMarkerPlugin.Assign(Source: TPersistent);
begin
if Source is TDraggableMarkerPlugin then
begin
FDraggableMarkerCanMoveEvent := TDraggableMarkerPlugin(Source).DraggableMarkerCanMoveEvent;
FDraggableMarkerMovedEvent := TDraggableMarkerPlugin(Source).DraggableMarkerMovedEvent;
FDragMouseButton := TDraggableMarkerPlugin(Source).DragMouseButton;
FTolerance := TDraggableMarkerPlugin(Source).Tolerance;
end;
inherited;
end;
function TDraggableMarkerPlugin.GetFirstMarkerAtMousePos(const AMapView: TMapView;
const AX, AY: Integer): TGPSPoint;
function FindInList(AGpsList: TGpsObjList): TGpsPoint;
var
i: Integer;
begin
if Assigned(AGpsList) then
for i := AGpsList.Count-1 downto 0 do
begin
if (AGpsList[i] is TGpsPoint) then
begin
Result := TGpsPoint(AGpsList[i]);
if (not Assigned(FDraggableMarkerCanMoveEvent)) or
DraggableMarkerCanMoveEvent(Self, Result)
then
exit;
end;
end;
Result := nil;
end;
var
aArea : TRealArea;
gpsList: TGpsObjList;
layer: TMapLayer;
i : Integer;
begin
Result := Nil;
aArea.TopLeft := AMapView.ScreenToLatLon(Point(AX - FTolerance, AY - FTolerance));
aArea.BottomRight := AMapView.ScreenToLatLon(Point(AX + FTolerance, AY + FTolerance));
// Search in GPSItems for all gps-type-of-points
gpsList := AMapView.GPSItems.GetObjectsInArea(aArea);
try
Result := FindInList(gpsList);
if Result <> nil then
exit;
finally
gpsList.Free;
end;
// Search in all layers for all map-type points
for i := AMapView.Layers.Count-1 downto 0 do
begin
layer := AMapView.Layers[i];
gpsList := layer.GetObjectsInArea(aArea);
try
Result := FindInList(gpsList);
if Result <> nil then
exit;
finally
gpsList.Free;
end;
end;
end;
function TDraggableMarkerPlugin.GetDraggedMarker(AMapView: TMapView): TGPSPoint;
var
lDraggableMarkerData : TDraggableMarkerData;
cnt : Integer;
begin
Result := Nil;
cnt := GetMapViewData(AMapView,lDraggableMarkerData,SizeOf(lDraggableMarkerData));
if (cnt >= SizeOf(lDraggableMarkerData)) then
Result := lDraggableMarkerData.FDraggedMarker;
end;
function TDraggableMarkerPlugin.GetOrgPosition(AMapView : TMapView): TRealPoint;
var
lDraggableMarkerData : TDraggableMarkerData;
cnt : Integer;
begin
Result.InitXY(0.0,0.0);
cnt := GetMapViewData(AMapView,lDraggableMarkerData,SizeOf(lDraggableMarkerData));
if (cnt >= SizeOf(lDraggableMarkerData)) then
Result := lDraggableMarkerData.FOrgPosition;
end;
procedure TDraggableMarkerPlugin.MouseDown(AMapView: TMapView; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer; var Handled: Boolean);
var
lDraggableMarkerData : TDraggableMarkerData;
begin
if Handled then Exit;
if not MapViewEnabled[AMapView] then Exit;
if FDragMouseButton <> Button then Exit;
lDraggableMarkerData.FDraggedMarker := GetFirstMarkerAtMousePos(AMapView,X,Y);
if Assigned(lDraggableMarkerData.FDraggedMarker) then
begin
lDraggableMarkerData.FOrgPosition.Lon:= lDraggableMarkerData.FDraggedMarker.Lon;
lDraggableMarkerData.FOrgPosition.Lat:= lDraggableMarkerData.FDraggedMarker.Lat;
SetMapViewData(AMapView,lDraggableMarkerData,SizeOf(lDraggableMarkerData));
Handled := True;
end;
end;
procedure TDraggableMarkerPlugin.MouseMove(AMapView: TMapView;
AShift: TShiftState; X, Y: Integer; var Handled: Boolean);
var
pt : TPoint;
rpt : TRealPoint;
ele : Double;
dt : TDateTime;
lDraggableMarkerData : TDraggableMarkerData;
cnt : Integer;
begin
cnt := GetMapViewData(AMapView,lDraggableMarkerData,SizeOf(lDraggableMarkerData));
if not MapViewEnabled[AMapView] then Exit;
if (cnt >= SizeOf(lDraggableMarkerData)) and
Assigned(lDraggableMarkerData.FDraggedMarker) then
begin
pt.X := X;
pt.Y := Y;
rpt := AMapView.ScreenToLatLon(pt);
ele := lDraggableMarkerData.FDraggedMarker.Elevation;
dt := lDraggableMarkerData.FDraggedMarker.DateTime;
lDraggableMarkerData.FDraggedMarker.MoveTo(rpt.Lon, rpt.Lat,ele,dt);
AMapView.Invalidate;
Handled := True; // Prevent the dragging of the map!!
end
else
begin
if Assigned(GetFirstMarkerAtMousePos(AMapView,X,Y)) then
begin
AMapView.Cursor := crHandPoint;
Handled := True;
end
else if not Handled then
AMapView.Cursor := crDefault;
end
end;
procedure TDraggableMarkerPlugin.MouseUp(AMapView: TMapView; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer; var Handled: Boolean);
var
lpDraggableMarkerData : PDraggableMarkerData;
begin
if not MapViewEnabled[AMapView] then Exit;
if FDragMouseButton <> Button then Exit;
lpDraggableMarkerData := MapViewDataPtr[AMapView];
if Assigned(lpDraggableMarkerData) and Assigned(lpDraggableMarkerData^.FDraggedMarker) then
begin
if Assigned(FDraggableMarkerMovedEvent) then
FDraggableMarkerMovedEvent(Self,lpDraggableMarkerData^.FDraggedMarker,lpDraggableMarkerData^.FOrgPosition);
Handled := True;
lpDraggableMarkerData^.FDraggedMarker := Nil;
end;
end;
initialization
RegisterPluginClass(TMarkerHintPlugin, 'Marker hint');
RegisterPluginClass(TMarkerClickPlugin, 'Marker click');
RegisterPluginClass(TMarkerEditorPlugin, 'Marker select and drag');
RegisterPluginClass(TDraggableMarkerPlugin, 'Draggable marker');
end.