Delphi-PRAXiS
Seite 23 von 55   « Erste     13212223 242533     Letzte »    

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Software-Projekte der Mitglieder (https://www.delphipraxis.net/26-software-projekte-der-mitglieder/)
-   -   Google Maps über COM (Component Object Model) (https://www.delphipraxis.net/157004-google-maps-ueber-com-component-object-model.html)

DelphiFan2008 15. Apr 2012 17:18

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:
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...
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:

Danke und Gruß DelphiFan2008

Thom 15. Apr 2012 19:10

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...

DelphiFan2008 15. Apr 2012 19:40

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

Thom 15. Apr 2012 21:32

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.

manfred_h 16. Apr 2012 16:11

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;

Thom 16. Apr 2012 17:29

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:
  Marker.Data:=InfoWindow;
  [...]
Die Anzeige im MarkerClick-Ereignis ist dann einfach:
Delphi-Quellcode:
 [...]
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;
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...

Ansonsten könntest Du natürlich auch ein Array mit allen Marker-InfoWindow-Paaren aufbauen und darüber die Zuordnung realisieren:
Delphi-Quellcode:
type
  TMarkerData = record
    Marker: TMarker;
    InfoWindow: TInfoWindow;
  end;

var
  MarkerData: array[0..x] of TMarkerData;
Desweiteren würde ich empfehlen, als Cast nur den
Delphi-Quellcode:
as
-Operator zu verwenden, da zukünftig Interfaces genutzt werden und damit die Umstellung schneller geht (IMarker(Sender) geht dann nämlich schief):
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;

manfred_h 17. Apr 2012 15:04

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:

Zitat von Thom (Beitrag 1162144)
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:
  Marker.Data:=InfoWindow;
  [...]
Die Anzeige im MarkerClick-Ereignis ist dann einfach:
Delphi-Quellcode:
 [...]
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;
[/DELPHI]

Mit dem Array bin ich noch nicht ganz klar gekommen, werde mich als nächstes daran versuchen.
Das ganze sieht aber schon wirklich super aus.
Manfred

Thom 17. Apr 2012 18:52

AW: Google Maps über COM (Component Object Model)
 
Ich habe gerade mal folgende Variante getestet (Framework Version 2.0 mit Delphi XE):
Delphi-Quellcode:
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;
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

Aber auch die Variante mit
Delphi-Quellcode:
Marker.Data:=InfoWindow;
funktioniert bei mir ohne Probleme. Allerdings muß noch
Delphi-Quellcode:
Marker.OwnsData:=false;
gesetzt werden, damit es bei Löschen eines Markers beziehungsweise beim Aufräumen am Programmende zu keinen Problemen kommt.

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:
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;
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.

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:
nil
- Objekt- bzw. Interface-Referenzen werden also automatisch gelöscht. Ebenso wird die in den anonymen Methoden gebundene Variable InfoWindow automatisch auf
Delphi-Quellcode:
nil
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).

manfred_h 18. Apr 2012 10:38

AW: Google Maps über COM (Component Object Model)
 
Hallo Thom

Danke für Deine Antwort.
Es geht jetzt. Habe bemerkt dass
Delphi-Quellcode:
      Marker.OwnsData:=false;
      Marker.Data:=InfoWindow;
erst nach MarkerOptions aufgerufen werden darf.

Delphi-Quellcode:
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;
Besten Dank

Thom 18. Apr 2012 11:19

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:
Marker:=New(Google.Maps.Marker(MarkerOptions));
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
Delphi-Quellcode:
Google.Maps.Marker(...);
ein neues Objekt angelegt wird. Delphi-Programmierer sind ja eher den Syntax
Delphi-Quellcode:
TMyObject.Create(...);
gewohnt. Die Erstellung eines Objektes über eine Funktion entspricht eher der Logik der "Fabrikmethoden".
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.
Seite 23 von 55   « Erste     13212223 242533     Letzte »    

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