AW: Google Maps über COM (Component Object Model)
Hallo Thom,
mit deinem Tip komme ich jetzt an die detaillierten Datenpunkte mit folgendem Code
Delphi-Quellcode:
jetzt wird es noch spannend die entsprechenden Höhewerte mittels "PathElevationRequest" bzw. "LocationElevationRequest" zu ermitteln. Bei meiner Demo-Route entstehen für den Track ca. 5100 Wertepaare LAT/LNG auf ca. 130km. Du hattest mir schon mal bei der Ermittlung geholfen. Damals war bei ca. 1000 Anfragen eine Pause von ca. 8 Sekunden notwendig. Werde mal verschiedene Ansätze versuchen. Der BikeRoutToaster scheint keine Pause zu verwenden, das Höhenprofil wird parallel erzeugt. Mir fehlen schon die Grundkenntnisse in JavaScript um an den Quelltext zu gelangen bzw. den zu lesen :lol:
var i,k,l : Integer;
f1,f2 : Double; MyRoute : TDirectionsRoute; TP : TTrackPoint; s : String; begin LB.Clear; TD.Clear; f1 := 0.0; with Script do begin MyRoute := DirectionsRenderer[0].Directions.Routes[0]; for i := 0 to MyRoute.Legs.Count-1 do begin s := MyRoute.Legs[i].StartAddress+' nach '+MyRoute.Legs[i].EndAddress; LB.Items.Add( Format( '[%d] %s',[i+1,s] )); LB.Items.Add( '' ); for k := 0 to MyRoute.Legs[i].Steps.Count-1 do begin // BEGIN Anweisungen für diesen Schritt in Form einer Zeichenfolge s := MyRoute.Legs[i].Steps[k].Instructions; f2 := MyRoute.Legs[i].Steps[k].Distance.Value / 1000; f1 := f1 + f2; LB.Items.Add( Format( '[%d] %s %1.2fkm',[k+1,s,f2] )); // END Anweisungen für diesen Schritt in Form einer Zeichenfolge for l := 0 to MyRoute.Legs[i].Steps[k].Path.Count-1 do begin // BEGIN Speicher für TrackPoint reservieren und Daten erfassen TP := TTrackPoint.Create; TP.LatitudeDegrees := MyRoute.Legs[i].Steps[k].Path[l].Lat; TP.LongitudeDegrees := MyRoute.Legs[i].Steps[k].Path[l].Lng; TD.TrackPointList.Add( TP ); // END Speicher für TrackPoint reservieren und Daten erfassen end; end; LB.Items.Add( '' ); LB.Items.Add( Format( 'Streckenlänge : %1.2fkm',[f1] )); end; end; TD.Export... Danke und Gruß DelphiFan2008 |
AW: Google Maps über COM (Component Object Model)
Hmmm... Auf meinem PC brauchen die aber auch eine ganze Weile, um das Höhenprofil zu ermitteln. Wie genau muß das Profil sein?
Bei einer PathElevationRequest-Anfrage kannst Du ja bis zu 1024 Höhenwerte ermitteln lassen. Das macht auf 130 km etwa einen Wert pro 130 m. Reicht Dir das? Das sollte dann ohne gelbe Karte von Google und Wartezeit gehen... |
AW: Google Maps über COM (Component Object Model)
Hi Thom,
du hast recht, beim spielen mit dem Programm fällt auf, dass die Abfragen auch ne Weile dauern. Da GoogleMap eine schöne feine Auflösung liefert - zuerst war es mir zu grob, wie gesagt nun ca. 5100 Wertepaare auf 130km - müssen diese falls notwendig auf kleiner 1024 reduziert werden -> PathElevationRequest. Ich habe die letzten zwei Stunden mit dem Douglas-Peucker Algorithmus experimentiert um die Punktezahl zu reduzieren, Quelle1 und Quelle2. Werde mich noch ein wenig beschäftigen. Mein Outdoor-Navi, für welches ich dies alles mache wird sich auch über die geringere Datenmenge freuen. Frage noch in die andere Richtung - kann GoogleMap evntl. per API schon analog zu Douglas-Peucker optimieren? Gruß DelphiFan2008 |
AW: Google Maps über COM (Component Object Model)
Nicht, daß ich wüßte...
Ich bilde mir aber ein (nicht getestet), daß der Pfad keine Begrenzung der Punktanzahl vorschreibt. Theoretisch sollte es möglich sein, den kompletten Pfad zu übergeben (also nicht jeden Punkt einzeln kopieren) und dann einfach die gewünschte Anzahl (max. 1024) Höherwerte ermitteln zu lassen (eventuell in Abhängigkeit von der Pfadlänge). Google interpoliert dann einfach zwischen den Stützpunkten, die den Pfad darstellen. |
AW: Google Maps über COM (Component Object Model)
Hallo Thom
Bin gerade weiter dabei die Karte in meine Anwendung zu integrieren. Was mir nicht ganz klar ist wie kann ich das InfoWindow einem Marker zuordnen? Beim Marker erscheint immer das erste InfoWindow, was eigentlich auch klar ist da ich ja dies aufrufe
Delphi-Quellcode:
.
Infowindows[0].Open(Maps[0],Marker);
Stecke hier ein wenig fest.... mit folgendem code werden die Marker gesetzt:
Delphi-Quellcode:
procedure Tfrm_map.btn_Encode_batchClick(Sender: TObject);
var LatLng: TLatLng; Lat, Lng, InfoWindow_String: String; InfoWindow: TInfoWindow; Marker: TMarker; MarkerOptions: TMarkerOptions; begin with Script do begin Marker:=nil; //falls keine Daten in der Tabelle vorliegen DM_map.Map.First; while not DM_map.Map.Eof do begin if DM_map.Map.FieldByName('POS_LAT').Text = '' then DM_map.Map.Next else Lat:=DM_map.Map.FieldByName('POS_LAT').Text; Lng:=DM_map.Map.FieldByName('POS_LNG').Text; InfoWindow_String:=( // DM_map.Map.FieldByName('AREA').AsString + '<br>'+ DM_map.Map.FieldByName('NAME').AsString + '<br>'+ DM_map.Map.FieldByName('STR').AsString + '<br>'+ DM_map.Map.FieldByName('PLZ').AsString + '-'+ DM_map.Map.FieldByName('CITY').AsString + '<br>'+'TEL:' + DM_map.Map.FieldByName('TEL').AsString + '<br>'+'______________________ <br>'+'Camp ' + '- '+ DM_map.Map.FieldByName('C_NO').AsString); // LatLng:=New(Google.Maps.LatLng(StrToFloatDef(Lat,0,'.'), StrToFloatDef(Lng,0,'.'))); InfoWindow:=New(Google.Maps.InfoWindow); InfoWindow.Content:=InfoWindow_String; MarkerOptions:=TMarkerOptions.Create; with MarkerOptions do begin Map:=Maps[0]; Position:=LatLng; MarkerOptions.Title:= DM_map.Map.FieldByName('NAME').AsString; end; Marker:=New(Google.Maps.Marker(MarkerOptions)); Marker.OnClick:=MarkerClick; MarkerOptions.Free; //momentan noch etwas auf den Speicher achten //-> in der neuen Version stelle ich auf Interfaces um, //so daß die Freigabe automatisch erfolgt LatLng.Free; //gleicher Grund DM_map.Map.Next; end; if assigned(Marker) then Maps[0].SetCenter(Marker.GetPosition); //nur den letzten Marker zentrieren end; end;
Delphi-Quellcode:
procedure Tfrm_map.MarkerClick(Sender: TObject; Event: TEvent);
var Marker: TMarker; Text: String; begin with Script do begin Marker:=TMarker(Sender); // Text:=Marker.Properties['PlaceName']; // Infowindows[0].SetContent(Text); Infowindows[0].Open(Maps[0],Marker); end; end; |
AW: Google Maps über COM (Component Object Model)
Du müßtest das Info-Fenster an den Marker "binden". Ab Delphi 2009 geht das sehr komfortabel über anonyme Methoden. Ansonsten könntest Du die Data-Eigenschaft des Markers nutzen und dort das InfoWindow-Objekt ablegen (vergleichbar mit TListItem.Data):
Delphi-Quellcode:
Die Anzeige im MarkerClick-Ereignis ist dann einfach:
Marker.Data:=InfoWindow;
[...]
Delphi-Quellcode:
Wobei ich bei diesem Tip etwas Bauchschmerzen habe, da ich noch nicht 100%ig weiß, ob das Data-Property noch in der kostenlosen Personal-Edition der nächsten Version vorkommen wird. Eigentlich sollen dort nur Objekte, Eigenschaften und Methoden enthalten sein, die auch im Google Maps API dokumentiert sind. Delphi "Komfort"-Funktionen (wozu auch die Data- und die Tag- Eigenschaft gehören) sollen eigentlich der Professional- und Enterprise-Edition vorbehalten sein. Da muß ich noch etwas darüber schlafen...
[...]
var Marker: TMarker; InfoWindow: TInfoWindow; begin Marker:=Sender as TMarker; InfoWindow:=(Marker.Data) as TInfoWindow; if assigned(InfoWindow) then InfoWindow.Open(Marker.GetMap,Marker); end; Ansonsten könntest Du natürlich auch ein Array mit allen Marker-InfoWindow-Paaren aufbauen und darüber die Zuordnung realisieren:
Delphi-Quellcode:
Desweiteren würde ich empfehlen, als Cast nur den
type
TMarkerData = record Marker: TMarker; InfoWindow: TInfoWindow; end; var MarkerData: array[0..x] of TMarkerData;
Delphi-Quellcode:
-Operator zu verwenden, da zukünftig Interfaces genutzt werden und damit die Umstellung schneller geht (IMarker(Sender) geht dann nämlich schief):
as
Delphi-Quellcode:
//bisher:
procedure TForm1.MarkerClick(Sender: TObject; Event: TEvent); begin with Sender as TMarker do begin end; end; //zukünftig: procedure TForm1.MarkerClick(Sender: IObject; Event: IEvent); begin with Sender as IMarker do begin end; end; |
AW: Google Maps über COM (Component Object Model)
Hallo Thom
Bei der Variante mit Marker.Data:=InfoWindow erhalte ich eine EAccessViolation ohne nähere angaben. Habe schon versucht das Info-Window nur einzeilg zu halten brachte aber keinen Unterschied. Zitat:
Das ganze sieht aber schon wirklich super aus. Manfred |
AW: Google Maps über COM (Component Object Model)
Ich habe gerade mal folgende Variante getestet (Framework Version 2.0 mit Delphi XE):
Delphi-Quellcode:
Das demonstriert recht schön die Datenbindung mit Hilfe von anonymen Methoden. Schon alleine das wäre ein Grund, sich mal eine aktuelle Delphi-Version zuzulegen... :-D
procedure TForm1.InitMap(Sender: TObject);
var Map: TMap; MapOptions: TMapOptions; begin with Sender as TScript do begin MapOptions:=TMapOptions.Create; with MapOptions do begin Zoom:=10; Center:=New(Google.Maps.LatLng(47.651743,-122.349243)); MapTypeID:=Google.Maps.MapTypeID.Roadmap; end; Map:=New(Google.Maps.Map(MapOptions)); Map.OnClick:= procedure(Sender: TObject; Event: TMouseEvent) var Marker: TMarker; InfoWindow: TInfoWindow; begin Marker:=New(Google.Maps.Marker); Marker.Position:=Event.LatLng; Marker.Map:=Map; //<- hier wird das Map-Objekt aus der InitMap-Methode genommen InfoWindow:=New(Google.Maps.InfoWindow); InfoWindow.SetContent(Event.LatLng.ToUrlValue(4)); Marker.OnClick:= procedure(Sender: TObject; Event: HTMLObjects.TEvent) begin InfoWindow.Open(Map,Marker); //<- hier wird das Map-Objekt aus der InitMap-Methode //sowie das Marker- und das InfoWindow-Objekt //aus der Map.OnClick-Methode verwendet end; end; end; end; Aber auch die Variante mit
Delphi-Quellcode:
funktioniert bei mir ohne Probleme. Allerdings muß noch
Marker.Data:=InfoWindow;
Delphi-Quellcode:
gesetzt werden, damit es bei Löschen eines Markers beziehungsweise beim Aufräumen am Programmende zu keinen Problemen kommt.
Marker.OwnsData:=false;
Sollte es dann immer noch Probleme in Deinem Projekt geben, poste mal bitte den betreffenden Quelltext. Zusatz: Im Folgenden noch ein kleines Beispiel, wie durch anonyme Methoden gebundene Objekte in der neuen Frameworkversion verwendet und freigegen werden können. Dazu wurde das neue Event OnWrapperDestroy eingeführt:
Delphi-Quellcode:
Hier wird eine einfache Karte erstellt, bei einem Klick darauf ein neuer Marker gesetzt und mit einem Info-Fenster verbunden, das die Koordinaten des Markers anzeigt und beim Klick auf den Marker eingeblendet wird. Jedem Marker wird das selbe Popup-Menu zugeordnet, das auf den Befehl Delete den angeklickten Marker löscht und das zugeordnete Info-Fenster freigibt.
procedure TForm1.InitMap(Sender: IObject);
var Map: IMap; MapOptions: IMapOptions; begin with Sender as IScript do begin MapOptions:=New(Google.Maps.MapOptions); with MapOptions do begin Zoom:=10; Center:=New(Google.Maps.LatLng(47.651743,-122.349243)); MapTypeID:=Google.Maps.MapTypeID.Roadmap; end; Map:=New(Google.Maps.Map(MapOptions)); Map.OnClick:= procedure(Sender: IObject; Event: IMouseEvent) var Marker: IMarker; InfoWindow: IInfoWindow; begin InfoWindow:=New(Google.Maps.InfoWindow); InfoWindow.SetContent(Event.LatLng.ToUrlValue(4)); Marker:=New(Google.Maps.Marker); Marker.Map:=Map; Marker.Position:=Event.LatLng; Marker.PopupMenu:=MarkerPopupMenu; Marker.OnClick:= procedure(Sender: IObject; Event: IMouseEvent) begin InfoWindow.Open(Map,Marker); end; Marker.OnContextPopup:= procedure(Sender: IObject; MousePos: TPoint; var Handled: Boolean) begin FActiveMarker:=Marker; end; Marker.OnWrapperDestroy:= procedure(Sender: IObject) begin if assigned(InfoWindow) then InfoWindow.Free; end; end; end; end; procedure TForm1.DeleteMarkerClick(Sender: TObject); begin if assigned(FActiveMarker) then FActiveMarker.Free; end; Interessant ist dabei die Freigabe eines Objektes über ein Interface (InfoWindow.Free und FActiveMarker.Free). Die Variable FActiveMarker enthält nach der Ausführung der Anweisung FActiveMarker.Free den Wert
Delphi-Quellcode:
- Objekt- bzw. Interface-Referenzen werden also automatisch gelöscht. Ebenso wird die in den anonymen Methoden gebundene Variable InfoWindow automatisch auf
nil
Delphi-Quellcode:
gesetzt, falls durch die Refresh-Taste F5 oder beim Beenden des Programmes das InfoWindow-Objekt vor dem Marker freigegeben werden sollte (daher auch der notwendige Test mit assigned(), um Zugriffsverletzungen zu verhindern).
nil
|
AW: Google Maps über COM (Component Object Model)
Hallo Thom
Danke für Deine Antwort. Es geht jetzt. Habe bemerkt dass
Delphi-Quellcode:
erst nach MarkerOptions aufgerufen werden darf.
Marker.OwnsData:=false;
Marker.Data:=InfoWindow;
Delphi-Quellcode:
Besten Dank
procedure Tfrm_map.btn_Encode_batchClick(Sender: TObject);
var LatLng: TLatLng; Lat, Lng, InfoWindow_String: String; InfoWindow: TInfoWindow; Marker: TMarker; MarkerData: array of TMarkerData; MarkerOptions: TMarkerOptions; MyImage, MyShadow: TMarkerImage; sql_count: Integer; begin with Script do begin Marker:=nil; //falls keine Daten in der Tabelle vorliegen DM_map.Map.First; while not DM_map.Map.Eof do begin if DM_map.Map.FieldByName('POS_LAT').Text = '' then DM_map.Map.Next else Lat:=DM_map.Map.FieldByName('POS_LAT').Text; Lng:=DM_map.Map.FieldByName('POS_LNG').Text; LatLng:=New(Google.Maps.LatLng(StrToFloatDef(Lat,0,'.'), StrToFloatDef(Lng,0,'.'))); // InfoWindow_String:=( DM_map.Map.FieldByName('AREA').AsString + '<br>'+ DM_map.Map.FieldByName('NAME').AsString + '<br>'+ DM_map.Map.FieldByName('STR').AsString + '<br>'+ DM_map.Map.FieldByName('PLZ').AsString + '-'+ DM_map.Map.FieldByName('CITY').AsString + '<br>'+'TEL:' + DM_map.Map.FieldByName('TEL').AsString + '<br>'+'______________________ <br>'+'Camp ' + '- '+ DM_map.Map.FieldByName('C_NO').AsString); // InfoWindow:=New(Google.Maps.InfoWindow); InfoWindow.Content:=InfoWindow_String; // Imagesettings for this Marker MyImage:=New(Google.Maps.MarkerImage(MarkerURL+table_name+'.png', //This marker is 28 pixels wide by 35 pixels tall. New(Google.Maps.Size(28,35)), //The origin for this image is 0,0. New(Google.Maps.Point(0,0)), //The anchor for this image is the base of the flagpole at 6,20. New(Google.Maps.Point(6,20)))); // // Marker.OwnsData:=false; // Marker.Data:=InfoWindow; MarkerOptions:=TMarkerOptions.Create; with MarkerOptions do begin Map:=Maps[0]; Position:=LatLng; MarkerOptions.Title:= DM_map.Map.FieldByName('NAME').AsString; IconImage:=MyImage; end; Marker:=New(Google.Maps.Marker(MarkerOptions)); Marker.OnClick:=MarkerClick; MarkerOptions.Free; //momentan noch etwas auf den Speicher achten //-> in der neuen Version stelle ich auf Interfaces um, //so daß die Freigabe automatisch erfolgt LatLng.Free; //gleicher Grund Marker.OwnsData:=false; Marker.Data:=InfoWindow; DM_map.Map.Next; end; if assigned(Marker) then Maps[0].SetCenter(Marker.GetPosition); //nur den letzten Marker zentrieren end; end; |
AW: Google Maps über COM (Component Object Model)
Schön, daß es funktioniert!
Aber was meinst Du mit "nach MarkerOptions"? Ich würde sagen, daß die Zuweisung nach der Erstellung des Marker-Objektes mit
Delphi-Quellcode:
erfolgen kann, da an dieser Stelle der Marker erstellt wird und damit für Zugriffe zur Verfügung steht. Um diesen Sachverhalt etwas deutlicher zu machen, hatte ich in der Version 2 die New-Funktionen eingeführt (analog dem new-Operator in JavaScript). Sonst ist nicht sofort ersichtlich, daß mit
Marker:=New(Google.Maps.Marker(MarkerOptions));
Delphi-Quellcode:
ein neues Objekt angelegt wird. Delphi-Programmierer sind ja eher den Syntax
Google.Maps.Marker(...);
Delphi-Quellcode:
gewohnt. Die Erstellung eines Objektes über eine Funktion entspricht eher der Logik der "Fabrikmethoden".
TMyObject.Create(...);
War die Verwendung der New-Funktionen bisher optional (sie reichen einfach den Parameter an den Result-Wert weiter), wird deren Anwendung zukünftig obligatorisch, da die "Fabrikmethoden" nicht mehr das reine Objekt, sondern nur noch einen Zwischenwert liefern, der durch die New-Methode "entpackt" wird. Der Sinn des Ganzen: Den Quelltext lesbarer zu gestalten. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 04:21 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz