Tipi di mappe

Seleziona la piattaforma: Android iOS JavaScript

Questo documento illustra i tipi di mappe che puoi visualizzare utilizzando l'API Maps JavaScript. L'API usa un oggetto MapType che contiene le informazioni su queste mappe. Un MapType è un'interfaccia che definisce la visualizzazione e l'utilizzo dei riquadri della mappa e la traduzione dei sistemi di coordinate dalle coordinate dello schermo alle coordinate mondiali (sulla mappa). Ogni MapType deve contenere alcuni metodi per gestire il recupero e il rilascio delle riquadri e le proprietà che ne definiscono il comportamento visivo.

Il funzionamento interno dei tipi di mappa all'interno dell'API Maps JavaScript è un argomento avanzato. La maggior parte degli sviluppatori può utilizzare i tipi di mappe di base indicati di seguito. Tuttavia, puoi anche modificare la presentazione dei tipi di mappa esistenti utilizzando le mappe stilizzate o definire i tuoi riquadri mappa utilizzando i tipi di mappa personalizzati. Quando fornisci tipi di mappe personalizzate, devi sapere come modificare il registro dei tipi di mappa della mappa.

Tipi di mappe di base

Nell'API Maps JavaScript sono disponibili quattro tipi di mappe. Oltre ai familiari riquadri delle mappe stradali "dipinte", l'API Maps JavaScript supporta anche altri tipi di mappe.

Nell'API Maps JavaScript sono disponibili i seguenti tipi di mappe:

  • roadmap mostra la visualizzazione della mappa stradale predefinita. Questo è il tipo di mappa predefinito.
  • satellite mostra le immagini satellitari di Google Earth.
  • hybrid mostra una combinazione di visualizzazioni normali e satellitari.
  • terrain mostra una mappa fisica basata sulle informazioni sul terreno.

Modifica il tipo di mappa in uso da Map impostando la relativa proprietà mapTypeId, all'interno del costruttore impostando il relativo oggetto Map options o chiamando il metodo setMapTypeId() della mappa. Il valore predefinito della proprietà mapTypeID è roadmap.

Impostazione di mapTypeId al momento della creazione:

var myLatlng = new google.maps.LatLng(-34.397, 150.644);
var mapOptions = {
  zoom: 8,
  center: myLatlng,
  mapTypeId: 'satellite'
};
var map = new google.maps.Map(document.getElementById('map'),
    mapOptions);

Modifica dinamica di mapTypeId:

map.setMapTypeId('terrain');

Tieni presente che non imposti direttamente il tipo di mappa, ma imposti il relativo mapTypeId in modo che faccia riferimento a un MapType utilizzando un identificatore. L'API Maps JavaScript utilizza un registry dei tipi di mappe, spiegato di seguito, per gestire questi riferimenti.

Immagini a 45°

L'API Maps JavaScript supporta immagini speciali a 45° per determinate località. Queste immagini ad alta risoluzione forniscono viste prospettiche in ciascuna delle quattro direzioni cardinali (nord, sud, est, ovest). Queste immagini sono disponibili a livelli di zoom superiori per i tipi di mappe supportati.

L'immagine seguente mostra una vista prospettica di New York a 45°:

I tipi di mappe satellite e hybrid supportano le immagini a 45° con livelli di zoom elevati (12 e superiori), se disponibili. Se l'utente ingrandisce una località per la quale esistono queste immagini, questi tipi di mappe modificano automaticamente le visualizzazioni nel seguente modo:

  • Le immagini satellitari o ibride vengono sostituite da immagini con una prospettiva di 45° centrata sulla posizione attuale. Per impostazione predefinita, queste viste sono rivolte verso nord. Se l'utente riduce lo zoom, vengono visualizzate di nuovo le immagini satellitari o ibride predefinite. Il comportamento varia in base al livello di zoom e al valore di tilt:
    • Tra i livelli di zoom 12 e 18, la mappa di base dall'alto verso il basso (0°) viene visualizzata per impostazione predefinita, a meno che tilt non sia impostato su 45.
    • A livelli di zoom pari o superiori a 18 viene visualizzata la mappa di base a 45°, a meno che tilt non sia impostato su 0.
  • Il controllo di rotazione diventa visibile. Il controllo di rotazione offre opzioni che consentono all'utente di attivare/disattivare l'inclinazione e di ruotare la vista in incrementi di 90° in entrambe le direzioni. Per nascondere il controllo di rotazione, imposta rotateControl su false.

Se diminuisci lo zoom di un tipo di mappa che mostra immagini a 45°, tutte queste modifiche vengono annullate e vengono ripristinati i tipi di mappa originali.

Attivare e disattivare le immagini a 45°

Puoi disattivare le immagini a 45° chiamando setTilt(0) sull'oggetto Map. Per attivare le immagini a 45° per i tipi di mappe supportati, chiama setTilt(45). Il metodo getTilt() di Map rifletterà sempre l'tilt corrente visualizzato sulla mappa. Se imposti un tilt su una mappa e poi lo rimuovi (ad esempio diminuendo lo zoom della mappa), il metodo getTilt() della mappa restituirà 0.tilt

Importante: le immagini a 45° sono supportate solo nelle mappe raster; queste immagini non possono essere utilizzate con le mappe vettoriali.

