Delphi-PRAXiS
Seite 24 von 55   « Erste     14222324 252634     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)

manfred_h 18. Apr 2012 13:19

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

Zitat von Thom (Beitrag 1162500)
Schön, daß es funktioniert!
Aber was meinst Du mit "nach MarkerOptions"?

2 Post vorher im Quelltext:
Delphi-Quellcode:
// Marker.OwnsData:=false;
// Marker.Data:=InfoWindow;
Wenn dies Codezeilen aktiviert werden und weiter unten ( 14 Zeilen ) dies deaktiviert werden, erhalte ich die EAccessViolation.

Der Code sieht nun so aus:
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;
      if DM_map.Map.FieldByName('ACCEPTANCE').AsString= '1' then
      begin
      // 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))));
      end;
      if DM_map.Map.FieldByName('ACCEPTANCE').AsString= '2' then
      begin
      // Imagesettings for this Marker
      MyImage:=New(Google.Maps.MarkerImage(MarkerURL+table_name+'_Conditional'+'.png',
        New(Google.Maps.Size(28,35)),
        New(Google.Maps.Point(0,0)),
        New(Google.Maps.Point(6,20))));
      end;
      if DM_map.Map.FieldByName('ACCEPTANCE').AsString= '3' then
      begin
      // Imagesettings for this Marker
      MyImage:=New(Google.Maps.MarkerImage(MarkerURL+table_name+'_no'+'.png',
        New(Google.Maps.Size(28,35)),
        New(Google.Maps.Point(0,0)),
        New(Google.Maps.Point(6,20))));
      end;
      if DM_map.Map.FieldByName('ACCEPTANCE').AsString= '4' then
      begin
      // Imagesettings for this Marker
      MyImage:=New(Google.Maps.MarkerImage(MarkerURL+table_name+'_maybe'+'.png',
        New(Google.Maps.Size(28,35)),
        New(Google.Maps.Point(0,0)),
        New(Google.Maps.Point(6,20))));
      end;
      if DM_map.Map.FieldByName('ACCEPTANCE').AsString= '' then
      begin
      MyImage:=New(Google.Maps.MarkerImage(MarkerURL+table_name+'.png',
        New(Google.Maps.Size(28,35)),
        New(Google.Maps.Point(0,0)),
        New(Google.Maps.Point(6,20))));
      end;
      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;

Thom 18. Apr 2012 13:32

AW: Google Maps über COM (Component Object Model)
 
Ahhh - Danke für den Hinweis! Ich hatte die auskommentierten Zeilen übersehen. Ja - genau: An dieser Stelle existiert der Marker noch nicht.
Da muß ich gleich noch einmal bezüglich der New-Funktion nachfragen (siehe mein letzes Posting): Macht die das Lesen des Quelltextes etwas einfacher oder ist sie eher überflüssig (was sie ja aus programmtechnischer Sicht tatsächlich ist)?

manfred_h 18. Apr 2012 14:08

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

Zitat von Thom (Beitrag 1162534)
Da muß ich gleich noch einmal bezüglich der New-Funktion nachfragen (siehe mein letzes Posting): Macht die das Lesen des Quelltextes etwas einfacher oder ist sie eher überflüssig (was sie ja aus programmtechnischer Sicht tatsächlich ist)?

Hallo Thom

Das ganze ist meiner Meinung nach wirklich übersichtlich. Muss aber sagen das ich kein Prof. Programmierer bin. Kann daher auch nur eine Aussage aus meiner Sicht machen. Naträglich gesehen ist es auch nachzuvollziehen dass an dieser Stelle der Marker noch nicht existiert. Da ja
Delphi-Quellcode:
 Marker:=New(Google.Maps.Marker(MarkerOptions));
erst nachher aufgerufen wird.
Danke für Deine Geduld. :thumb:

Shalom
Manfred

Thom 18. Apr 2012 19:21

AW: Google Maps über COM (Component Object Model)
 
Gern geschehen. :thumb:
Und vielen Dank für Deine Rückmeldung!

DelphiFan2008 18. Apr 2012 21:45

AW: Google Maps über COM (Component Object Model)
 
Liste der Anhänge anzeigen (Anzahl: 1)
Hi Thom,

bin eifrig dabei mehr über GoogleMap zu lernen und natürlich mit dem Framework umzusetzen.

Wie schon in den vorherigen Anfragen habe ich auf Basis des "Directions Draggable"-Demo Erweiterungen vorgenommen. So langsam wird mir die Ojektstruktur klarer. Unklar ist mir wie GoogleMaps die zusätzlichen Wegmarken/Waypoints...siehe Rot eingekreist verwaltet.

Mein Ziel ist es den aktuelen Zustand in einer Datei abzulegen um diesen wieder 1-zu-1 einzulesen/anzuzeigen. Ansatz beim Speichern "MyRoute.Legs[i]..." als Schritt speichern, wahrscheinlich "MyRoute.Legs[i].StartLocation" und "MyRoute.Legs[i].StartAddress" und beim Einlesen "WayPoint:=TDirectionsWayPoint.Create;" etc. einfügen.

Wie kann ich auf die rot eingekreisten Marken zugreifen - Daten ermitteln (Position LAT/LNG) und sie beim Einlesen wieder erzeugen?

... nächster Schritt wird dann "rot eingekreisten Marken" und "Marker" selektieren und mit "Abfrage/Info-Dialog" bei Bestätigung löschen, dazu aber später ... :lol:, hinzufügen geht einfacher...

Gruß DelphiFan2008

Thom 19. Apr 2012 23:34

AW: Google Maps über COM (Component Object Model)
 
