LitElement 中的传单使用

问题描述

我目前正在使用 LitElement 组件开发一个应用程序。我想将 Leaflet 集成到其中,但在显示地图时遇到问题。我已经使用 npm 在我的项目中安装了 Leaflet,并创建了一个看起来像这样的类。

import {LitElement,html,css} from 'lit-element';
import './node_modules/leaflet/dist/leaflet';

class Map extends LitElement{

    static get styles() {
        return [css``];
    }

    constructor() {
        super();
    }
    connectedCallback() {
        super.connectedCallback();
        let map = L.map('mapid').setView([51.505,-0.09],13);

        let urlTemplate = 'http://{s}.tile.osm.org/{z}/{x}/{y}.png';
        map.addLayer(L.tileLayer(urlTemplate,{minZoom: 4}));
    }

    render() {
        return html`
            <link rel="stylesheet" href="./node_modules/leaflet/dist/leaflet.css">
            <div id="mapid" style="height: 100%"></div>

        `;
    }
}

customElements.define("my-map",Map);

运行我的应用程序导致以下错误:未捕获的类型错误:无法设置未定义的属性“L”。 我对如何使用 Leaflet 在我的 LitElement 应用程序中显示地图有点迷茫,如果能帮助我找到正确的方向,我会很感激。

解决方法

对于 Lit 项目,我们倾向于建议人们坚持使用 es-module 导入(尽管这是可选的)。所以,而不是从导入

import './node_modules/leaflet/dist/leaflet';

改为尝试:

import {map as createMap,tileLayer} from './node_modules/leaflet/dist/leaflet-src.esm.js';

这是因为它通常更适合 webcomponents 的模块化特性,并且 es 模块可以比调用 window.L

更有效地被捆绑器优化

接下来,针对此语法重写您的传单调用。例如

let map = L.map('mapid').setView([51.505,-0.09],13);

// should be
let map = createMap('mapid').setView([51.505,13);

// and
map.addLayer(L.tileLayer(urlTemplate,{minZoom: 4}))

// should be
map.addLayer(tileLayer(urlTemplate,{minZoom: 4}))

接下来,需要全局 ID 才能运行的库在使用 shadow DOM 作为 Shadow Roots 范围 DOM 并因此查询其根的 Web 组件中往往会出现问题。不过,幸运的是,leaflet's documentation 表明我们可以向它传递一个元素引用,而不仅仅是一个 id。这意味着我们需要像这样获取元素引用:

const mapEl = this.shadowRoot.querySelector('#mapid');

然后我们可以将其传递给 createMap 函数

const mapEl = this.shadowRoot.querySelector('#mapid');
let map = createMap(mapEl).setView([51.505,13);

最后我们会看到我们遇到了 mapElnull 的问题。这是因为 LitElement 在 firstUpdated lifecycle callback 之前不会呈现它的内容。这意味着我们必须将地图创建的位置从 connectedCallback 更改为 firstUpdated

这是您的代码的一个工作示例,其中我还添加了一些样式以使自定义元素具有一定的高度:https://jsbin.com/fumusodake/edit?html,output