L'esempio seguente mostra una vista a 45° di New York:

TypeScript

function initMap(): void {
  const map = new google.maps.Map(
    document.getElementById("map") as HTMLElement,
    {
      center: { lat: 40.76, lng: -73.983 },
      zoom: 15,
      mapTypeId: "satellite",
    }
  );

  map.setTilt(45);
}

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

function initMap() {
  const map = new google.maps.Map(document.getElementById("map"), {
    center: { lat: 40.76, lng: -73.983 },
    zoom: 15,
    mapTypeId: "satellite",
  });

  map.setTilt(45);
}

window.initMap = initMap;
Visualizza esempio

Prova Sample

Visualizza l'esempio.

Ruotare le immagini di 45°

Le immagini a 45° sono in realtà costituite da una raccolta di immagini per ogni direzione cardinale (nord, sud, est, ovest). Una volta che la mappa visualizza le immagini a 45°, puoi orientarle verso una delle direzioni cardinali chiamando setHeading() sull'oggetto Map, passando un valore numerico espresso in gradi a partire da nord.

L'esempio riportato di seguito mostra una mappa aerea e la ruota automaticamente ogni 3 secondi quando viene fatto clic sul pulsante:

TypeScript

let map: google.maps.Map;

function initMap(): void {
  map = new google.maps.Map(document.getElementById("map") as HTMLElement, {
    center: { lat: 40.76, lng: -73.983 },
    zoom: 15,
    mapTypeId: "satellite",
    heading: 90,
    tilt: 45,
  });

  // add listener to button
  document.getElementById("rotate")!.addEventListener("click", autoRotate);
}

function rotate90(): void {
  const heading = map.getHeading() || 0;

  map.setHeading(heading + 90);
}

function autoRotate(): void {
  // Determine if we're showing aerial imagery.
  if (map.getTilt() !== 0) {
    window.setInterval(rotate90, 3000);
  }
}

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

let map;

function initMap() {
  map = new google.maps.Map(document.getElementById("map"), {
    center: { lat: 40.76, lng: -73.983 },
    zoom: 15,
    mapTypeId: "satellite",
    heading: 90,
    tilt: 45,
  });
  // add listener to button
  document.getElementById("rotate").addEventListener("click", autoRotate);
}

function rotate90() {
  const heading = map.getHeading() || 0;

  map.setHeading(heading + 90);
}

function autoRotate() {
  // Determine if we're showing aerial imagery.
  if (map.getTilt() !== 0) {
    window.setInterval(rotate90, 3000);
  }
}

window.initMap = initMap;
Visualizza esempio

Prova Sample

Visualizza l'esempio.

Modificare il registry dei tipi di mappa

Il mapTypeId di una mappa è un identificatore di stringa utilizzato per associare un MapType a un valore univoco. Ogni oggetto Map gestisce un MapTypeRegistry contenente la raccolta di MapType disponibili per la mappa. Questo registry viene utilizzato per selezionare i tipi di mappe disponibili, ad esempio nel controllo MapType della mappa.

