32

I got a problem with leaflet.

Everything is working fine in development, but in production, my app isn't able to locate the marker-icon.png and marker-shadow.png images.

It is looking for the path assets/station/images/marker-icon.png

Leaflet js is including like this in my html.erb file

<script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet/0.7.5/leaflet.js"></script>

<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/leaflet/0.7.5/leaflet.css" />

If someone can help!

maluss
  • 602
  • 1
  • 8
  • 14

12 Answers12

56

This is a known bug in Leaflet, the root issue is that Leaflet's icon image location is been wrongly referenced during bundling.

You can verify that this is your issue, buy validating this parameter (in run time): L.Icon.Default.prototype._getIconUrl().
The correct value should be <some_directory>/leaflet/dist/images/.
However if this bug is happening to you, it's value is: ....K5CYII=")undefined

There are different solutions (work around) depending on which bundle-loader you are using (Vanila WebPack, Angular-Cli - superset of WebPack, etc...).

You can see the original issue here (as well as different solutions depending on your bandle-loader):
https://github.com/Leaflet/Leaflet/issues/4968

If you are using Angular-Cli this solution will work for you. Add this code somewhere before setting the Maker:

import { icon, Marker } from 'leaflet';
const iconRetinaUrl = 'assets/marker-icon-2x.png';
const iconUrl = 'assets/marker-icon.png';
const shadowUrl = 'assets/marker-shadow.png';
const iconDefault = icon({
  iconRetinaUrl,
  iconUrl,
  shadowUrl,
  iconSize: [25, 41],
  iconAnchor: [12, 41],
  popupAnchor: [1, -34],
  tooltipAnchor: [16, -28],
  shadowSize: [41, 41]
});
Marker.prototype.options.icon = iconDefault;

