MapLibre GL JS第49课:用表达式创建渐变线

📌 学习目标

  • 掌握用表达式创建渐变线的实现方法
  • 理解相关API的使用
  • 能够独立完成类似功能开发

🎯 核心概念

使用表达式创建渐变线条。

💻 完 整 代 码

代码示例

html 复制代码
<!DOCTYPE html>
<html lang="en">
<head>
    <title>Create a gradient line using an expression</title>
    <meta property="og:description" content="使用线渐变 paint 属性和表达式来可视化从线条起点开始距离。" />
    <meta property="og:created" content="2025-06-25" />
    <meta charset='utf-8'>
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <link rel='stylesheet' href='https://unpkg.com/maplibre-gl@5.24.0/dist/maplibre-gl.css' />
    <script src='https://unpkg.com/maplibre-gl@5.24.0/dist/maplibre-gl.js'></script>
    <style>
        body { margin: 0; padding: 0; }
        html, body, #map { height: 100%; }
    </style>
</head>
<body>
<div id="map"></div>

<script>
    const map = new maplibregl.Map({
        container: 'map',
        style: 'https://tiles.openfreemap.org/styles/bright',
        center: [-77.035, 38.875],
        zoom: 12
    });

    const geojson = {
        'type': 'FeatureCollection',
        'features': [
            {
                'type': 'Feature',
                'properties': {},
                'geometry': {
                    'coordinates': [
                        [-77.044211, 38.852924],
                        [-77.045659, 38.860158],
                        [-77.044232, 38.862326],
                        [-77.040879, 38.865454],
                        [-77.039936, 38.867698],
                        [-77.040338, 38.86943],
                        [-77.04264, 38.872528],
                        [-77.03696, 38.878424],
                        [-77.032309, 38.87937],
                        [-77.030056, 38.880945],
                        [-77.027645, 38.881779],
                        [-77.026946, 38.882645],
                        [-77.026942, 38.885502],
                        [-77.028054, 38.887449],
                        [-77.02806, 38.892088],
                        [-77.03364, 38.892108],
                        [-77.033643, 38.899926]
                    ],
                    'type': 'LineString'
                }
            }
        ]
    };

    map.on('load', () => {
        // 'line-gradient' 只能与GeoJSON源一起使用
        // 并且源必须将'lineMetrics'选项设置为true
        map.addSource('line', {
            type: 'geojson',
            lineMetrics: true,
            data: geojson
        });

        // 图层必须是'line'类型
        map.addLayer({
            type: 'line',
            source: 'line',
            id: 'line',
            paint: {
                'line-color': 'red',
                'line-width': 14,
                // 'line-gradient' 必须使用表达式指定
                // 使用特殊的'line-progress'属性
                'line-gradient': [
                    'interpolate',
                    ['linear'],
                    ['line-progress'],
                    0,
                    'blue',
                    0.1,
                    'royalblue',
                    0.3,
                    'cyan',
                    0.5,
                    'lime',
                    0.7,
                    'yellow',
                    1,
                    'red'
                ]
            },
            layout: {
                'line-cap': 'round',
                'line-join': 'round'
            }
        });
    });
</script>
</body>
</html>

🔍 代码解析

初始化地图

使用 new maplibregl.Map() 创建地图实例,配置基本参数。本示例的核心特色是展示如何使用 line-gradient 属性和表达式创建渐变线条。

关键配置项

  • container: 地图容器的 DOM 元素 ID
  • style : 使用 OpenStreetMap 亮色样式 https://tiles.openfreemap.org/styles/bright
  • center : 地图初始中心点 [-77.035, 38.875](美国华盛顿特区)
  • zoom: 初始缩放级别为 12,显示城市级别视图

GeoJSON 数据源配置

javascript 复制代码
map.addSource('line', {
    type: 'geojson',
    lineMetrics: true,  // 必须启用此选项才能使用 line-gradient
    data: geojson
});

line-gradient 表达式配置

javascript 复制代码
map.addLayer({
    type: 'line',
    source: 'line',
    id: 'line',
    paint: {
        'line-color': 'red',
        'line-width': 14,
        'line-gradient': [
            'interpolate',
            ['linear'],
            ['line-progress'],  // 特殊属性,表示线条进度(0-1)
            0, 'blue',          // 起点:蓝色
            0.1, 'royalblue',   // 10%:皇家蓝
            0.3, 'cyan',        // 30%:青色
            0.5, 'lime',        // 中点:石灰绿
            0.7, 'yellow',      // 70%:黄色
            1, 'red'            // 终点:红色
        ]
    },
    layout: {
        'line-cap': 'round',
        'line-join': 'round'
    }
});

