Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Marker not appearing for simple example #453

Closed
7 tasks done
toby-griffiths opened this issue Feb 28, 2018 · 18 comments
Closed
7 tasks done

Marker not appearing for simple example #453

toby-griffiths opened this issue Feb 28, 2018 · 18 comments

Comments

@toby-griffiths
Copy link

Please make sure to check the following boxes before submitting an issue.
Issues opened without using this template will be closed unless they have a good
reason not to follow this template.

  • All peer dependencies are installed: React, ReactDOM and Leaflet.
  • Using a supported version of React and ReactDOM (v15.x or v16.x).
  • Using the supported version of Leaflet (v1.3.x) and its corresponding CSS file is loaded.
  • Using the latest stable version of React-Leaflet.
  • The issue has not already been reported.
  • Make sure you have followed the quick start guide for Leaflet.
  • Make sure you have fully read the documentation and that you understand the limitations.

Expected behavior

Add the simple, example map to an App, and expect to see a map with a map marker that can be clicked to open the popup.

Actual behavior

Map appear, the marker does not (or more accurately the image of the map is broken. The element exists on the page).

Clicking on the (invisible) marker opens the popup.

It appears that the map marker src attribute is garbled…

")marker-icon.png

You'll notice the )marker-icon.png part on the end, that if removed, allows the marker to appear.

Steps to reproduce

Create App component containing example map & render on page…

import PropTypes                         from 'prop-types';
import React, { Component }              from 'react';
import './App.css';
import 'leaflet/dist/leaflet.css';
import { Map, Marker, Popup, TileLayer } from 'react-leaflet';

const coordinates = [42, 35];

class App extends Component {

  render() {

    return <Map center={coordinates} zoom={12} style={{height: '350px'}}>
      <TileLayer
        attribution="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>OpenStreetMap</a> contributors"
        url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
      />
      <Marker position={coordinates}>
        <Popup>
            <span>
              A pretty CSS3 popup. <br/> Easily customizable.
            </span>
        </Popup>
      </Marker>
    </Map>;

  }

}

export default App;

… and render the App…

ReactDOM.render(
  <App/>,
  document.getElementById('root')
);

Please let me know if you need more info to help resolve this.

@toby-griffiths
Copy link
Author

OK. It turns out that the problem was caused by including the leaflet CSS in the component's imports.

I've now just included a link to the CDN hosted leaflet.css file and it's working OK, but it would be good if this could be patched to work with create-react-app webpack config.

@PaulLeCam
Copy link
Owner

Yes you need to properly setup Leaflet in order to make this library work.

@benneq
Copy link

benneq commented Aug 4, 2018

I've copied some code from google, and now it's working without CDN:

import * as React from "react";
import { Map, TileLayer, Marker, Popup } from "react-leaflet";
import 'leaflet/dist/leaflet.css';
import L from 'leaflet';

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')
});

export const MyMapComponent = (props) => {
    return (
        <Map center={props.center} zoom={props.zoom} style={{height: '350px'}}>
          <TileLayer
            attribution="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>OpenStreetMap</a> contributors"
            url="https://cartodb-basemaps-{s}.global.ssl.fastly.net/light_all/{z}/{x}/{y}.png"
          />
          <Marker position={props.center}>
            <Popup>
              A pretty CSS3 popup. <br /> Easily customizable.
            </Popup>
          </Marker>
        </Map>
    )
}

@gurre
Copy link

gurre commented Apr 12, 2019

FYI: This does not work for server-side rendering since leaflet assumes window is defined.

@PredokMiF
Copy link

PredokMiF commented Apr 28, 2019

When it will work "out of the box"? Strange prop deleteing and monkey patching of "mergeOptions' - smells bad ((

@ivnsch
Copy link

ivnsch commented Jun 11, 2019

What is the solution? My marker is also invisible. I'm importing the css like this, in App.css:

@import url('https://unpkg.com/leaflet@1.5.1/dist/leaflet.css');

@PredokMiF
Copy link

PredokMiF commented Jun 11, 2019

What is the solution? My marker is also invisible. I'm importing the css like this, in App.css:

@import url('https://unpkg.com/leaflet@1.5.1/dist/leaflet.css');

You need insert code from few topics above with L.Icon.Default.mergeOptions. It helped me

@ivnsch
Copy link

ivnsch commented Jun 11, 2019

@PredokMiF I had an error running that code (can try later again and add it here) and it's anyway a workaround, isn't it?

@ebartan
Copy link

ebartan commented Jul 30, 2019

don't forget delete L.Icon.Default.prototype._getIconUrl;

@jpbow
Copy link

jpbow commented Oct 11, 2019

If anyone's trying to use this with SSR, this worked for me (I'm using react-leaflet-universal too).