(this code will change the broken Marker's url, to a valid image from your assets folder).

And add this code at you angular.json (for Angular version >= 6.x) or at your angular-cli.json (for Angular version <= 5.x):

"assets":
[
   "src/favicon.ico",
   "src/assets",
   {
      "glob": "**/*",
      "input": "node_modules/leaflet/dist/images/", // you may need to change this path, according to your files structure
      "output": "./assets/"
   }
] ...

(this code will copy the original Marker images to the /assets folder so angular-cli could load them)

0x6adb015
  • 7,473
  • 4
  • 24
  • 38
Gil Epshtain
  • 8,670
  • 7
  • 63
  • 89
  • While this could be a nice explanation for Angular, notice that OP never mentioned using a JS bundler, their incorrect path is not a data URI, and includes old (0.x) Leaflet assets from CDN. – ghybs Jul 09 '18 at 01:57
  • 1
    This fix works for a vite (rollup) & vue3 build. Go into `node_modules/leaflet/dist/images` and pluck those three marker png files. Stick them in `public/assets` and build away. – evendiagram Apr 16 '21 at 02:14
  • 1
    Should have been the accepted answer. This works like a charm. Thanks! – Dinesh Shekhawat May 27 '22 at 07:16
24
import "leaflet/dist/images/marker-shadow.png";

if using webpack, just import the image

user9547708
  • 357
  • 2
  • 5
23

Below code worked for me

delete L.Icon.Default.prototype._getIconUrl;
L.Icon.Default.mergeOptions({
  iconRetinaUrl: require("leaflet/dist/images/marker-icon-2x.png"),
  iconUrl: require("leaflet/dist/images/marker-icon.png"),
  shadowUrl: require("leaflet/dist/images/marker-shadow.png")
});
Dostonbek Oripjonov
  • 1,508
  • 1
  • 12
  • 28
  • 1
    got error with above code snippet. Error: "TS2591: Cannot find name 'require'. Do you need to install type definitions for node? Try `npm i @types/node` and then add `node` to the types field in your tsconfig" – Mahesh Vemula May 01 '20 at 10:04
  • 1
    I'm not clear, where are you saying to add this bit of code to? – edjm Jun 15 '21 at 11:35
  • 1
    @NashGC can you elaborate on how you did this with Nuxt? I'm importing leaflet locally into one single component instead of using a plugin, where did you place this block of code? – Thorvald Jul 17 '21 at 17:39
  • 1
    @Thorvald, I've used it in mounted() method of a map component. (component where I use map) – NashGC Jul 22 '21 at 10:31
  • 1
    @NashGC This does work for me.. however my marker disappears in Safari browsers. Any idea how this would be fixed? I'm on leaflet 1.7.1, vue2-leaflet 2.7.0 – Thorvald Oct 30 '21 at 09:16
  • 1
    getting error `Property '_getIconUrl' does not exist on type 'Default'` Using create-react-app with TypeScript – Denis Molodtsov Nov 04 '21 at 16:40
  • 1
    Thanks, it worked for me too ... but without the require function. I just added iconUrl: "/webroot/assets/js/leaflet/images/marker-icon.png", and so on ... of course fitting to my base url. – dlg_ Feb 09 '22 at 23:32
7

Working in Angular 10

For me worked copy paste PNGs files to assets and one command

ngOnInit() { L.Icon.Default.ImagePath = "assets/leaflet/" }

For Angular 13 with Leaflet 1.7

ngOnInit() { L.Icon.Default.imagePath = "assets/leaflet/" }

dc7
  • 401
  • 1
  • 4
  • 13
Mises
  • 4,251
  • 2
  • 19
  • 32
3

You can change the current path of the icon in leaflet css. So don't use the url leaflet.css, download the file and change it.

.leaflet-default-icon-path {
    background-image: url(https://unpkg.com/leaflet@1.7.1/dist/images/marker-icon.png);
}
3

For Vue3 with typescript and vue-cli + Webpack 5, I use a workaround by explicitly defining the required icons in created() hook.

    created() {
        L.Marker.prototype.options.icon = L.icon({
            iconRetinaUrl: require("leaflet/dist/images/marker-icon-2x.png"),
            iconUrl: require("leaflet/dist/images/marker-icon.png"),
            shadowUrl: require("leaflet/dist/images/marker-shadow.png"),
            iconSize: [25, 41],
            iconAnchor: [12, 41],
            popupAnchor: [1, -34],
            tooltipAnchor: [16, -28],
            shadowSize: [41, 41],
        });
    },

After that I create my map in the mounted() hook:

    mounted() {
        const map:L.Map = L.map('map').setView([51.505, -0.09], 13);
        L.tileLayer('https://api.mapbox.com/styles/v1/{id}/tiles/{z}/{x}/{y}?access_token={accessToken}', {
            attribution: 'Map data &copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors, Imagery © <a href="https://www.mapbox.com/">Mapbox</a>',
            maxZoom: 18,
            id: 'mapbox/streets-v11',
            tileSize: 512,
            zoomOffset: -1,
            accessToken: process.env.VUE_APP_MAPBOX_TOKEN
        }).addTo(map);
    }

note: I store my Mapbox in a VUE_APP_ prefixed env variable.

Albert Tinon
  • 136
  • 5
1

I ran into a similar issue using Parcel as the bundler in combination with TypeScript and Leaflet v1.4 (installed via npm, as well as its typings) and solved it using Gil's answer in a slightly different way.

import 'leaflet/dist/leaflet.css';
import 'leaflet-draw/dist/leaflet.draw.css';
import L, {
  LatLngExpression,
  FeatureGroup,
  TileLayerOptions,
  LayerEvent,
  LeafletMouseEvent,
  Marker,
  Layer,
  icon,
  LayerGroup,
  GeoJSON
} from 'leaflet';
import 'leaflet-draw';
import iconRetinaUrl from './assets/marker-icon-2x.png';
import iconUrl from './assets/marker-icon.png';
import shadowUrl from './assets/marker-shadow.png';

// Assign the imported image assets before you do anything with Leaflet.
Marker.prototype.options.icon = icon({
  iconRetinaUrl,
  iconUrl,
  shadowUrl,
  iconSize: [25, 41],
  iconAnchor: [12, 41],
  popupAnchor: [1, -34],
  tooltipAnchor: [16, -28],
  shadowSize: [41, 41],
});

And in another file, I added the required declarations so TypeScript allows me to import png images, e.g.

declare module "*.png" {
  const content: string;
  export default content;
}

Also notice that, in case you use a Leaflet plugin that requires access to these images, you may need to explicitly assign it too, e.g. the Leaflet draw plugin required it as well. Example:

        map.addLayer(drawLayer);
        const drawControl = new L.Control.Draw({
          draw: {
            circle: false,
            circlemarker: false,
            marker: {
              icon: Marker.prototype.options.icon, // Assign icon explicitly
            },
          },
          edit: {
            featureGroup: drawLayer,
          },
        });
        map.addControl(drawControl);
        map.on(L.Draw.Event.CREATED, event => {
          const layer = (event as LayerEvent).layer;
          drawLayer.addLayer(layer);
        });
Erik Vullings
  • 5,550
  • 2
  • 28
  • 23
1

The answer is to update your index.html with leaflet css

 <link rel="stylesheet" href="https://unpkg.com/leaflet@1.7.1/dist/leaflet.css"
   integrity="sha512-xodZBdNfgdfTC5n17Xddt2atTPudf34rdf45erfggrrddfdffdfEdf1HxjVMSvLVW9odfdfcdfdf3rfddfdferdfdfqdfgdfUKLsCC5gdfdfdCXdbqCmblFFAshOdfdfMAS6/kddderfdfeqq/sMZMZ19scRerdfggdfddffd4PsZChyertfgRRRSR7A=="
   crossorigin=""/>

Put the css in header

1

The most portable solution (and well suited for Rails 7) is:

In your Javascript controller, put this before the usual Leaflet map config code:

import { Controller } from "@hotwired/stimulus"

import L from "leaflet";    
delete L.Icon.Default.prototype._getIconUrl;

export default class extends Controller {
  connect() {

    L.Icon.Default.mergeOptions({
      iconRetinaUrl: '',
      iconUrl:       '',
      shadowUrl:     ''
    });


...
... // [Put the rest of the usual Leaflet map config code here]
...
TomDogg
  • 3,803
  • 5
  • 33
  • 60
0

@GilEpshtain 's answer worked for me on Angular 9.1.9.

Additional note - for debugging purposes, you should be able to confirm that the change to angular.json#assets specified by @GilEpshtain has taken effect, by putting one of the icon URLs directly into your browser. For instance http://localhost:4200/assets/marker-icon.png should show you the marker icon.

marracuene
  • 142
  • 11
0

I got it to work by downloading marker-icon.png and marker-shadow.png from https://unpkg.com/leaflet@1.7.1/dist/images/marker-icon.png and https://unpkg.com/leaflet@1.7.1/dist/images/marker-shadow.png respectively and deployed them as image assets to the "/" route of my web app.

-1

Perhaps try upgrading to current version of leaflet- I have never encountered this problem with the marker icons.

https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.0.2/leaflet.css
https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.0.2/leaflet.js
Alex G Rice
  • 1,561
  • 11
  • 16