⚙️ 参数说明

参数 类型 必填 默认值 说明
container string - 地图容器元素的 ID
style string/object - 地图样式 URL 或内联样式对象
center number, number [0, 0] 初始中心点坐标
zoom number 0 初始缩放级别

GeoJSON 数据源配置

参数 类型 必填 默认值 说明
lineMetrics boolean false 是否计算线条度量(启用后支持 line-progress

line-gradient 表达式

参数 类型 说明
line-progress expression 特殊属性,返回当前点在线条上的进度(0 到 1)
interpolate expression 插值表达式,用于创建平滑过渡
linear keyword 线性插值方式

🎨 效果说明

运行代码后,地图上会显示一条彩虹渐变的路径:

  • 起点 : 坐标 [-77.0442, 38.8529],颜色为蓝色
  • 终点 : 坐标 [-77.0336, 38.8999],颜色为红色
  • 渐变过渡: 蓝 → 皇家蓝 → 青 → 石灰绿 → 黄 → 红

渐变原理:

  1. line-progress 返回 0(起点)到 1(终点)的值
  2. interpolate 表达式根据 line-progress 值进行线性插值
  3. 在指定的进度点之间平滑过渡颜色

💡 常 见 问 题

Q1: line-gradient 为什么不生效?

A: 需要满足两个条件:

  1. 数据源必须是 GeoJSON 类型
  2. 数据源必须设置 lineMetrics: true

Q2: line-progress 是什么?

A: line-progress 是一个特殊的表达式属性,返回当前点在线条路径上的进度,值从 0(起点)到 1(终点)。

Q3: 可以使用其他插值方式吗?

A: 可以。除了 linear,还可以使用 exponentialcubic-bezier 插值方式。

Q4: 可以基于其他属性创建渐变吗?

A: line-gradient 只能使用 line-progress 作为输入,无法直接使用其他属性。

📝 练习任务

  1. 基础练习:修改渐变颜色和颜色节点位置
  2. 进阶挑战:创建双向渐变(从两端向中间渐变)
  3. 拓展思考:如何实现动态渐变动画?

🌟 最佳实践

  1. 性能优化 : lineMetrics 会增加内存使用,只在需要时启用
  2. 颜色数量: 适当控制颜色节点数量,过多会影响性能
  3. 线条宽度: 较宽的线条更能展示渐变效果
  4. 端点处理 : 使用 line-cap: round 使端点更平滑
  5. 兼容性检查: 确保目标浏览器支持此功能
  6. 备用方案 : 为不支持 line-gradient 的环境提供纯色备用

🔗 延伸阅读


本文是MapLibre GL JS实践课程系列的一部分,欢迎关注收藏

相关推荐
丷丩1 小时前
MapLibre GL JS第48课:用数据驱动属性样式化线
gis·地图·样式·mapbox·maplibre gl js
丷丩21 小时前
MapLibre GL JS第47课:添加动画图标
javascript·gis·动画·mapbox·maplibre
丷丩1 天前
错误处理与容错机制:GeoAI-UP的降级策略设计
架构·gis·容错设计
丷丩2 天前
MapLibre GL JS第46课:用Markers添加自定义图标
gis·可视化·mapbox·maplibre gl js
duansamve2 天前
使用Python生成带20W有色立方体的矢量瓦片 MVT用于在Mapbox中加载的完整流程
mapbox
王小王-1232 天前
基于机器学习的重庆主城的二手房价格影响因素分析与预测分析
机器学习·可视化·二手房价格预测·重庆二手房·成都二手房
是潮汕的灿灿展吖3 天前
程序员开发的地图的选择
地图
丷丩3 天前
MapLibre GL JS第44课:生成并添加缺失图标
前端·javascript·gis·mapblibre gl js
搞科研的小刘选手4 天前
【智能计算方向专题研讨会】第三届智能计算与数据分析国际学术会议(ICDA 2026)
大数据·算法·机器学习·数据挖掘·数据分析·可视化·计算