源码分析之Openlayers中MousePosition鼠标位置控件

news/2024/12/23 4:14:10/

概述

本文主要介绍 Openlayers 中的MousePosition鼠标位置控件,该控件会创建一个元素在页面的右上方用来实时显示鼠标光标的位置坐标。该控件在实际应用很有效,可以实时获取鼠标位置,但是一般控件元素都会自定义。

源码分析

MousePosition类是继承于Control类,关于Control类,可以参考这篇文章源码分析之Openlayers中的控件篇Control基类介绍。

MousePosition类实现如下:

class MousePosition extends Control {constructor(options) {options = options ? options : {};const element = document.createElement("div");element.className =options.className !== undefined ? options.className : "ol-mouse-position";super({element: element,render: options.render,target: options.target,});this.on;this.once;this.un;this.addChangeListener(PROJECTION, this.handleProjectionChanged_);if (options.coordinateFormat) {this.setCoordinateFormat(options.coordinateFormat);}if (options.projection) {this.setProjection(options.projection);}this.renderOnMouseOut_ = options.placeholder !== undefined;this.placeholder_ = this.renderOnMouseOut_ ? options.placeholder : " ";this.renderedHTML_ = element.innerHTML;this.mapProjection_ = null;this.transform_ = null;this.wrapX_ = options.wrapX === false ? false : true;}handleProjectionChanged_() {this.transform_ = null;}getCoordinateFormat() {return this.get(COORDINATE_FORMAT);}handleMouseOut(event) {this.updateHTML_(null);}getProjection() {return this.get(PROJECTION);}handleMouseMove(event) {const map = this.getMap();this.updateHTML_(map.getEventPixel(event));}setMap(map) {super.setMap(map);if (map) {const viewport = map.getViewport();this.listenerKeys.push(listen(viewport, EventType.POINTERMOVE, this.handleMouseMove, this));if (this.renderOnMouseOut_) {this.listenerKeys.push(listen(viewport, EventType.POINTEROUT, this.handleMouseOut, this));}this.updateHTML_(null);}}setCoordinateFormat(format) {this.set(COORDINATE_FORMAT, format);}setProjection(projection) {this.set(PROJECTION, getProjection(projection));}updateHTML_(pixel) {let html = this.placeholder_;if (pixel && this.mapProjection_) {if (!this.transform_) {const projection = this.getProjection();if (projection) {this.transform_ = getTransformFromProjections(this.mapProjection_,projection);} else {this.transform_ = identityTransform;}}const map = this.getMap();const coordinate = map.getCoordinateFromPixelInternal(pixel);if (coordinate) {const userProjection = getUserProjection();if (userProjection) {this.transform_ = getTransformFromProjections(this.mapProjection_,userProjection);}this.transform_(coordinate, coordinate);if (this.wrapX_) {const projection =userProjection || this.getProjection() || this.mapProjection_;wrapX(coordinate, projection);}const coordinateFormat = this.getCoordinateFormat();if (coordinateFormat) {html = coordinateFormat(coordinate);} else {html = coordinate.toString();}}}if (!this.renderedHTML_ || html !== this.renderedHTML_) {this.element.innerHTML = html;this.renderedHTML_ = html;}}render(mapEvent) {const frameState = mapEvent.frameState;if (!frameState) {this.mapProjection_ = null;} else {if (this.mapProjection_ != frameState.viewState.projection) {this.mapProjection_ = frameState.viewState.projection;this.transform_ = null;}}}
}

MousePosition类构造函数

MousePosition类构造函数接受一个参数对象options,该参数可以包含如下属性:

  • className:控件类名,默认为ol-mouse-position
  • render:自定义render方法,默认undefined
  • target:控件容器,默认undefined
  • coordinateFormat:坐标格式化,默认undefined
  • projection:分辨率,默认undefined
  • placeholder:提示填充字符
  • wrapX:是否水平方向重复延申

构造函数首先会先注册projection的监听事件this.handleProjectionChanged_,若该值发生变化,则将this.transform_null;然后判断,若options.coordinateFormat存在,则调用this.setCoordinateFormat方法;若options.projection存在,则调用this.setProjection方法;

