unit mvMarkerPlugins; {$mode objfpc}{$H+} interface uses Classes, SysUtils, Contnrs, 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; { TMarkerClickPlugin } TMarkerCanClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint; var CanClick: Boolean) of object; TMarkerClickEvent = procedure (AMapView: TMapView; APoint: TGPSPoint) of object; TMarkerClickPlugin = class(TMvMarkerPlugin) private FCursor: TCursor; FShift: TShiftState; FOnCanClick: TMarkerCanClickEvent; FOnMarkerClick: TMarkerClickEvent; 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; public constructor Create(AOwner: TComponent); override; published property Cursor: TCursor read FCursor write FCursor default crHandPoint; property Shift: TShiftState read FShift write FShift default [ssLeft]; property OnCanClick: TMarkerCanClickEvent read FOnCanClick write FOnCanClick; property OnMarkerClick: TMarkerClickEvent read FOnMarkerClick write FOnMarkerClick; end; { TMarkerSelectAndDragPlugin } TMarkerDrawPointEvent = procedure (AMapView: TMapView; ADrawingEngine: TMvCustomDrawingEngine; AGPSPoint: TGPSPoint; AScreenPoint: TPoint; AMarkerSize: Integer) of object; TMarkerClickMode = (mcmAddToSelection, mcmToggleSelection); TMarkerSelectAndDragPlugin = class(TMarkerClickPlugin) private FClickMode: TMarkerClickMode; FDragCursor: TCursor; FDragging: Boolean; FMultiSelect: Boolean; FSelection: TGPSPointList; FOrigSelection: array of TRealPoint; // Selection before dragging starts FOnDrawPoint: TMarkerDrawPointEvent; FOnSelectionChange: TNotifyEvent; procedure SetMultiSelect(AValue: Boolean); protected procedure AddToSelection(AMapView: TMapView; APoint: TGPSPoint); 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 DrawSelection(AMapView: TMapView); procedure MoveSelectionBy(AMapView: TMapView; dx, dy: 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 ConvertSelectedPointsToMapArea(AMapView: TMapView; ALayer: TMapLayer): TMapArea; function ConvertSelectedPointsToMapTrack(AMapView: TMapView; ALayer: TMapLayer): TMapTrack; procedure DeleteSelectedPoints(AMapView: TMapView); property Selection: TGPSPointList read FSelection; published property ClickMode: TMarkerClickMode read FClickMode write FClickMode default mcmAddToSelection; property DragCursor: TCursor read FDragCursor write FDragCursor default crSizeAll; property MultiSelect: Boolean read FMultiSelect write SetMultiSelect default false; property OnDrawPoint: TMarkerDrawPointEvent read FOnDrawPoint write FOnDrawPoint; property OnSelectionChange: TNotifyEvent read FOnSelectionChange write FOnSelectionChange; 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; { TMarkerClickPlugin } constructor TMarkerClickPlugin.Create(AOwner: TComponent); begin inherited; FCursor := crHandPoint; FSavedCursor := crDefault; FShift := [ssLeft]; end; procedure TMarkerClickPlugin.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 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; if not FMouseDownOnMarker then AMapView.Cursor := IfThen(canClick, FCursor, FSavedCursor); end; procedure TMarkerClickPlugin.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 TMarkerClickPlugin.SetMapView(AValue: TMapView); begin inherited; if Assigned(MapView) then FSavedCursor := MapView.Cursor else FSavedCursor := crDefault; end; { TMarkerSelectAndDragPlugin } 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 TMarkerSelectAndDragPlugin.Create(AOwner: TComponent); begin inherited; FDragCursor := crSizeAll; FSelection := TGPSPointList.Create(false); // false = do not free objects end; destructor TMarkerSelectAndDragPlugin.Destroy; begin FSelection.Free; inherited; end; procedure TMarkerSelectAndDragPlugin.AddToSelection(AMapView: TMapView; APoint: TGPSPoint); var idx: Integer; begin if FMultiSelect 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; function TMarkerSelectAndDragPlugin.ConvertSelectedPointsToMapArea( AMapView: TMapView; ALayer: TMapLayer): TMapArea; var M: TMarkerData; P: TMapPoint; begin if FSelection.Count < 2 then raise EMvPluginException.Create('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 TMarkerSelectAndDragPlugin.ConvertSelectedPointsToMapTrack( AMapView: TMapView; ALayer: TMapLayer): TMapTrack; var M: TMarkerData; P: TMapPoint; begin if FSelection.Count < 2 then raise EMvPluginException.Create('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; procedure TMarkerSelectAndDragPlugin.AfterDrawObjects(AMapView: TMapView; var {%H-}Handled: Boolean); begin inherited; DrawSelection(AMapView); end; procedure TMarkerSelectAndDragPlugin.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 TMarkerSelectAndDragPlugin.DoSelectionChange(AMapView: TMapView); begin if Assigned(FOnSelectionChange) then FOnSelectionChange(AMapView); end; procedure TMarkerSelectAndDragPlugin.DragStart(AMapView: TMapView); var i: Integer; begin 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 TMarkerSelectAndDragPlugin.DragTo(AMapView: TMapView; X, Y: Integer); var dX, dY: Integer; begin if FDragging then begin // AMapView.Cursor := DragCursor; dX := X - FMousePoint.X; dY := Y - FMousePoint.Y; MoveSelectionBy(AMapView, dX, dY); Update; FMousePoint := Point(X, Y); end; end; procedure TMarkerSelectAndDragPlugin.DragEnd(AMapView: TMapView); begin FDragging := false; AMapView.Cursor := FSavedCursor; end; { Draw the selection marker for the given point. The drawing engine already has been setup for the correct settings. } procedure TMarkerSelectAndDragPlugin.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 TMarkerSelectAndDragPlugin.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; procedure TMarkerSelectAndDragPlugin.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; procedure TMarkerSelectAndDragPlugin.MouseDown(AMapView: TMapView; {%H-}Button: TMouseButton; {%H-}AShift: TShiftState; X, Y: Integer; var Handled: Boolean); begin inherited; if FMouseDownOnMarker then begin case FClickMode of mcmAddToSelection : AddToSelection(AMapView, FOrigGPSPoint); mcmToggleSelection: ToggleSelected(AMapView, FOrigGPSPoint); end; Update; Handled := true; end else begin FSelection.Clear; Update; end; end; procedure TMarkerSelectAndDragPlugin.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; end; procedure TMarkerSelectAndDragPlugin.MouseUp(AMapView: TMapView; {%H-}Button: TMouseButton; {%H-}AShift: TShiftState; X, Y: Integer; var Handled: Boolean); begin inherited; if FDragging then DragEnd(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 TMarkerSelectAndDragPlugin.DeleteFromList(AMapView: TMapView; APoint: TGPSPoint); var i, j, k: Integer; gpsLayer: TGPSObjectList; gpsPolyline: TGPSPolyLine; item: TGPSObj; p: TMapPoint; mapLayer: TMapLayer; mapTrack: TMapTrack; mapArea: TMapArea; function IsSamePoint(AItem: TGPSObj): Boolean; begin Result := (AItem is TGPSPoint) and TGPSPoint(AItem).RealPoint.Equal(APoint.RealPoint); end; begin // Check 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) then begin gpsLayer.Delete(item); 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) then begin gpsPolyLine.Points.Delete(k); exit; end; end; end; end; end; // Check the 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) then begin mapLayer.PointsOfInterest.Delete(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) then begin mapTrack.Points.Delete(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) then begin mapArea.Points.Delete(k); exit; end; end; end; end; end; procedure TMarkerSelectAndDragPlugin.SetMultiSelect(AValue: Boolean); begin if FMultiSelect = AValue then exit; FMultiSelect := AValue; if not FMultiSelect then begin FSelection.Clear; FSelection.Add(FOrigGPSPoint); end; Update; end; procedure TMarkerSelectAndDragPlugin.ToggleSelected(AMapView: TMapView; APoint: TGPSPoint); var idx: Integer; begin idx := FSelection.IndexOf(APoint); if idx = -1 then begin if not FMultiSelect 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(TMarkerSelectAndDragPlugin, 'Marker select and drag'); RegisterPluginClass(TDraggableMarkerPlugin, 'Draggable marker'); end.