Non leggi direttamente dal registry dei tipi di mappa. Al contrario, modifichi il registry aggiungendo tipi di mappe personalizzate e associandole a un identificatore di stringa di tua scelta. Non puoi modificare o alterare i tipi di mappe di base (anche se puoi rimuoverli dalla mappa modificando l'aspetto del mapTypeControlOptions associato alla mappa).

Il seguente codice imposta la mappa in modo da mostrare solo due tipi di mappa in mapTypeControlOptions della mappa e modifica il registry per aggiungere l'associazione con questo identificatore all'implementazione effettiva dell'interfaccia MapType.

// Modify the control to only display two maptypes, the
// default ROADMAP and the custom 'mymap'.
// Note that because this is an association, we
// don't need to modify the MapTypeRegistry beforehand.

var MY_MAPTYPE_ID = 'mymaps';

var mapOptions = {
  zoom: 12,
  center: brooklyn,
  mapTypeControlOptions: {
     mapTypeIds: ['roadmap', MY_MAPTYPE_ID]
  },
  mapTypeId: MY_MAPTYPE_ID
};

// Create our map. This creation will implicitly create a
// map type registry.
map = new google.maps.Map(document.getElementById('map'),
    mapOptions);

// Create your custom map type using your own code.
// (See below.)
var myMapType = new MyMapType();

// Set the registry to associate 'mymap' with the
// custom map type we created, and set the map to
// show that map type.
map.mapTypes.set(MY_MAPTYPE_ID, myMapType);

Mappe con stile

StyledMapType ti consente di personalizzare la presentazione delle mappe di base di Google standard, modificando la visualizzazione di elementi come strade, parchi e aree urbanizzate in modo da riflettere uno stile diverso da quello utilizzato nel tipo di mappa predefinito.

Per ulteriori informazioni su StyledMapType, consulta Utilizzare le dichiarazioni di stile JSON incorporate.

Tipi di mappe personalizzate

L'API Maps JavaScript supporta la visualizzazione e la gestione di tipi di mappe personalizzate, consentendoti di implementare le tue immagini della mappa o gli overlay delle schede.

Nell'API Maps JavaScript esistono diverse possibili implementazioni di tipi di mappe:

  • Insiemi di riquadri standard costituiti da immagini che insieme costituiscono mappe cartografiche complete. Questi set di riquadri sono noti anche come tipi di mappe di base. Questi tipi di mappa agiscono e si comportano come i tipi di mappa predefiniti esistenti: roadmap, satellite, hybrid e terrain. Puoi aggiungere il tuo tipo di mappa personalizzata all'array mapTypes di una mappa per consentire all'interfaccia utente dell'API Maps JavaScript di trattare il tipo di mappa personalizzata come un tipo di mappa standard (includendolo, ad esempio, nel controllo MapType).
  • Overlay di riquadri di immagini visualizzati sopra i tipi di mappe di base esistenti. In genere, questi tipi di mappe vengono utilizzati per integrare un tipo di mappa esistente al fine di visualizzare informazioni aggiuntive e sono spesso limitati a località e/o livelli di zoom specifici. Tieni presente che questi riquadri possono essere trasparenti, consentendoti di aggiungere elementi alle mappe esistenti.
  • Tipi di mappe non in formato immagine, che ti consentono di manipolare la visualizzazione delle informazioni sulla mappa al livello più fondamentale.

Ciascuna di queste opzioni si basa sulla creazione di un Cursore che implementa l'interfaccia MapType. Inoltre, la classe ImageMapType fornisce alcuni comportamenti integrati per semplificare la creazione di tipi di mappe di immagini.

Interfaccia MapType

Prima di creare classi che implementano MapType, è importante capire come Google Maps determina le coordinate e decide quali parti della mappa mostrare. Devi implementare una logica simile per qualsiasi tipo di mappa di base o in overlay. Leggi la guida alle coordinate di mappe e riquadri.

I tipi di mappe personalizzate devono implementare l'interfaccia MapType. Questa interfaccia specifica determinate proprietà e metodi che consentono all'API di avviare richieste ai tuoi tipi di mappa quando l'API determina che deve visualizzare i riquadri della mappa all'interno dell'area visibile e del livello di zoom corrente. Gestisci queste richieste per decidere quale riquadro caricare.

Nota: puoi creare la tua classe per implementare questa interfaccia. In alternativa, se hai immagini compatibili, puoi utilizzare la classe ImageMapType che implementa già questa interfaccia.

Le classi che implementano l'interfaccia MapType richiedono di definire e compilare le seguenti proprietà:

  • tileSize (obbligatorio) specifica le dimensioni del riquadro (di tipo google.maps.Size). Le dimensioni devono essere rettangolari anche se non devono essere quadrate.
  • maxZoom (obbligatorio) specifica il livello di zoom massimo a cui visualizzare i riquadri di questo tipo di mappa.
  • minZoom (facoltativo) specifica il livello di zoom minimo a cui visualizzare il riquadro di questo tipo di mappa. Per impostazione predefinita, questo valore è 0, a indicare che non esiste un livello di zoom minimo.
  • name (facoltativo) specifica il nome di questo tipo di mappa. Questa proprietà è necessaria solo se vuoi che questo tipo di mappa sia selezionabile all'interno di un controllo MapType. (consulta la sezione Opzioni di controllo).
  • alt (facoltativo) specifica il testo alternativo per questo tipo di mappa, visualizzato come testo visualizzato quando si passa il mouse sopra. Questa proprietà è necessaria solo se vuoi che questo tipo di mappa sia selezionabile in un controllo MapType. (consulta la sezione Opzioni di controllo).

Inoltre, le classi che implementano l'interfaccia MapType devono implementare i seguenti metodi:

  • getTile() (obbligatorio) viene chiamato ogni volta che l'API determina che la mappa deve visualizzare nuovi riquadri per l'area visibile specificata. Il metodo getTile() deve avere la seguente firma:

    getTile(tileCoord:Point,zoom:number,ownerDocument:Document):Node

    L'API determina se deve chiamare getTile() in base alle proprietà tileSize, minZoom e maxZoom di MapType e all'area visibile e al livello di zoom attuali della mappa. L'handler per questo metodo deve restituire un elemento HTML dato un valore passato per coordinate, livello di zoom e elemento DOM a cui aggiungere l'immagine del riquadro.

  • releaseTile() (facoltativo) viene chiamato ogni volta che l'API determina che la mappa deve rimuovere un riquadro quando non è più visibile. Questo metodo deve avere la seguente firma:

    releaseTile(tile:Node)

    In genere, devi gestire la rimozione di eventuali elementi collegati ai riquadri della mappa al momento dell'aggiunta alla mappa. Ad esempio, se hai collegato listener di eventi agli overlay delle schede della mappa, devi rimuoverli qui.

Il metodo getTile() funge da controller principale per determinare quali riquadri caricare in un determinato viewport.

Tipi di mappe di base

I tipi di mappe che crei in questo modo possono essere utilizzati da soli o combinati con altri tipi di mappe come overlay. I tipi di mappa autonomo sono noti come tipi di mappe base. Potresti volere che l'API trattenga questi MapType personalizzati come qualsiasi altro tipo di mappa di base esistente (ROADMAP, TERRAIN e così via). Per farlo, aggiungi il MapType personalizzato alla proprietà mapTypes di Map. Questa proprietà è di tipo MapTypeRegistry.

Il seguente codice crea una base MapType per visualizzare le coordinate dei riquadri di una mappa e disegna un contorno dei riquadri:

TypeScript

/*
 * This demo demonstrates how to replace default map tiles with custom imagery.
 * In this case, the CoordMapType displays gray tiles annotated with the tile
 * coordinates.
 *
 * Try panning and zooming the map to see how the coordinates change.
 */

class CoordMapType {
  tileSize: google.maps.Size;
  maxZoom = 19;
  name = "Tile #s";
  alt = "Tile Coordinate Map Type";

  constructor(tileSize: google.maps.Size) {
    this.tileSize = tileSize;
  }

  getTile(
    coord: google.maps.Point,
    zoom: number,
    ownerDocument: Document
  ): HTMLElement {
    const div = ownerDocument.createElement("div");

    div.innerHTML = String(coord);
    div.style.width = this.tileSize.width + "px";
    div.style.height = this.tileSize.height + "px";
    div.style.fontSize = "10";
    div.style.borderStyle = "solid";
    div.style.borderWidth = "1px";
    div.style.borderColor = "#AAAAAA";
    div.style.backgroundColor = "#E5E3DF";
    return div;
  }

  releaseTile(tile: HTMLElement): void {}
}

function initMap(): void {
  const map = new google.maps.Map(
    document.getElementById("map") as HTMLElement,
    {
      zoom: 10,
      center: { lat: 41.85, lng: -87.65 },
      streetViewControl: false,
      mapTypeId: "coordinate",
      mapTypeControlOptions: {
        mapTypeIds: ["coordinate", "roadmap"],
        style: google.maps.MapTypeControlStyle.DROPDOWN_MENU,
      },
    }
  );

  map.addListener("maptypeid_changed", () => {
    const showStreetViewControl =
      (map.getMapTypeId() as string) !== "coordinate";

    map.setOptions({
      streetViewControl: showStreetViewControl,
    });
  });

  // Now attach the coordinate map type to the map's registry.
  map.mapTypes.set(
    "coordinate",
    new CoordMapType(new google.maps.Size(256, 256))
  );
}

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

/*
 * This demo demonstrates how to replace default map tiles with custom imagery.
 * In this case, the CoordMapType displays gray tiles annotated with the tile
 * coordinates.
 *
 * Try panning and zooming the map to see how the coordinates change.
 */
class CoordMapType {
  tileSize;
  maxZoom = 19;
  name = "Tile #s";
  alt = "Tile Coordinate Map Type";
  constructor(tileSize) {
    this.tileSize = tileSize;
  }
  getTile(coord, zoom, ownerDocument) {
    const div = ownerDocument.createElement("div");

    div.innerHTML = String(coord);
    div.style.width = this.tileSize.width + "px";
    div.style.height = this.tileSize.height + "px";
    div.style.fontSize = "10";
    div.style.borderStyle = "solid";
    div.style.borderWidth = "1px";
    div.style.borderColor = "#AAAAAA";
    div.style.backgroundColor = "#E5E3DF";
    return div;
  }
  releaseTile(tile) {}
}

function initMap() {
  const map = new google.maps.Map(document.getElementById("map"), {
    zoom: 10,
    center: { lat: 41.85, lng: -87.65 },
    streetViewControl: false,
    mapTypeId: "coordinate",
    mapTypeControlOptions: {
      mapTypeIds: ["coordinate", "roadmap"],
      style: google.maps.MapTypeControlStyle.DROPDOWN_MENU,
    },
  });

  map.addListener("maptypeid_changed", () => {
    const showStreetViewControl = map.getMapTypeId() !== "coordinate";

    map.setOptions({
      streetViewControl: showStreetViewControl,
    });
  });
  // Now attach the coordinate map type to the map's registry.
  map.mapTypes.set(
    "coordinate",
    new CoordMapType(new google.maps.Size(256, 256)),
  );
}

window.initMap = initMap;
Visualizza esempio

Prova Sample

Tipi di mappe in overlay

Alcuni tipi di mappe sono progettati per funzionare su quelli esistenti. Questi tipi di mappe possono avere livelli trasparenti che indicano punti di interesse o mostrano dati aggiuntivi all'utente.

In questi casi, non vuoi che il tipo di mappa venga trattato come entità separata, ma come overlay. Per farlo, aggiungi il tipo di mappa a un MapType esistente utilizzando direttamente la proprietà overlayMapTypes di Map. Questa proprietà contiene un MVCArray di MapType. Tutti i tipi di mappa (base e in overlay) vengono visualizzati all'interno del livello mapPane. I tipi di mappe in overlay vengono visualizzati sopra la mappa di base a cui sono associati, nell'ordine in cui appaiono nell'array Map.overlayMapTypes (gli overlay con valori di indice più elevati vengono visualizzati davanti a quelli con valori di indice più bassi).

L'esempio seguente è identico a quello precedente, tranne per il fatto che abbiamo creato un overlay di riquadri MapType sul tipo di mappa ROADMAP:

TypeScript

/*
 * This demo illustrates the coordinate system used to display map tiles in the
 * API.
 *
 * Tiles in Google Maps are numbered from the same origin as that for
 * pixels. For Google's implementation of the Mercator projection, the origin
 * tile is always at the northwest corner of the map, with x values increasing
 * from west to east and y values increasing from north to south.
 *
 * Try panning and zooming the map to see how the coordinates change.
 */

class CoordMapType implements google.maps.MapType {
  tileSize: google.maps.Size;
  alt: string|null = null;
  maxZoom: number = 17;
  minZoom: number = 0;
  name: string|null = null;
  projection: google.maps.Projection|null = null;
  radius: number = 6378137;

  constructor(tileSize: google.maps.Size) {
    this.tileSize = tileSize;
  }
  getTile(
    coord: google.maps.Point,
    zoom: number,
    ownerDocument: Document
  ): HTMLElement {
    const div = ownerDocument.createElement("div");

    div.innerHTML = String(coord);
    div.style.width = this.tileSize.width + "px";
    div.style.height = this.tileSize.height + "px";
    div.style.fontSize = "10";
    div.style.borderStyle = "solid";
    div.style.borderWidth = "1px";
    div.style.borderColor = "#AAAAAA";
    return div;
  }
  releaseTile(tile: Element): void {}
}

function initMap(): void {
  const map = new google.maps.Map(
    document.getElementById("map") as HTMLElement,
    {
      zoom: 10,
      center: { lat: 41.85, lng: -87.65 },
    }
  );

  // Insert this overlay map type as the first overlay map type at
  // position 0. Note that all overlay map types appear on top of
  // their parent base map.
  const coordMapType = new CoordMapType(new google.maps.Size(256, 256))
  map.overlayMapTypes.insertAt(
    0,
    coordMapType
  );
}

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

/*
 * This demo illustrates the coordinate system used to display map tiles in the
 * API.
 *
 * Tiles in Google Maps are numbered from the same origin as that for
 * pixels. For Google's implementation of the Mercator projection, the origin
 * tile is always at the northwest corner of the map, with x values increasing
 * from west to east and y values increasing from north to south.
 *
 * Try panning and zooming the map to see how the coordinates change.
 */
class CoordMapType {
  tileSize;
  alt = null;
  maxZoom = 17;
  minZoom = 0;
  name = null;
  projection = null;
  radius = 6378137;
  constructor(tileSize) {
    this.tileSize = tileSize;
  }
  getTile(coord, zoom, ownerDocument) {
    const div = ownerDocument.createElement("div");

    div.innerHTML = String(coord);
    div.style.width = this.tileSize.width + "px";
    div.style.height = this.tileSize.height + "px";
    div.style.fontSize = "10";
    div.style.borderStyle = "solid";
    div.style.borderWidth = "1px";
    div.style.borderColor = "#AAAAAA";
    return div;
  }
  releaseTile(tile) {}
}

function initMap() {
  const map = new google.maps.Map(document.getElementById("map"), {
    zoom: 10,
    center: { lat: 41.85, lng: -87.65 },
  });
  // Insert this overlay map type as the first overlay map type at
  // position 0. Note that all overlay map types appear on top of
  // their parent base map.
  const coordMapType = new CoordMapType(new google.maps.Size(256, 256));

  map.overlayMapTypes.insertAt(0, coordMapType);
}

window.initMap = initMap;
Visualizza esempio

Prova Sample

Tipi di mappe di immagini

L'implementazione di un MapType per fungere da tipo di mappa base può essere un'attività laboriosa e dispendiosa in termini di tempo. L'API fornisce una classe speciale che implementa l'interfaccia MapType per i tipi di mappe più comuni: mappe costituite da riquadri composti da singoli file immagine.

Questa classe, la classe ImageMapType, viene costruita utilizzando una specifica dell'oggetto ImageMapTypeOptions che definisce le seguenti proprietà obbligatorie:

  • tileSize (obbligatorio) specifica le dimensioni del riquadro (di tipo google.maps.Size). Le dimensioni devono essere rettangolari anche se non devono essere quadrate.
  • getTileUrl (obbligatorio) specifica la funzione, solitamente fornita come literal di funzione in linea, per gestire la selezione del riquadro dell'immagine corretto in base alle coordinate geografiche e al livello di zoom forniti.

Il codice seguente implementa un ImageMapType di base utilizzando i riquadri della luna di Google. L'esempio utilizza una funzione di normalizzazione per assicurarsi che i riquadri si ripetano sull'asse x, ma non sull'asse y della mappa.

TypeScript

function initMap(): void {
  const map = new google.maps.Map(
    document.getElementById("map") as HTMLElement,
    {
      center: { lat: 0, lng: 0 },
      zoom: 1,
      streetViewControl: false,
      mapTypeControlOptions: {
        mapTypeIds: ["moon"],
      },
    }
  );

  const moonMapType = new google.maps.ImageMapType({
    getTileUrl: function (coord, zoom): string {
      const normalizedCoord = getNormalizedCoord(coord, zoom);

      if (!normalizedCoord) {
        return "";
      }

      const bound = Math.pow(2, zoom);
      return (
        "https://mw1.google.com/mw-planetary/lunar/lunarmaps_v1/clem_bw" +
        "/" +
        zoom +
        "/" +
        normalizedCoord.x +
        "/" +
        (bound - normalizedCoord.y - 1) +
        ".jpg"
      );
    },
    tileSize: new google.maps.Size(256, 256),
    maxZoom: 9,
    minZoom: 0,
    // @ts-ignore TODO 'radius' does not exist in type 'ImageMapTypeOptions'
    radius: 1738000,
    name: "Moon",
  });

  map.mapTypes.set("moon", moonMapType);
  map.setMapTypeId("moon");
}

// Normalizes the coords that tiles repeat across the x axis (horizontally)
// like the standard Google map tiles.
function getNormalizedCoord(coord, zoom) {
  const y = coord.y;
  let x = coord.x;

  // tile range in one direction range is dependent on zoom level
  // 0 = 1 tile, 1 = 2 tiles, 2 = 4 tiles, 3 = 8 tiles, etc
  const tileRange = 1 << zoom;

  // don't repeat across y-axis (vertically)
  if (y < 0 || y >= tileRange) {
    return null;
  }

  // repeat across x-axis
  if (x < 0 || x >= tileRange) {
    x = ((x % tileRange) + tileRange) % tileRange;
  }

  return { x: x, y: y };
}

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

function initMap() {
  const map = new google.maps.Map(document.getElementById("map"), {
    center: { lat: 0, lng: 0 },
    zoom: 1,
    streetViewControl: false,
    mapTypeControlOptions: {
      mapTypeIds: ["moon"],
    },
  });
  const moonMapType = new google.maps.ImageMapType({
    getTileUrl: function (coord, zoom) {
      const normalizedCoord = getNormalizedCoord(coord, zoom);

      if (!normalizedCoord) {
        return "";
      }

      const bound = Math.pow(2, zoom);
      return (
        "https://mw1.google.com/mw-planetary/lunar/lunarmaps_v1/clem_bw" +
        "/" +
        zoom +
        "/" +
        normalizedCoord.x +
        "/" +
        (bound - normalizedCoord.y - 1) +
        ".jpg"
      );
    },
    tileSize: new google.maps.Size(256, 256),
    maxZoom: 9,
    minZoom: 0,
    // @ts-ignore TODO 'radius' does not exist in type 'ImageMapTypeOptions'
    radius: 1738000,
    name: "Moon",
  });

  map.mapTypes.set("moon", moonMapType);
  map.setMapTypeId("moon");
}

// Normalizes the coords that tiles repeat across the x axis (horizontally)
// like the standard Google map tiles.
function getNormalizedCoord(coord, zoom) {
  const y = coord.y;
  let x = coord.x;
  // tile range in one direction range is dependent on zoom level
  // 0 = 1 tile, 1 = 2 tiles, 2 = 4 tiles, 3 = 8 tiles, etc
  const tileRange = 1 << zoom;

  // don't repeat across y-axis (vertically)
  if (y < 0 || y >= tileRange) {
    return null;
  }

  // repeat across x-axis
  if (x < 0 || x >= tileRange) {
    x = ((x % tileRange) + tileRange) % tileRange;
  }
  return { x: x, y: y };
}

window.initMap = initMap;
Visualizza esempio

Prova Sample

Proiezioni

La Terra è una sfera tridimensionale (approssimativamente), mentre una mappa è una superficie bidimensionale piatta. La mappa che vedi nell'API Maps JavaScript, come qualsiasi mappa piana della Terra, è una proiezione di quella sfera su una superficie piana. Nei termini più semplici, una proiezione può essere definita come una mappatura dei valori di latitudine/longitudine in coordinate sulla mappa della proiezione.

Le proiezioni nell'API Maps JavaScript devono implementare l'interfaccia Projection. Un'implementazione di Projection deve fornire non solo una mappatura da un sistema di coordinate all'altro, ma una mappatura bidirezionale. In altre parole, devi definire come tradurre le coordinate terrestri (oggetti LatLng) nel sistema di coordinate mondiali della classe Projection e viceversa. Google Maps utilizza la proiezione di Mercatore per creare le sue mappe dai dati geografici e convertire gli eventi sulla mappa in coordinate geografiche. Puoi ottenere questa proiezione chiamando getProjection() su Map (o su uno dei tipi di base standard MapType). Per la maggior parte degli utilizzi, questo valore standard Projection sarà sufficiente, ma puoi anche definire e utilizzare le tue proiezioni personalizzate.

Implementare una proiezione

Quando implementi una proiezione personalizzata, devi definire alcuni elementi:

  • Le formule per mappare le coordinate di latitudine e longitudine su un piano cartesiano e le formule corrispondenti per la mappatura da un piano cartesiano alle coordinate di latitudine e longitudine. L'interfaccia Projection supporta solo le trasformazioni in coordinate rettilinee.
  • La dimensione del riquadro di base. Tutti i riquadri devono essere rettangolari.
  • Le "dimensioni del mondo" di una mappa che utilizza il set di riquadri di base impostato sul livello di zoom 0. Tieni presente che per le mappe costituite da un riquadro con zoom 0, le dimensioni del mondo e le dimensioni del riquadro di base sono identiche.

Trasformazioni coordinate nelle proiezioni

Ogni proiezione fornisce due metodi per tradurre tra questi due sistemi di coordinate, consentendo di convertire le coordinate geografiche e mondiali:

  • Il metodo Projection.fromLatLngToPoint() converte un valore LatLng in una coordinata mondiale. Questo metodo viene utilizzato per posizionare gli overlay sulla mappa (e per posizionare la mappa stessa).
  • Il metodo Projection.fromPointToLatLng() converte una coordinata mondiale in un valore LatLng. Questo metodo viene utilizzato per convertire eventi come i clic che si verificano sulla mappa in coordinate geografiche.

Google Maps presuppone che le proiezioni siano rettilinee.

In genere, puoi utilizzare una proiezione in due casi: per creare una mappa del mondo o una mappa di un'area locale. Nel primo caso, devi assicurarti che la proiezione sia anche rettangolare e normale a tutte le longitudini. Alcune proiezioni (in particolare quelle coniche) possono essere "localmente normali" (ovvero puntare a nord) ma deviare dal nord geografico; ad esempio, più la mappa è posizionata rispetto a una longitudine di riferimento. Puoi utilizzare una proiezione di questo tipo a livello locale, ma tieni presente che la proiezione è necessariamente imprecisa e gli errori di trasformazione diventeranno sempre più evidenti man mano che ti allontani dalla longitudine di riferimento.

Selezione dei riquadri della mappa nelle proiezioni

Le proiezioni non sono utili solo per determinare le posizioni di località o overlay, ma anche per posizionare i singoli riquadri della mappa. L'API Maps JavaScript esegue il rendering delle mappe di base utilizzando un'interfaccia MapType che deve dichiarare sia una proprietà projection per identificare la proiezione della mappa sia un metodo getTile() per recuperare i riquadri della mappa in base ai valori delle coordinate dei riquadri. Le coordinate dei riquadri si basano sia sulle dimensioni di base dei riquadri (che devono essere rettangolari) sia sulle "dimensioni del mondo" della mappa, ovvero le dimensioni in pixel del mondo della mappa al livello di zoom 0. Per le mappe costituite da un riquadro a livello di zoom 0, le dimensioni del riquadro e quelle del mondo sono identiche.

Definisci le dimensioni della piastrella di base all'interno della proprietà tileSize di MapType. Le dimensioni del mondo vengono definite implicitamente all'interno dei metodi fromLatLngToPoint() e fromPointToLatLng() della proiezione.

Poiché la selezione delle immagini dipende da questi valori trasmessi, è utile nominare le immagini che possono essere selezionate in modo programmatico in base a questi valori, ad esempio map_zoom_tileX_tileY.png.

L'esempio seguente definisce un ImageMapType utilizzando la proiezione Gall-Peters:

TypeScript

// This example defines an image map type using the Gall-Peters
// projection.
// https://en.wikipedia.org/wiki/Gall%E2%80%93Peters_projection

function initMap(): void {
  // Create a map. Use the Gall-Peters map type.
  const map = new google.maps.Map(
    document.getElementById("map") as HTMLElement,
    {
      zoom: 0,
      center: { lat: 0, lng: 0 },
      mapTypeControl: false,
    }
  );

  initGallPeters();
  map.mapTypes.set("gallPeters", gallPetersMapType);
  map.setMapTypeId("gallPeters");

  // Show the lat and lng under the mouse cursor.
  const coordsDiv = document.getElementById("coords") as HTMLElement;

  map.controls[google.maps.ControlPosition.TOP_CENTER].push(coordsDiv);
  map.addListener("mousemove", (event: google.maps.MapMouseEvent) => {
    coordsDiv.textContent =
      "lat: " +
      Math.round(event.latLng!.lat()) +
      ", " +
      "lng: " +
      Math.round(event.latLng!.lng());
  });

  // Add some markers to the map.
  map.data.setStyle((feature) => {
    return {
      title: feature.getProperty("name") as string,
      optimized: false,
    };
  });
  map.data.addGeoJson(cities);
}

let gallPetersMapType;

function initGallPeters() {
  const GALL_PETERS_RANGE_X = 800;
  const GALL_PETERS_RANGE_Y = 512;

  // Fetch Gall-Peters tiles stored locally on our server.
  gallPetersMapType = new google.maps.ImageMapType({
    getTileUrl: function (coord, zoom) {
      const scale = 1 << zoom;

      // Wrap tiles horizontally.
      const x = ((coord.x % scale) + scale) % scale;

      // Don't wrap tiles vertically.
      const y = coord.y;

      if (y < 0 || y >= scale) return "";

      return (
        "https://developers.google.com/maps/documentation/" +
        "javascript/examples/full/images/gall-peters_" +
        zoom +
        "_" +
        x +
        "_" +
        y +
        ".png"
      );
    },
    tileSize: new google.maps.Size(GALL_PETERS_RANGE_X, GALL_PETERS_RANGE_Y),
    minZoom: 0,
    maxZoom: 1,
    name: "Gall-Peters",
  });

  // Describe the Gall-Peters projection used by these tiles.
  gallPetersMapType.projection = {
    fromLatLngToPoint: function (latLng) {
      const latRadians = (latLng.lat() * Math.PI) / 180;
      return new google.maps.Point(
        GALL_PETERS_RANGE_X * (0.5 + latLng.lng() / 360),
        GALL_PETERS_RANGE_Y * (0.5 - 0.5 * Math.sin(latRadians))
      );
    },
    fromPointToLatLng: function (point, noWrap) {
      const x = point.x / GALL_PETERS_RANGE_X;
      const y = Math.max(0, Math.min(1, point.y / GALL_PETERS_RANGE_Y));

      return new google.maps.LatLng(
        (Math.asin(1 - 2 * y) * 180) / Math.PI,
        -180 + 360 * x,
        noWrap
      );
    },
  };
}

// GeoJSON, describing the locations and names of some cities.
const cities = {
  type: "FeatureCollection",
  features: [
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-87.65, 41.85] },
      properties: { name: "Chicago" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-149.9, 61.218] },
      properties: { name: "Anchorage" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-99.127, 19.427] },
      properties: { name: "Mexico City" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-0.126, 51.5] },
      properties: { name: "London" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [28.045, -26.201] },
      properties: { name: "Johannesburg" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [15.322, -4.325] },
      properties: { name: "Kinshasa" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [151.207, -33.867] },
      properties: { name: "Sydney" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [0, 0] },
      properties: { name: "0°N 0°E" },
    },
  ],
};