import * as React from "react";
import { Map, TileLayer, Marker, Popup } from "react-leaflet-universal";
import "leaflet/dist/leaflet.css";

export default () => {
  React.useEffect(() => {
    const L = require("leaflet");

    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")
    });
  }, []);

  return (
    <Map center={[51.505, -0.09]} zoom={13} style={{ height: "100vh" }}>
      <TileLayer
        attribution='&amp;copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors'
        url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
      />
      <Marker position={[51.505, -0.09]}>
        <Popup>
          A pretty CSS3 popup. <br /> Easily customizable.
        </Popup>
      </Marker>
    </Map>
  );
};

@ulantiger
Copy link

  1. Copy all files from node_modules/leaflet/dist/images into public/img or any other folder inside public directory
  2. import L from 'leaflet'
  3. and add this line inside your class or function
    L.Icon.Default.imagePath='img/' or point to your directory where your marker files are located
    I found it as the easiest way to fix it
    Good luck to everyone!

@Kenska
Copy link

Kenska commented Jul 30, 2020

Another quick fix that I found: you can use the background-image CSS property of leaflet-marker-icon and set the Base64 marker icon data.

img.leaflet-marker-icon
{
    background-image: url("");
}

You can suppress the errors caused by the original issue by setting the icon's imagePath to /.

import L from "leaflet";
L.Icon.Default.imagePath = "/";

@Mloweedgar
Copy link

I've copied some code from google, and now it's working without CDN:

import * as React from "react";
import { Map, TileLayer, Marker, Popup } from "react-leaflet";
import 'leaflet/dist/leaflet.css';
import L from 'leaflet';

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')
});

export const MyMapComponent = (props) => {
    return (
        <Map center={props.center} zoom={props.zoom} style={{height: '350px'}}>
          <TileLayer
            attribution="&amp;copy <a href=&quot;http://osm.org/copyright&quot;>OpenStreetMap</a> contributors"
            url="https://cartodb-basemaps-{s}.global.ssl.fastly.net/light_all/{z}/{x}/{y}.png"
          />
          <Marker position={props.center}>
            <Popup>
              A pretty CSS3 popup. <br /> Easily customizable.
            </Popup>
          </Marker>
        </Map>
    )
}

this worked for me. thanks

@zacwang89
Copy link

The following code from Stackoverflow works for me on React-Leaflet v3

import React, { Component } from 'react';
import L from 'leaflet';
import {
    Map, TileLayer, Marker, Popup
} from 'react-leaflet'
import 'leaflet/dist/leaflet.css';
import './style.css';


import icon from 'leaflet/dist/images/marker-icon.png';
import iconShadow from 'leaflet/dist/images/marker-shadow.png';

let DefaultIcon = L.icon({
    iconUrl: icon,
    shadowUrl: iconShadow
});

L.Marker.prototype.options.icon = DefaultIcon;

@injecteer
Copy link

It has to work out of the box. We as lib users do not want to hack our way thru 3rd party library jungle all day long!

gavinr added a commit to gavinr-maps/esri-leaflet-react-demo that referenced this issue Jan 6, 2021
@ghybs
Copy link

ghybs commented Jan 17, 2021

Hi all,

For reference, this is caused by webpack fiddling with URL's in CSS. This intereferes with Leaflet algorithm to detect paths to its default icon images, which dates from before webpack was so used.
You can find more details in issue Leaflet/Leaflet#4968

Leaflet works out-of-the-box.

Webpack (and other build engines) combined with Leaflet does not work out-of-the-box.

Please refer to solution proposed in #4968 (comment): leaflet-defaulticon-compatibility

While I can understand that many developers would have preferred such feature to be incorporated directly in Leaflet core, it has been argued that the added code is useless to a majority of end-users. Therefore, aligning with Leaflet spirit of keeping its core simple, I decided to make it a plugin.

With webpack, another typical solution, once you have configured your loaders:

import L from 'leaflet';
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'),
});

@alekhranjan75
Copy link

The following code from Stackoverflow works for me on React-Leaflet v3

import React, { Component } from 'react';
import L from 'leaflet';
import {
    Map, TileLayer, Marker, Popup
} from 'react-leaflet'
import 'leaflet/dist/leaflet.css';
import './style.css';


import icon from 'leaflet/dist/images/marker-icon.png';
import iconShadow from 'leaflet/dist/images/marker-shadow.png';

let DefaultIcon = L.icon({
    iconUrl: icon,
    shadowUrl: iconShadow
});

L.Marker.prototype.options.icon = DefaultIcon;

Thanks man it worked for react-leaflet version > 3

@DavidNorena
Copy link

why is this closed if it is not solved yet ?

Repository owner locked as off-topic and limited conversation to collaborators Mar 5, 2021
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

No branches or pull requests