Leaflet标记在生产环境中未找到

33

我在使用leaflet时遇到了问题。

在开发中一切都正常,但是在生产环境中,我的应用程序无法定位marker-icon.pngmarker-shadow.png图像。

它正在寻找路径assets/station/images/marker-icon.png

在我的html.erb文件中,Leaflet js是这样包含的

<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" />

如果有人可以帮忙!


2
Leaflet的哪个版本?如何部署Leaflet JS和CSS文件?这些文件来自哪里(框架、手动下载等)? - IvanSanchez
我已经编辑了我的帖子! - maluss
请查看下面的答案,如果有帮助,请点赞或标记为已接受。 - Alex G Rice
12个回答

58

Leaflet存在一个已知的bug,根本问题是在打包过程中错误地引用了Leaflet图标的位置。

您可以通过验证此参数(在运行时)来验证这是否是您的问题:L.Icon.Default.prototype._getIconUrl()
正确的值应该是<some_directory>/leaflet/dist/images/
但是,如果您遇到此bug,则其值为:....K5CYII=")undefined

根据您使用的捆绑加载器(Vanila WebPack、Angular-Cli-WebPack的超集等),有不同的解决方案(解决方法)。

您可以在此处查看原始问题(以及根据您的捆绑加载器提供的不同解决方案):
https://github.com/Leaflet/Leaflet/issues/4968

如果您正在使用Angular-Cli,则此解决方案适用于您。 在设置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;

(此代码将更改损坏的标记URL,以从您的资产文件夹中获取有效的图像。)

并在您的 angular.json 中添加此代码(适用于 Angular 版本>=6.x),或在您的 angular-cli.json 中添加此代码(适用于 Angular 版本<=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/"
   }
] ...

(此代码将原始标记图像复制到/assets文件夹中,以便angular-cli可以加载它们)


虽然这可能是一个不错的Angular解释,但请注意OP从未提到使用JS捆绑器,他们不正确的路径不是数据URI,并且包括来自CDN的旧(0.x)Leaflet资产。 - ghybs
1
此修复适用于 vite (rollup) 和 vue3 构建。进入 node_modules/leaflet/dist/images 目录并获取这三个标记 png 文件。将它们放在 public/assets 中并进行构建。 - evendiagram
1
应该是被接受的答案。这个方法非常有效。谢谢! - Dinesh Shekhawat

24
import "leaflet/dist/images/marker-shadow.png";

如果使用webpack,只需导入图像


1
你确定这个回答提供了其他答案没有提供的新东西吗? - user3956566
3
这是一个好的一行代码修复吗? - user43284
1
这个解决方案很简单,适用于所有版本的Angular,非常棒。 - Juanma Font
太棒了。这解决了我的问题。我正在使用ngx-leaflet。谢谢! - Kelvin Cayman
我同意用户43284的观点 - 更少、更简单的代码在我看来更好、更干净。我可以确认这适用于Angular10。 - Philipp Doerner

23

以下代码对我有用:

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

1
我不清楚,你在哪里建议添加这段代码? - edjm
1
@NashGC,你能详细说明一下你是如何在Nuxt中实现的吗?我将leaflet本地导入到一个单独的组件中,而不是使用插件,你把这个代码块放在哪里? - Thorvald
1
@NashGC 这对我来说确实有效.. 但是我的标记在Safari浏览器中消失了。有什么办法可以解决吗?我使用的是leaflet 1.7.1,vue2-leaflet 2.7.0。 - Thorvald
1
使用 TypeScript 和 create-react-app 时出现错误 Property '_getIconUrl' does not exist on type 'Default' - Denis Molodtsov
1
谢谢,这对我也有用...但没有使用require函数。我只是添加了iconUrl: "/webroot/assets/js/leaflet/images/marker-icon.png"等等...当然要适应我的基本url。 - dlg_
显示剩余2条评论

8

在Angular 10中工作

对我来说,将PNG文件复制粘贴到assets文件夹并执行一个命令即可。

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

对于Leaflet 1.7和Angular 13

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


3
你可以更改leaflet css中图标的当前路径。因此,请勿使用url leaflet.css,下载文件并进行更改。
.leaflet-default-icon-path {
    background-image: url(https://unpkg.com/leaflet@1.7.1/dist/images/marker-icon.png);
}

谢谢!这正是我所需要的,而且是一个非常干净的解决方案。 - ItaiRoded

3

对于使用Vue3、TypeScript和vue-cli + Webpack 5的项目,我采用了一种解决方法,即在created()钩子中显式定义所需的图标。

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

然后我在mounted()钩子中创建我的地图:

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

注意:我将我的Mapbox存储在以VUE_APP_为前缀的环境变量中。


1

最便携的解决方案(适用于Rails 7)是:

在您的Javascript控制器中,在通常的Leaflet地图配置代码之前放置以下代码:

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]
...

1
我在使用Parcel作为打包工具与TypeScriptLeaflet v1.4(通过npm安装,以及其类型定义)结合时遇到了类似的问题,并通过Gil的答案以稍微不同的方式解决了它。
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],
});

在另一个文件中,我添加了必要的声明,以便TypeScript允许我导入png图片,例如:

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

此外需要注意的是,如果您使用了需要访问这些图片的Leaflet插件,您可能需要显式地分配它,例如Leaflet绘图插件也需要它。示例:
        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);
        });

1
答案是使用leaflet css更新您的index.html。
 <link rel="stylesheet" href="https://unpkg.com/leaflet@1.7.1/dist/leaflet.css"
   integrity="sha512-xodZBdNfgdfTC5n17Xddt2atTPudf34rdf45erfggrrddfdffdfEdf1HxjVMSvLVW9odfdfcdfdf3rfddfdferdfdfqdfgdfUKLsCC5gdfdfdCXdbqCmblFFAshOdfdfMAS6/kddderfdfeqq/sMZMZ19scRerdfggdfddffd4PsZChyertfgRRRSR7A=="
   crossorigin=""/>

将CSS放在头部。

0

@GilEpshtain的答案对我在Angular 9.1.9上起了作用。

额外说明-为了调试目的,您应该能够确认由@GilEpshtain指定的angular.json#assets更改已生效,方法是将其中一个图标URL直接放入浏览器中。例如,http://localhost:4200/assets/marker-icon.png应该会显示标记图标。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接