declare global {
  interface Window {
    initMap: () => void;
  }
}
window.initMap = initMap;

JavaScript

// This example defines an image map type using the Gall-Peters
// projection.
// https://en.wikipedia.org/wiki/Gall%E2%80%93Peters_projection
function initMap() {
  // Create a map. Use the Gall-Peters map type.
  const map = new google.maps.Map(document.getElementById("map"), {
    zoom: 0,
    center: { lat: 0, lng: 0 },
    mapTypeControl: false,
  });

  initGallPeters();
  map.mapTypes.set("gallPeters", gallPetersMapType);
  map.setMapTypeId("gallPeters");

  // Show the lat and lng under the mouse cursor.
  const coordsDiv = document.getElementById("coords");

  map.controls[google.maps.ControlPosition.TOP_CENTER].push(coordsDiv);
  map.addListener("mousemove", (event) => {
    coordsDiv.textContent =
      "lat: " +
      Math.round(event.latLng.lat()) +
      ", " +
      "lng: " +
      Math.round(event.latLng.lng());
  });
  // Add some markers to the map.
  map.data.setStyle((feature) => {
    return {
      title: feature.getProperty("name"),
      optimized: false,
    };
  });
  map.data.addGeoJson(cities);
}

let gallPetersMapType;

function initGallPeters() {
  const GALL_PETERS_RANGE_X = 800;
  const GALL_PETERS_RANGE_Y = 512;

  // Fetch Gall-Peters tiles stored locally on our server.
  gallPetersMapType = new google.maps.ImageMapType({
    getTileUrl: function (coord, zoom) {
      const scale = 1 << zoom;
      // Wrap tiles horizontally.
      const x = ((coord.x % scale) + scale) % scale;
      // Don't wrap tiles vertically.
      const y = coord.y;

      if (y < 0 || y >= scale) return "";
      return (
        "https://developers.google.com/maps/documentation/" +
        "javascript/examples/full/images/gall-peters_" +
        zoom +
        "_" +
        x +
        "_" +
        y +
        ".png"
      );
    },
    tileSize: new google.maps.Size(GALL_PETERS_RANGE_X, GALL_PETERS_RANGE_Y),
    minZoom: 0,
    maxZoom: 1,
    name: "Gall-Peters",
  });
  // Describe the Gall-Peters projection used by these tiles.
  gallPetersMapType.projection = {
    fromLatLngToPoint: function (latLng) {
      const latRadians = (latLng.lat() * Math.PI) / 180;
      return new google.maps.Point(
        GALL_PETERS_RANGE_X * (0.5 + latLng.lng() / 360),
        GALL_PETERS_RANGE_Y * (0.5 - 0.5 * Math.sin(latRadians)),
      );
    },
    fromPointToLatLng: function (point, noWrap) {
      const x = point.x / GALL_PETERS_RANGE_X;
      const y = Math.max(0, Math.min(1, point.y / GALL_PETERS_RANGE_Y));
      return new google.maps.LatLng(
        (Math.asin(1 - 2 * y) * 180) / Math.PI,
        -180 + 360 * x,
        noWrap,
      );
    },
  };
}

// GeoJSON, describing the locations and names of some cities.
const cities = {
  type: "FeatureCollection",
  features: [
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-87.65, 41.85] },
      properties: { name: "Chicago" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-149.9, 61.218] },
      properties: { name: "Anchorage" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-99.127, 19.427] },
      properties: { name: "Mexico City" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [-0.126, 51.5] },
      properties: { name: "London" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [28.045, -26.201] },
      properties: { name: "Johannesburg" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [15.322, -4.325] },
      properties: { name: "Kinshasa" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [151.207, -33.867] },
      properties: { name: "Sydney" },
    },
    {
      type: "Feature",
      geometry: { type: "Point", coordinates: [0, 0] },
      properties: { name: "0°N 0°E" },
    },
  ],
};

window.initMap = initMap;
Visualizza esempio

Prova Sample