All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.lynden.gmapsfx.GoogleMapView Maven / Gradle / Ivy

/*
 * Copyright 2014 Lynden, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.lynden.gmapsfx;

import com.lynden.gmapsfx.javascript.JavaFxWebEngine;
import com.lynden.gmapsfx.javascript.JavascriptRuntime;
import com.lynden.gmapsfx.javascript.event.MapStateEventType;
import com.lynden.gmapsfx.javascript.object.DirectionsPane;
import com.lynden.gmapsfx.javascript.object.GoogleMap;
import com.lynden.gmapsfx.javascript.object.LatLong;
import com.lynden.gmapsfx.javascript.object.MapOptions;


import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CyclicBarrier;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.concurrent.Worker;
import javafx.event.Event;
import javafx.event.EventDispatchChain;
import javafx.event.EventDispatcher;
import javafx.event.EventHandler;
import javafx.geometry.Point2D;
import javafx.scene.input.MouseButton;
import javafx.scene.input.MouseEvent;
import javafx.scene.layout.AnchorPane;
import javafx.scene.web.WebErrorEvent;
import javafx.scene.web.WebEvent;
import javafx.scene.web.WebView;
import netscape.javascript.JSObject;

/**
 *
 * @author Rob Terpilowski
 */
public class GoogleMapView extends AnchorPane {

    protected static final String GOOGLE_MAPS_API_LINK = "https://maps.googleapis.com/maps/api/js?v=3.exp";
    protected static final String GOOGLE_MAPS_API_VERSION = "3.exp";
    
    private boolean usingCustomHtml;
    
    protected final String language;
    protected final String region;
    protected final String key;
    protected final WebView webview;
    protected final JavaFxWebEngine webengine;
    protected boolean initialized = false;
    protected final CyclicBarrier barrier = new CyclicBarrier(2);
    protected final List mapInitializedListeners = new ArrayList<>();
    protected final List mapReadyListeners = new ArrayList<>();
    protected GoogleMap map;
    protected DirectionsPane direc;
    protected boolean disableDoubleClick = false;

    public GoogleMapView() {
        this(false);
    }

    public GoogleMapView(boolean debug) {
        this(null, debug);
    }