Du bist ja fleißig! :thumb:

Die zusätzlichen Punkte sind im Array ViaWayPoints des TDirectionsLeg-Objektes enthalten. Siehe DirectionsLeg.
Da es sich um ein normales Array handelt, kann man dort Werte hinzufügen oder löschen. Allerdings befürchte ich, daß der DirectionsRenderer auf derartige Veränderungen nicht reagiert - also die Strecke neu berechnet und anzeigt.
Soll die Strecke neu eingelesen werden (zum Beispiel aus einer Datei oder Datenbank) müssen die zusätzlichen Wegpunkte wahrscheinlich dem WayPoints-Array des TDirectionsRequest-Objektes hinzugefügt werden, um bei der Berechnung berücksichtigt zu werden.

DelphiFan2008 21. Apr 2012 19:51

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

danke für die schnelle Antwort (wie immer :)). Nun habe ich alle Informationen für die Streckenplanung d.h. ich bekomme alle Daten aus GoogleMap um die geplante Route abzuspeichern und somit zu rekonstruieren und die feine Auflösung um daraus (evntl. reduziert mit Douglas-Peucker Alg.) ein Track zu erzeugen. Gefällt mir sehr gut. Nun folgt die GUI mit der entsprechenden Bedienung Marker setzen bzw. löschen etc. inkl. für mich wichtig Höhenprofil möglichst parallel zu ermitteln - evntl. Threat - macht da GoogleMap mit, wenn die Hauptapplikation auf GoogleMap "parallel" zum Threat zugreift?

Gruss DelphiFan2008

Thom 21. Apr 2012 20:53

AW: Google Maps über COM (Component Object Model)
 
Das Framework ist nicht threadsicher. Darauf habe ich bisher keinen Wert gelegt, da die JavaScript-Engine des Internet Explorer, auf die das Framework zugreift, auch nicht threadsicher ist. Solange das Framework diese Script-Engine verwendet, wird sich daran auch nichts ändern. Geplant ist aber, irgendwann auf eine andere Engine umzusteigen. Dann könnte sich das auch ändern. An JavaScript als Basis führt aber zur Zeit kein Weg vorbei, da das JavaScript-API momentan den einzigen legalen Zugriff auf Google Maps darstellt (das Flash-API ist als deprecated gekennzeichnet).

Da aber alle Abfragen des API's asynchron erfolgen (das heißt, daß zum Beispiel während der Abfrage des Höhenprofils die Anwendung nicht blockiert wird) und nach Ausführung eine Callback-Methode aufgerufen wird, sollte der Zeitverlust im Vergleich zur Verwendung von Threads nur minimal sein. Das heißt: Im Prinzip ist das nicht viel anders als eine Datenabfrage per Thread mit anschließendem Synchronize.

Bezüglich der Wegpunkte bei einer DirectionsRequest-Anfrage:
Man kann die zusätzlichen Wegpunkte aus dem ViaWayPoints-Array mit dem Flag
Delphi-Quellcode:
StopOver:=false
versehen (ist default true). Damit wird die Route an dieser Stelle nicht geteilt. Siehe DirectionsWaypoint.

DelphiFan2008 22. Apr 2012 12:02

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

Thread werde ich wohl mal zurückstellen, auch nicht so wichtig. Jetzt muß ich mich jedoch wegen dem Thema "wie erfasse ich ein Klick auf eine Route, Leg oder ViaWayPoints", bzw. wie verknüpfe ich ein OnXxx Ereignis um z.B. ein "Leg" oder "ViaWayPoints" zu löschen?

In meiner Applikation, in welcher ich mit Marker arbeite, welche zur Laufzeit durch mein Programm entstehen ist die Sache klar.

Routine in der MainForm :
Delphi-Quellcode:
procedure OnMarkerDrag(Sender: TObject; MouseEvent: TMouseEvent);
dann
Delphi-Quellcode:
MyMarker.MarkerPos.OnDrag   := OnMarkerDrag;
und
Delphi-Quellcode:
procedure MainForm.OnMarkerDrag(Sender: TObject; MouseEvent: TMouseEvent);
begin
  ...
end;
Bei der Verwendung von directions sieht die Sache (für mich) anders aus, hier verwaltet GoogleMap die Erstellung/Reaktion direkt während der Laufzeit. Wie kann ich hier auf die Events reagieren?

Gruß DelphiFan2008

Thom 24. Apr 2012 08:02

AW: Google Maps über COM (Component Object Model)
 
Frustrierende Antwort: Gar nicht... :cry:

Der DirectionsRenderer besitzt leider nur ein einziges dokumentiertes Event, das ausgelöst wird, nachdem die Route durch den Benutzer verändert wurde. Alles andere (wie zum Beispiel ein Klick auf die Route oder einen Marker) wird intern nach dem Black-Box-Prinzip verarbeitet. Deshalb kann ich Dir nur empfehlen, ein eigenes TDirektionsRenderer-Objekt zu programmieren. Alles was dazu notwendig ist (TMarker, TPolyline, TDirectionsResult), ist vorhanden.

Entschuldige bitte die verspätete Antwort: Ich hatte lange herumgebastelt, um eine Lösung auf Basis des TDirectionsRenderer-Objektes aus dem API zu finden - sich nur auf undokumentierte Dinge zu verlassen, bringt auf Dauer aber leider auch nichts.

Das bringt mich aber auf die Idee, später - wenn die Homepage fertig ist - einen Bereich für User-Erweiterungen einzurichten.


Alle Zeitangaben in WEZ +1. Es ist jetzt 02:04 Uhr.
Seite 24 von 55   « Erste     14222324 252634     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