101.在 Vue 3 + OpenLayers 使用 declutter 避免文字标签重叠

news2025/3/14 19:17:39

1. 前言

在使用 OpenLayers 进行地图开发时,我们经常需要在地图上添加点、线、区域等图形,并给它们附加文字标签。但当地图上的标注较多时,文字标签可能会发生重叠,导致用户无法清晰地查看地图信息。

幸运的是,OpenLayers 提供了 declutter 选项,能够有效地避免文字标签的重叠问题。本文将介绍如何在 Vue 3 + OpenLayers 项目中使用 declutter,并通过示例代码展示如何添加带有时间标签的轨迹点。


2. declutter 介绍

declutter 是 OpenLayers VectorLayer(矢量图层)的一个属性,专门用于处理标注(文本或图标)的重叠问题。如果 declutter: true,OpenLayers 会自动调整标注的显示方式,避免文本或图标重叠在一起,提高地图的可读性。


3. 项目环境

  • Vue 3 + Composition API
  • OpenLayers
  • Vite

安装 OpenLayers

npm install ol

4. 代码实现

4.1 创建 Vue 组件

我们创建一个 Vue 组件,在其中初始化 OpenLayers 地图,并使用 declutter 避免标注重叠。

完整代码

<!--
 * @Author: 彭麒
 * @Date: 2025/3/13
 * @Email: 1062470959@qq.com
 * @Description: 此源码版权归吉檀迦俐所有,可供学习和借鉴或商用。
 -->
<template>
  <div class="container">
    <div class="w-full flex justify-center flex-wrap">
      <div class="font-bold text-[24px]">在Vue3中+OpenLayers使用declutter,避免文字标签重叠</div>
    </div>
    <div id="vue-openlayers"></div>
  </div>
</template>

<script setup>
import "ol/ol.css";
import {onMounted, ref} from "vue";
import {Map, View} from "ol";
import Tile from "ol/layer/Tile";
import OSM from "ol/source/OSM";
import VectorLayer from "ol/layer/Vector";
import VectorSource from "ol/source/Vector";
import Style from "ol/style/Style";
import Icon from "ol/style/Icon";
import Text from "ol/style/Text";
import Fill from "ol/style/Fill";
import Stroke from "ol/style/Stroke";
import Feature from "ol/Feature";
import {Point, LineString} from "ol/geom";
import startPoint from "@/assets/OpenLayers/startPoint.png";
import endPoint from "@/assets/OpenLayers/endPoint.png";
import pointImg from "@/assets/OpenLayers/point.png";
const map = ref(null);
const trackSource = new VectorSource();
const markersData = ref([
  [111.44, 24.18, 1604627953],
  [112.26, 24.48, 1604714353],
  [113.96, 24.65, 1604800753],
  [113.44, 24.78, 1604887153],
  [113.44, 24.98, 1605059953],
  [113.54, 25.68, 1605146353],
]);

const setTrackStyle = (text, img) => {
  return [
    new Style({
      image: new Icon({
        src: img,
        anchor: [0.5, 0.5],
        scale: 0.2,
      }),
      text: new Text({
        font: "12px sans-serif",
        maxAngle: 30,
        offsetY: 20,
        text: text,
        fill: new Fill({color: "#fff"}),
        backgroundFill: new Fill({color: "rgba(255,0,0,0.6)"}),
        backgroundStroke: new Stroke({color: "rgba(255,0,0,0.6)", width: 8}),
      }),
    }),
  ];
};

const showTrace = (data) => {
  const lineFeature = new Feature(new LineString(data));
  lineFeature.setStyle(new Style({stroke: new Stroke({color: "#00f", width: 2})}));
  trackSource.addFeature(lineFeature);

  const features = [];
  data.forEach((point, index) => {
    let img;
    if (index === 0) img = startPoint;
    else if (index === data.length - 1) img = endPoint;
    else img = pointImg

    const feature = new Feature({
      geometry: new Point([point[0], point[1]]),
      data: point,
    });

    const time = new Date(point[2] * 1000).toISOString().split("T")[0];
    feature.setStyle(setTrackStyle(`时间: ${time}`, img));
    features.push(feature);
  });

  trackSource.addFeatures(features);
};

const initMap = () => {
  const OSMlayer = new Tile({source: new OSM()});
  const trackLayer = new VectorLayer({
    source: trackSource,
    declutter: true, // 避免标签重叠
  });

  map.value = new Map({
    target: "vue-openlayers",
    layers: [OSMlayer, trackLayer],
    view: new View({
      center: [113.448, 23.986],
      zoom: 7,
      projection: "EPSG:4326",
    }),
  });
};

onMounted(() => {
  initMap();
  showTrace(markersData.value);
});

</script>

<style scoped>
.container {
  width: 840px;
  height: 590px;
  margin: 50px auto;
  border: 1px solid #42B983;
}

#vue-openlayers {
  width: 800px;
  height: 470px;
  margin: 0 auto;
  border: 1px solid #42B983;
  position: relative;
}
</style>

5. 代码解析

5.1 setTrackStyle 方法

该方法用于给轨迹点设置样式:

  • 图标:设置 Icon 作为点的标识
  • 文本:使用 Text 组件显示时间,并设置 fillbackgroundFill 等属性
  • 避免重叠:OpenLayers 通过 declutter 自动处理文字的布局,防止多个标注挤在一起

5.2 showTrace 方法

  • 绘制轨迹线路:使用 LineString 创建轨迹线
  • 添加轨迹点:遍历 markersData 数据,为每个点创建 Feature
  • 设置时间标签:使用 setTrackStyle 给每个点添加时间信息

5.3 initMap 方法

  • 创建地图实例
  • 添加 OSM 瓦片地图
  • 添加 VectorLayer(启用 declutter

6. declutter 前后效果对比

在未启用 declutter 时,多个标注可能会重叠在一起,影响阅读:

开启 declutter 后,OpenLayers 会自动优化文本排布,防止文字叠加:


7. 结论

本文介绍了如何在 Vue 3 + OpenLayers 项目中使用 declutter 避免文字标签重叠。declutter 是 OpenLayers 提供的强大功能,能够自动优化地图标注,使得地图数据更加清晰可读。希望本文对你有所帮助!

如果觉得有用,请点赞收藏支持!🚀
📢 关注我,获取更多前端 & GIS 开发技巧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2315034.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

uniapp移动端图片比较器组件,仿英伟达官网rtx光追图片比较器功能

组件下载地址&#xff1a;https://ext.dcloud.net.cn/plugin?id22609 已测试h5和微信小程序&#xff0c;理论支持全平台 亮点&#xff1a; 简单易用 使用js计算而不是resize属性&#xff0c;定制化程度更高 组件挂在后可播放指示线动画&#xff0c;提示用户可以拖拽比较图片…

深度学习与大模型-矩阵

矩阵其实在我们的生活中也有很多应用&#xff0c;只是我们没注意罢了。 1. 矩阵是什么&#xff1f; 简单来说&#xff0c;矩阵就是一个长方形的数字表格。比如你有一个2行3列的矩阵&#xff0c;可以写成这样&#xff1a; 这个矩阵有2行3列&#xff0c;每个数字都有一个位置&a…

搭建基于chatgpt的问答系统

一、语言模型&#xff0c;提问范式与 Token 1.语言模型 大语言模型&#xff08;LLM&#xff09;是通过预测下一个词的监督学习方式进行训练的&#xff0c;通过预测下一个词为训练目标的方法使得语言模型获得强大的语言生成能力。 a.基础语言模型 &#xff08;Base LLM&…

LuaJIT 学习(2)—— 使用 FFI 库的几个例子

文章目录 介绍Motivating Example: Calling External C Functions例子&#xff1a;Lua 中调用 C 函数 Motivating Example: Using C Data StructuresAccessing Standard System FunctionsAccessing the zlib Compression LibraryDefining Metamethods for a C Type例子&#xf…

解锁 AI 开发的无限可能:邀请您加入 coze-sharp 开源项目

大家好&#xff01;今天我要向大家介绍一个充满潜力的开源项目——coze-sharp&#xff01;这是一个基于 C# 开发的 Coze 客户端&#xff0c;旨在帮助开发者轻松接入 Coze AI 平台&#xff0c;打造智能应用。项目地址在这里&#xff1a;https://github.com/zhulige/coze-sharp&a…

全面解析与实用指南:如何有效解决ffmpeg.dll丢失问题并恢复软件正常运行

在使用多媒体处理软件或进行视频编辑时&#xff0c;你可能会遇到一个常见的问题——ffmpeg.dll文件丢失。这个错误不仅会中断你的工作流程&#xff0c;还可能导致软件无法正常运行。ffmpeg.dll是FFmpeg库中的一个关键动态链接库文件&#xff0c;负责处理视频和音频的编码、解码…

Python----计算机视觉处理(opencv:像素,RGB颜色,图像的存储,opencv安装,代码展示)

一、计算机眼中的图像 像素 像素是图像的基本单元&#xff0c;每个像素存储着图像的颜色、亮度和其他特征。一系列像素组合到一起就形成 了完整的图像&#xff0c;在计算机中&#xff0c;图像以像素的形式存在并采用二进制格式进行存储。根据图像的颜色不 同&#xff0c;每个像…

小米路由器SSH下安装DDNS-GO

文章目录 前言一、下载&#xff06;安装DDNS-GO二、配置ddns-go设置开机启动 前言 什么是DDNS&#xff1f; DDNS&#xff08;Dynamic Domain Name Server&#xff09;是动态域名服务的缩写。 目前路由器拨号上网获得的多半都是动态IP&#xff0c;DDNS可以将路由器变化的外网I…

go语言zero框架拉取内部平台开发的sdk报错的修复与实践

在开发过程中&#xff0c;我们可能会遇到由于认证问题无法拉取私有 SDK 的情况。这种情况常发生在使用 Go 语言以及 Zero 框架时&#xff0c;尤其是在连接到私有平台&#xff0c;如阿里云 Codeup 上托管的 Go SDK。如果你遇到这种错误&#xff0c;通常是因为 Go 没有适当的认证…

手机屏幕摔不显示了,如何用其他屏幕临时显示,用来导出资料或者清理手机

首先准备一个拓展坞 然后 插入一个外接的U盘 插入鼠标 插入有数字小键盘区的键盘 然后准备一根高清线&#xff0c;一端链接电脑显示器,一端插入拓展坞 把拓展坞的连接线&#xff0c;插入手机充电口&#xff08;可能会需要转接头&#xff09; 然后确保手机开机 按下键盘…

工业三防平板AORO-P300 Ultra,开创铁路检修与调度数字化新范式

在现代化铁路系统的庞大网络中&#xff0c;其设备维护与运营调度的精准性直接影响着运输效率和公共安全。在昼夜温差大、电磁环境复杂、震动粉尘交织的铁路作业场景中&#xff0c;AORO-P300 Ultra工业三防平板以高防护标准与智能化功能体系&#xff0c;开创了铁路行业移动端数字…

LInux基础--apache部署网站

httpd的安装 yum -y install httpdhttpd的使用 启动httpd systemctl enable --now httpd使用enable --now 进行系统设置时&#xff0c;会将该服务设置为开机自启并且同时开启服务 访问httpd 创建虚拟主机 基于域名 在一台主机上配置两个服务server1和server2&#xff0c;其…

Linux内核套接字以及分层模型

一、套接字通信 内核开发工程师将网络部分的头文件存储到一个专门的目录include/net中&#xff0c;而不是存储到标准位置include/linux。 计算机之间通信是一个非常复杂的问题&#xff1a; 如何建立物理连接&#xff1f;使用什么样的线缆&#xff1f;通信介质有那些限制和特殊…

Linux《基础开发工具(中)》

在之前的Linux《基础开发工具&#xff08;上&#xff09;》当中已经了解了Linux当中到的两大基础的开发工具yum与vim&#xff1b;了解了在Linux当中如何进行软件的下载以及实现的基本原理、知道了编辑器vim的基本使用方式&#xff0c;那么接下来在本篇当中将接下去继续来了解另…

使用1Panel一键搭建WordPress网站的详细教程(全)

嘿&#xff0c;各位想搭建自己网站的朋友们&#xff01;今天我要跟大家分享我用1Panel搭建WordPress网站的全过程。说实话&#xff0c;我之前对服务器运维一窍不通&#xff0c;但通过这次尝试&#xff0c;我发现原来建站可以这么简单&#xff01;下面是我的亲身经历和一些小技巧…

uni-app学习笔记——自定义模板

一、流程 1.这是一个硬性的流程&#xff0c;只要按照如此程序化就可以实现 二、步骤 1.第一步 2.第二步 3.第三步 4.每一次新建页面&#xff0c;都如第二步一样&#xff1b;可以选择自定义的模版&#xff08;vue3Setup——这是我自己的模版&#xff09;&#xff0c;第二步的…

数据结构——顺序表seqlist

前言&#xff1a;大家好&#x1f60d;&#xff0c;本文主要介绍了数据结构——顺序表部分的内容 目录 一、线性表的定义 二、线性表的基本操作 三.顺序表 1.定义 2. 存储结构 3. 特点 四 顺序表操作 4.1初始化 4.2 插入 4.2.1头插 4.2.2 尾插 4.2.3 按位置插 4.3 …

使用位运算如何找到数组中只出现一次的数?

题目链接&#xff1a;137. 只出现一次的数字 II - 力扣&#xff08;LeetCode&#xff09; 算法解析 位运算是用于二进制的运算符号。而对于多次出现的数字&#xff0c;其二进制都是一模一样的&#xff0c;这里是3次重复的出现是数字。由此我们可以想到&#xff0c;如果我们由低…

Linux笔记之通配符和正则表达式的区别

Linux笔记之通配符和正则表达式的区别 code review! 参考笔记 1.Linux笔记之通配符和正则表达式的区别 2.C++笔记之C语言中的换行符和转义符 文章目录 Linux笔记之通配符和正则表达式的区别1.通配符概念2.通配符和正则表达式的区别3.C++或C语言中有没有通配符?4.Linux Bash脚…

防汛应急包,快速响应,守护安全

根据中国水利部统计&#xff0c;自1949年以来&#xff0c;我国几乎每年都面临洪水威胁&#xff0c;其中20世纪90年代后洪涝灾害频率显著增加&#xff0c;仅1990-2009年间就发生超4000起较大灾害&#xff0c;直接经济损失近3万亿元&#xff0c;受灾人口达20亿人次。在2020年长江…