MousePosition类中的方法

  • getCoordinateFormat方法:获取坐标格式化
  • getProjection方法:获取投影
  • handleMouseMove方法:接受一个参数event,该方法是鼠标在地图上移动时调用,会更新控件的内容坐标的值,getEventPixel就是根据参数event获取鼠标的位置以及viewport的某些属性,然后计算屏幕坐标
  • handleMouseOut方法:鼠标移除地图时调用
  • setCoordinateFormat方法:设置坐标格式化
  • setProjection方法:设置投影
  • updateHTML_方法:根据屏幕坐标获取地理坐标
  • render方法:在调用父类的setMap方法时会调用,主要用于设置this.mapProject_
  • setMap方法:sepMap方法会在Map类中调用,内部首先会调用父类的setMap方法,然后判断参数map是否存在,若存在,则注册viewport视口对象pointermove类型的监听,事件为this.handleMouseMove;若构造函数参数options.placeholder设置了,还会注册viewportpointeroutthis.handleMouseOut事件。

总结

本文主要介绍了 OpenlayersMousePosition鼠标位置控件的源码实现,核心就是注册viewport对象上pointermove类型的监听事件获取屏幕坐标,然后调用内部方法map.getCoordinateFromPixelInternal将屏幕坐标转化为实际的地理位置坐标。


http://www.ppmy.cn/news/1557368.html

相关文章

Spring Cloud 2023的新特性与改进

随着技术的不断演进,Java生态系统中的重要框架Spring也在不断更新和改进。2023年,Spring Cloud发布了多个新版本,带来了许多令人兴奋的新特性和改进。本文将深入探讨Spring Cloud 2023的新特性与改进,帮助开发者更好地理解和应用这…

使用Python实现基于AR的教育应用:打破课堂的墙壁

友友们好! 我的新专栏《Python进阶》正式启动啦!这是一个专为那些渴望提升Python技能的朋友们量身打造的专栏,无论你是已经有一定基础的开发者,还是希望深入挖掘Python潜力的爱好者,这里都将是你不可错过的宝藏。 在这个专栏中,你将会找到: ● 深入解析:每一篇文章都将…

备忘一个FDBatchMove数据转存的问题

使用FDBatchMove的SQL导入excel表到sql表,设置条件时一头雾水,函数不遵守sql的规则。 比如替换字段的TAB键值为空,replace(字段名,char(9),)竟然提示错误,百思不得其解。 试遍了几乎所有的函数,竟然是chr(9)。 这个…

vue-element-admin npm install 安装失败,tui-editor更名导致

导语: 本失败原因是由于tui-editor(富文本编辑器插件)更名造成的,现在已经更名为toast-ui/editor; 在一个是一直以为是我的git问题 报错代码:code 128 ..........,困扰了我好长时间&#xff…

相机主要调试参数

解析度测试 - 解释如何衡量摄像头捕捉细节的能力,确保图像清晰。锐度评估 - 教你如何判断图像边缘的清晰程度,以优化视觉效果。色散与色彩还原 - 分析色彩准确性,确保所见即所得的色彩一致性。白平衡校正 - 确保在各种光源下拍摄的照片颜色自…

基于单片机的病房呼叫系统设计

摘 要: 文章基于 51 系列的单片机设计的病房呼叫系统 。 在以 AT89C51 单片机为核心,以 74HC573 锁存器 、数码管显示模块、 矩阵按键模块等为辅组成的,按键分布在各个病床的床头,可以节约接口资源,当按下按键&a…

InnoDB 事务系统(一):认识事务

事务(Transaction)是数据库区别于文件系统的重要特性之一。在文件系统中,如果正在写文件,但是操作系统突然崩溃了,这个文件就很可能被破坏。当然有一些机制可以把文件恢复到某个时间点。不过,如果需要保证两…

【Prompt Engineering】4 推断

一、引言 推断任务涉及模型接收文本输入并执行分析,如提取标签、实体、理解情感等。传统机器学习需要大量工作:收集数据、训练模型、部署模型。LLM(大型语言模型)允许通过编写Prompt快速开始任务,无需大量工作。可以使…