    /**
     * Allows for the creation of the map using external resources from another
     * jar for the html page and markers. The map html page must be sourced from
     * the jar containing any marker images for those to function.
     * 

* The html page is, at it's simplest: * {@code * * * * * * My Map * * * * *

* * * } *

* If you store this file in your project jar, under * my.gmapsfx.project.resources as mymap.html then you should call using * "/my/gmapsfx/project/resources/mymap.html" for the mapResourcePath. *

* Your marker images should be stored in the same folder as, or below the * map file. You then reference them using relative notation. If you put * them in a subpackage "markers" you would create your MarkerOptions object * as follows: * {@code * myMarkerOptions.position(myLatLong) * .title("My Marker") * .icon("markers/mymarker.png") * .visible(true); * } * * @param mapResourcePath */ public GoogleMapView(String mapResourcePath) { this(mapResourcePath, false); } /** * Creates a new map view and specifies if the FireBug pane should be * displayed in the WebView * * @param mapResourcePath * @param debug true if the FireBug pane should be displayed in the WebView. */ public GoogleMapView(String mapResourcePath, boolean debug) { this(mapResourcePath, null, null, debug); } /** * Creates a new map view and specifies the display language and API key. * * @param language map display language, null for default * @param key Google Maps API key or null */ public GoogleMapView(String language, String key) { this(null, language, key, false); } /** * Creates a new map view and specifies the display language and API key. * * @param mapResourcePath * @param language map display language, null for default * @param key Google Maps API key or null * @param debug true if the FireBug pane should be displayed in the WebView. */ public GoogleMapView(String mapResourcePath, String language, String key, boolean debug) { this(mapResourcePath, language, null, key, debug); } /** * Creates a new map view and specifies the display language and API key. *

* If you are specifying your own HTML page for mapResourcePath in a jar of * your own then you should include a script element to pull in the * Google Maps API with any API keys, language and region parameters. * * @param mapResourcePath * @param language map display language, null for default * @param region * @param key Google Maps API key or null * @param debug true if the FireBug pane should be displayed in the WebView. */ public GoogleMapView(String mapResourcePath, String language, String region, String key, boolean debug) { this.language = language; this.region = region; this.key = key; String htmlFile; if (mapResourcePath == null) { if (debug) { htmlFile = "/html/maps-debug.html"; } else { htmlFile = "/html/maps.html"; } } else { htmlFile = mapResourcePath; usingCustomHtml = true; } System.out.println("htmlFile: " + htmlFile); webview = new WebView(); EventDispatcher originalDispatcher = webview.getEventDispatcher(); webview.setEventDispatcher(new MyEventDispatcher(originalDispatcher)); webview.focusedProperty().addListener( (ObservableValue observable, Boolean oldValue, Boolean newValue) -> { //System.out.println("Webview focus changed from: " + oldValue + " to " + newValue ); }); webengine = new JavaFxWebEngine(webview.getEngine()); JavascriptRuntime.setDefaultWebEngine(webengine); setTopAnchor(webview, 0.0); setLeftAnchor(webview, 0.0); setBottomAnchor(webview, 0.0); setRightAnchor(webview, 0.0); getChildren().add(webview); webview.widthProperty().addListener(e -> mapResized()); webview.heightProperty().addListener(e -> mapResized()); webview.widthProperty().addListener(e -> mapResized()); webview.heightProperty().addListener(e -> mapResized()); webengine.setOnAlert(new EventHandler>() { @Override public void handle(WebEvent e) { System.out.println("Alert: " + e.getData()); } }); webengine.setOnError(new EventHandler() { @Override public void handle(WebErrorEvent e) { System.out.println("Error: " + e.getMessage()); } }); webengine.getLoadWorker().stateProperty().addListener( new ChangeListener() { public void changed(ObservableValue ov, Worker.State oldState, Worker.State newState) { if (newState == Worker.State.SUCCEEDED) { initialiseScript(); //setInitialized(true); //fireMapInitializedListeners(); } } }); webengine.load(getClass().getResource(htmlFile).toExternalForm()); } private void initialiseScript() { if (! usingCustomHtml) { JSObject window = (JSObject) webengine.executeScript("window"); window.setMember("libLoadBridge", new MapLibraryLoadBridge()); String script = "loadMapLibrary('" + GOOGLE_MAPS_API_VERSION + "','" + key + "','" + language + "','" + region + "');"; //System.out.println("Loading script with call: " + script); webengine.executeScript(script); } else { setInitialized(true); fireMapInitializedListeners(); } } private void mapResized() { if (initialized) { System.out.println("GoogleMapView.mapResized: triggering resize event"); webengine.executeScript("google.maps.event.trigger(" + map.getVariableName() + ", 'resize')"); } } public void setZoom(int zoom) { checkInitialized(); map.setZoom(zoom); } public void setCenter(double latitude, double longitude) { checkInitialized(); LatLong latLong = new LatLong(latitude, longitude); map.setCenter(latLong); } public GoogleMap getMap() { checkInitialized(); return map; } public GoogleMap createMap(MapOptions mapOptions) { return createMap(mapOptions,false); } public GoogleMap createMap() { return createMap(null, false); } public GoogleMap createMap(boolean withDirectionsPanel) { return createMap(null, withDirectionsPanel); } public GoogleMap createMap(MapOptions mapOptions, boolean withDirectionsPanel ) { checkInitialized(); if( mapOptions != null ) { map = new GoogleMap(mapOptions); } else { map = new GoogleMap(); } direc = new DirectionsPane(); if( withDirectionsPanel ) { map.showDirectionsPane(); } map.addStateEventHandler(MapStateEventType.projection_changed, () -> { if (map.getProjection() != null) { mapResized(); fireMapReadyListeners(); } }); return map; } public DirectionsPane getDirec() { return direc; } public void addMapInializedListener(MapComponentInitializedListener listener) { synchronized (mapInitializedListeners) { mapInitializedListeners.add(listener); } } public void removeMapInitializedListener(MapComponentInitializedListener listener) { synchronized (mapInitializedListeners) { mapInitializedListeners.remove(listener); } } public void addMapReadyListener(MapReadyListener listener) { synchronized (mapReadyListeners) { mapReadyListeners.add(listener); } } public void removeReadyListener(MapReadyListener listener) { synchronized (mapReadyListeners) { mapReadyListeners.remove(listener); } } public Point2D fromLatLngToPoint(LatLong loc) { checkInitialized(); return map.fromLatLngToPoint(loc); } public void panBy(double x, double y) { checkInitialized(); map.panBy(x, y); } public boolean isDisableDoubleClick() { return disableDoubleClick; } public void setDisableDoubleClick(boolean disableDoubleClick) { this.disableDoubleClick = disableDoubleClick; } protected void init() { } protected void setInitialized(boolean initialized) { this.initialized = initialized; } protected void fireMapInitializedListeners() { synchronized (mapInitializedListeners) { for (MapComponentInitializedListener listener : mapInitializedListeners) { listener.mapInitialized(); } } } protected void fireMapReadyListeners() { synchronized (mapReadyListeners) { for (MapReadyListener listener : mapReadyListeners) { listener.mapReady(); } } } protected JSObject executeJavascript(String function) { Object returnObject = webengine.executeScript(function); return (JSObject) returnObject; } protected String getJavascriptMethod(String methodName, Object... args) { StringBuilder sb = new StringBuilder(); sb.append(methodName).append("("); for (Object arg : args) { sb.append(arg).append(","); } sb.replace(sb.length() - 1, sb.length(), ")"); return sb.toString(); } protected void checkInitialized() { if (!initialized) { throw new MapNotInitializedException(); } } public WebView getWebview() { return webview; } public class JSListener { public void log(String text) { System.out.println(text); } } public class MapLibraryLoadBridge { public MapLibraryLoadBridge() { } public void mapLibraryLoaded() { setInitialized(true); fireMapInitializedListeners(); } } public class MyEventDispatcher implements EventDispatcher { private EventDispatcher originalDispatcher; public MyEventDispatcher(EventDispatcher originalDispatcher) { this.originalDispatcher = originalDispatcher; } @Override public Event dispatchEvent(Event event, EventDispatchChain tail) { if (event instanceof MouseEvent) { MouseEvent mouseEvent = (MouseEvent) event; if( mouseEvent.getClickCount() == 2 ) { //System.out.println("Mouse event: " + event); if( disableDoubleClick ) { mouseEvent.consume(); } } } return originalDispatcher.dispatchEvent(event, tail); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy