
git-svn-id: https://svn.code.sf.net/p/lazarus-ccr/svn@9703 8e941d3f-bd1b-0410-a28a-d453659cc2b4
1441 lines
44 KiB
ObjectPascal
1441 lines
44 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, LCLIntf, Forms, Dialogs,
|
|
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, cmRubberband);
|
|
|
|
TMarkerNewPointType = (nptGPSPoint, nptMapPoint);
|
|
|
|
TMarkerEditorPlugin = class(TMarkerClickPlugin)
|
|
private
|
|
type
|
|
TPluginStateEnum = (psDragging, psRubberBandMode, psExtendingSelection, psPendingUnselect);
|
|
TPluginState = set of TPluginStateEnum;
|
|
const
|
|
DEFAULT_CLICKMODE = cmSelectPoint;
|
|
DEFAULT_RUBBERBAND_BORDERCOLOR = clGray;
|
|
DEFAULT_RUBBERBAND_FILLCOLOR = clWhite;
|
|
DEFAULT_RUBBERBAND_OPACITY = 0.55;
|
|
DEFAULT_SHIFT_FOR_NEW_POINT = [ssRight];
|
|
DEFAULT_SHIFT_TO_SELECT_POINT = [ssLeft];
|
|
DEFAULT_SHIFT_TO_SELECT_SHAPE = [ssLeft, ssShift];
|
|
DEFAULT_SHIFT_TO_SELECT_BY_RUBBERBAND = [ssLeft, ssAlt];
|
|
DEFAULT_SHIFT_TO_EXTEND_SELECTION = [ssCtrl]; // to be used in addition to the others
|
|
private
|
|
FClickMode: TMarkerClickMode;
|
|
FDragCursor: TCursor;
|
|
FNewPointType: TMarkerNewPointType;
|
|
FRubberbandBorderColor: TColor;
|
|
FRubberbandFillColor: TColor;
|
|
FRubberbandOpacity: Single;
|
|
FRubberbandStartPt: TPoint;
|
|
FRubberbandEndPt: TPoint;
|
|
FSelection: TGPSPointList;
|
|
FShiftForNewPoint: TShiftState;
|
|
FShiftToSelectPoint: TShiftState;
|
|
FShiftToSelectShape: TShiftState;
|
|
FShiftToSelectByRubberband: TShiftState;
|
|
FShiftToExtendSelection: TShiftState;
|
|
FState: TPluginState;
|
|
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 IsShiftOfClickMode(AShift: TShiftState; AClickMode: TMarkerClickMode): Boolean;
|
|
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 UnselectPoint(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 ShiftForNewPoint: TShiftState read FShiftForNewPoint write FShiftForNewPoint
|
|
default DEFAULT_SHIFT_FOR_NEW_POINT;
|
|
property ShiftToExtendSelection: TShiftState read FShiftToExtendSelection write FShiftToExtendSelection
|
|
default DEFAULT_SHIFT_TO_EXTEND_SELECTION;
|
|
property ShiftToSelectPoint: TShiftState read FShiftToSelectPoint write FShiftToSelectPoint
|
|
default DEFAULT_SHIFT_TO_SELECT_POINT;
|
|
property ShiftToSelectShape: TShiftState read FShiftToSelectShape write FShiftToSelectShape
|
|
default DEFAULT_SHIFT_TO_SELECT_SHAPE;
|
|
property ShiftToSelectByRubberband: TShiftState read FShiftToSelectByRubberband write FShiftToSelectByRubberband
|
|
default DEFAULT_SHIFT_TO_SELECT_BY_RUBBERBAND;
|
|
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
|
|
FShiftForNewPoint := DEFAULT_SHIFT_FOR_NEW_POINT;
|
|
FShiftToSelectPoint := DEFAULT_SHIFT_TO_SELECT_POINT;
|
|
FShiftToSelectShape := DEFAULT_SHIFT_TO_SELECT_SHAPE;
|
|
FShiftToSelectByRubberband := DEFAULT_SHIFT_TO_SELECT_BY_RUBBERBAND;
|
|
FShiftToExtendSelection := [ssCtrl]; // modifier to the others to add point(s) to current selection
|
|
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 (psRubberBandMode in FState) 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;
|
|
FState := FState + [psDragging] - [psPendingUnselect];
|
|
AMapView.Cursor := DragCursor;
|
|
// 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 (psDragging in FState) 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
|
|
Exclude(FState, psDragging);
|
|
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;
|
|
|
|
function TMarkerEditorPlugin.IsShiftOfClickMode(AShift: TShiftState;
|
|
AClickMode: TMarkerClickMode): Boolean;
|
|
begin
|
|
case AClickMode of
|
|
cmNewPoint:
|
|
Result := (AShift = FShiftForNewPoint) or
|
|
(AShift = FShiftForNewPoint + FShiftToExtendSelection);
|
|
cmSelectPoint:
|
|
Result := (AShift = FShiftToSelectPoint) or
|
|
(AShift = FShiftToSelectPoint + FShiftToExtendSelection);
|
|
cmSelectShape:
|
|
Result := (AShift = FShiftToSelectShape) or
|
|
(AShift = FShiftToSelectShape + FShiftToExtendSelection);
|
|
cmRubberband:
|
|
Result := (AShift = FShiftToSelectByRubberband) or
|
|
(AShift = FShiftToSelectByRubberBand + FShiftToExtendSelection);
|
|
end;
|
|
end;
|
|
|
|
{ Moves the selection by the given pixel count 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 amount 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;
|
|
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
|
|
if IsShiftOfClickMode(AShift, cmNewPoint) then
|
|
begin
|
|
FClickMode := cmNewPoint;
|
|
Shift := FShiftForNewPoint;
|
|
end else
|
|
if IsShiftOfClickMode(AShift, cmRubberband) then
|
|
begin
|
|
FClickMode := cmRubberband;
|
|
Shift := FShiftToSelectPoint;
|
|
end else
|
|
if IsShiftOfClickMode(AShift, cmSelectShape) then
|
|
begin
|
|
FClickMode := cmSelectShape;
|
|
Shift := FShiftToSelectShape;
|
|
end else
|
|
if IsShiftOfClickMode(AShift, cmSelectPoint) then
|
|
begin
|
|
FClickMode := cmSelectPoint;
|
|
Shift := FShiftToSelectPoint;
|
|
end else
|
|
exit;
|
|
|
|
if (AShift * FShiftToExtendSelection = FShiftToExtendSelection) then
|
|
Include(FState, psExtendingSelection)
|
|
else
|
|
Exclude(FState, psExtendingSelection);
|
|
if (psExtendingSelection in FState) then
|
|
Shift := Shift + FShiftToExtendSelection;
|
|
|
|
inherited;
|
|
|
|
if FMouseDownOnMarker then
|
|
begin
|
|
case FClickMode of
|
|
cmSelectPoint:
|
|
if FSelection.IndexOf(FOrigGPSPoint) = -1 then
|
|
begin
|
|
AddToSelection(AMapView, FOrigGPSPoint, psExtendingSelection in FState);
|
|
Exclude(FState, psPendingUnselect);
|
|
end else
|
|
if (psExtendingSelection in FState) then
|
|
Include(FState, psPendingUnselect);
|
|
cmSelectShape:
|
|
SelectAllPointsOfShape(AMapView, FOrigGPSPoint, psExtendingSelection in FState);
|
|
end;
|
|
Update;
|
|
Handled := true;
|
|
end else
|
|
begin
|
|
case FClickMode of
|
|
cmNewPoint:
|
|
begin
|
|
FOrigGPSPoint := NewPoint(AMapView, X, Y);
|
|
AddToSelection(AMapView, FOrigGPSPoint, psExtendingSelection in FState);
|
|
Handled := true;
|
|
end;
|
|
cmRubberband:
|
|
;
|
|
else
|
|
if (FSelection.Count > 0) then
|
|
begin
|
|
if (MessageDlg('Do you really want to unselect these points?', mtConfirmation, [mbYes, mbNo], 0) = mrYes) then
|
|
FSelection.Clear;
|
|
Handled := true;
|
|
end;
|
|
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 (psDragging in FState) 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
|
|
exit;
|
|
DragStart(AMapView);
|
|
end;
|
|
DragTo(AMapView, X, Y);
|
|
Handled := true;
|
|
end else
|
|
if not (psDragging in FState) and (FClickMode = cmRubberband) and IsShiftOfClickMode(AShift, cmRubberband) then
|
|
begin
|
|
if not (psRubberbandMode in FState) 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 (psDragging in FState) then
|
|
DragEnd(AMapView);
|
|
if (psRubberbandMode in FState) then
|
|
RubberbandEnd(AMapView, X, Y);
|
|
if (psPendingUnselect in FState) then
|
|
UnselectPoint(AMapView, FOrigGPSPoint);
|
|
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
|
|
Exclude(FState, psRubberbandMode);
|
|
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
|
|
Include(FState, psRubberbandMode);
|
|
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 (psExtendingSelection in FState) 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 AValue = (psExtendingSelection in FState) then exit;
|
|
if AValue then
|
|
Include(FState, psExtendingSelection)
|
|
else
|
|
Exclude(FState, psExtendingSelection);
|
|
if not (psExtendingSelection in FState) then
|
|
begin
|
|
FSelection.Clear;
|
|
if (FOrigGPSPoint <> nil) then
|
|
FSelection.Add(FOrigGPSPoint);
|
|
end;
|
|
Update;
|
|
end;
|
|
|
|
procedure TMarkerEditorPlugin.UnselectPoint(AMapView: TMapView;
|
|
APoint: TGPSPoint);
|
|
var
|
|
idx: Integer;
|
|
begin
|
|
idx := FSelection.IndexOf(APoint);
|
|
if idx > -1 then
|
|
begin
|
|
FSelection.Delete(idx);
|
|
DoSelectionChange(AMapView);
|
|
end;
|
|
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.
|
|
|