Angular封装高德地图组件实现输入框搜索,地图点击选地点

Angular封装高德地图组件实现输入框搜索,地图点击选地点(Angular17版本)

话不多说直接上代码

创建一个独立组件
在这里插入图片描述
html代码:

<div style="position: relative;">
  <input #searchInput nz-input placeholder="请输入地址"/>

  <div #mapContainer style="width: 100%;height: 350px;"></div>
</div>

样式less

@import "src/styles/themes/mixin";

.themeMixin({
  :host {
    position: relative;

    .toolbar {
      z-index: 9999;
      top: 8px;
      right: 8px;
      width: 200px;
    }
  }
});

ts代码:

import {
  AfterViewInit,
  Component,
  ElementRef,
  EventEmitter,
  Input,
  OnInit,
  Output, SimpleChanges,
  ViewChild
} from '@angular/core';
import {Observable, Observer, Subject} from "rxjs";
import {getPointsCenter} from "../../../util/gis";
import {NzInputDirective} from "ng-zorro-antd/input";
@Component({
  selector: 'app-site-pick',
  standalone: true,
  imports: [
    NzInputDirective
  ],
  templateUrl: './site-pick.component.html',
  styleUrl: './site-pick.component.less'
})
export class SitePickComponent implements OnInit, AfterViewInit{
  @ViewChild('mapContainer', {static: false}) mapContainer: ElementRef
  @ViewChild('searchInput', {static: false}) searchInput: ElementRef
  @Output("inputChange") inputChange = new EventEmitter<{ lonlat: any, siteName: any, adCode: any }>();
  @Input() lonlat: string;
  @Input() locationName: string;
  @Input("boundary") boundary: string
  map: any;
  overlays: any[] = [];
  searchAdCode = '010'
  defaultCenter = [116.397755, 39.903179]
  currentMarker: any; // 用于存储当前标记的引用
  drawMapEvent = new Subject()
  mapLoaded = false; // 标志位,判断地图是否已加载
  private mapLoadSubject = new Subject<void>(); // 用于触发地图加载完成事件
  ngOnInit(): void {
    this.drawMapEvent.subscribe(next => {
      this.currentPosition().subscribe(center => {
        this.addSearchPlugin();
      });
    });
  }

  addSearchPlugin(): void {
    const placeSearch = new window['AMap'].PlaceSearch({
      map: this.map,
      city: this.searchAdCode
    });
    const auto = new window['AMap'].Autocomplete({
      input: this.searchInput.nativeElement,
      city: this.searchAdCode
    });

    window['AMap'].Event.addListener(auto, "select", (e) => {
      placeSearch.search(e.poi.name, (status, result) => {
        if (status === 'complete' && result.info === 'OK' && result.poiList.pois.length > 0) {
          const poi = result.poiList.pois[0];
          placeSearch.getDetails(poi.id, (detailStatus, detailResult) => {
            if (detailStatus === 'complete' && detailResult.poiList.pois.length > 0) {
              const detailedPoi = detailResult.poiList.pois[0];
              const adCode = [
                detailedPoi.pname,
                detailedPoi.cityname,
                detailedPoi.adname
              ].filter(ac => ac);

              if (adCode.length === 2) {
                adCode.splice(1, 0, adCode[0]);
              }
              const adCodeStr = adCode.join(',');
              const location = detailedPoi.location;
              const siteName = detailedPoi.name;
              const lonlat = location.lng + ',' + location.lat;
              this.inputChange.emit({ lonlat: lonlat, siteName: siteName, adCode: adCodeStr });
            }
          });
        }
      });
    });
  }

  currentPosition(): Observable<any> {
    return new Observable<any>((observer: Observer<any>) => {
      new window['AMap'].Geolocation({
        enableHighAccuracy: false,
        timeout: 5000,
        offset: [10, 20],
        zoomToAccuracy: true,
        position: 'RB'
      }).getCityInfo((status, result) => {
        if (status == 'complete') {
          if (this.boundary && typeof this.boundary === 'string') {
            try {
              const center = getPointsCenter(this.boundary.split(';'));
              observer.next(center);
            } catch (e) {
              observer.next(this.defaultCenter);
            }
          } else {
            observer.next(result.position);
            this.searchAdCode = result.adcode;
            this.addSearchPlugin();
          }
        } else {
          console.error(result, 'Geolocation');
          observer.next(this.defaultCenter);
        }
      });
    });
  }

  ngAfterViewInit(): void {
    setTimeout(() => {
      this.map = new window['AMap'].Map(this.mapContainer.nativeElement, {
        resizeEnable: true,
        zoom: 14
      });

      this.map.on('click', (e) => {
        const lonlat = e.lnglat.getLng() + ',' + e.lnglat.getLat();
        this.resolveLocation(lonlat);
      });

      this.map.on('complete', () => {
        this.mapLoaded = true; // 地图加载完成
        this.mapLoadSubject.next(); // 触发地图加载完成事件
        this.initMarker();
      });

      this.drawMapEvent.next(null);
    }, 0);
  }

  initMarker(): void {
    if (!this.map) {
      console.error('地图尚未加载完成');
      return;
    }
    if (this.currentMarker) {
      this.map.remove(this.currentMarker);
    }
    if (this.lonlat) {
      const [lon, lat] = this.lonlat.split(',').map(Number);
      const position = new window['AMap'].LngLat(lon, lat);
      this.currentMarker = new window['AMap'].Marker({
        map: this.map,
        position: position
      });
    }
  }

  resolveLocation(lonlat: string): void {
    const [lng, lat] = lonlat.split(',').map(Number);
    const position = new window['AMap'].LngLat(lng, lat);
    const geocoder = new window['AMap'].Geocoder();

    geocoder.getAddress(position, (status, result) => {
      if (status === 'complete' && result.regeocode) {
        const address = result.regeocode.formattedAddress;
        const addressComponent = result.regeocode.addressComponent;
        const adCode = [addressComponent.province, addressComponent.city, addressComponent.district]
        .map(ac => ac && typeof ac === 'object' ? ac.adcode : ac)
        .filter(ac => ac);
        if (adCode.length === 2) {
          adCode.splice(1, 0, adCode[0]);
        }
        const adCodeStr = adCode.join(',');
        this.searchInput.nativeElement.value = address;
        this.inputChange.emit({ lonlat: lonlat, siteName: address, adCode: adCodeStr });
      } else {
        console.error('根据经纬度获取地址失败:', result);
      }
    });
    this.initMarker();
  }

  updateMapLocation(): Promise<void> {
    return new Promise((resolve, reject) => {
      if (!this.map) {
        console.error('地图尚未加载完成');
        return reject('地图尚未加载完成');
      }
      const [lon, lat] = this.lonlat.split(',').map(Number);
      const position = new window['AMap'].LngLat(lon, lat);
      if (this.currentMarker) {
        this.currentMarker.setPosition(position);
      } else {
        this.currentMarker = new window['AMap'].Marker({
          map: this.map,
          position: position
        });
      }
      this.map.setCenter([lon, lat]);
      resolve();
    });
  }

  getAddressFromLonLat(): void {
    const [lng, lat] = this.lonlat.split(',').map(Number);
    const geocoder = new window['AMap'].Geocoder();
    const position = new window['AMap'].LngLat(lng, lat);

    geocoder.getAddress(position, (status, result) => {
      if (status === 'complete' && result.regeocode) {
        const address = result.regeocode.formattedAddress;
        const addressComponent = result.regeocode.addressComponent;
        const adCode = [addressComponent.province, addressComponent.city, addressComponent.district]
        .map(ac => ac && typeof ac === 'object' ? ac.adcode : ac)
        .filter(ac => ac);
        if (adCode.length === 2) {
          adCode.splice(1, 0, adCode[0]);
        }
        const adCodeStr = adCode.join(',');
        this.searchInput.nativeElement.value = address;
        this.inputChange.emit({ lonlat: this.lonlat, siteName: address, adCode: adCodeStr });
      } else {
        console.error('根据经纬度获取地址失败:', result);
      }
    });
  }

  ngOnChanges(changes: SimpleChanges): void {
    if (changes['lonlat'] && this.lonlat) {
      if (this.mapLoaded) {
        this.updateMapLocation().then(() => {
          this.getAddressFromLonLat(); //根据 lonlat 获取地名
        });
      } else {
        this.mapLoadSubject.subscribe(() => { // 订阅地图加载完成事件
          this.updateMapLocation().then(() => {
            this.getAddressFromLonLat(); //根据 lonlat 获取地名
          });
        });
      }
    }
  }
}

如果 this.drawMapEvent.next(null); 报错改成 this.drawMapEvent.next();即可 因为我引入的 rxjs@7.5.7,

我这里对数据进行了处理:传出外部的数据类型:{ lonlat: any, siteName: any, adCode: any }

lonlat是经纬度,用",“逗号分隔
siteName是地点名
adCode是行政区code 用”,"分隔

使用

由于我做了表单的传值 可以直接在Form表单使用

      <nz-form-item>
        <nz-form-label [nzSpan]="4" nzFor="lonlat">场地地址</nz-form-label>
        <nz-form-control [nzSpan]="16" nzHasFeedback nzErrorTip="lonlat">
          <app-site-pick
              [lonlat]="form.get('lonlat').value"
              (inputChange)="inputChange($event)">
          </app-site-pick>
        </nz-form-control>
      </nz-form-item>
  /**
   * 地图input框选中返回lonlat+name
   * @param $event
   */
  inputChange($event: any) {
    this.form.get('lonlat').setValue($event.lonlat);
    this.form.get('address').setValue($event.siteName)
  }

这里我只需要传入lonlat即可回显地点
inputChange()方法可以监听改变的数据,然后数据格式就自己处理吧
当然也可以通过[(ngModel)]进行绑定

还有最关键的高德地图的key,securityJsCode(自己去官网上注册)

在全局上配置写上:app.config.ts

export const appConfig: ApplicationConfig = {
  providers: [
    importProvidersFrom(HttpClientModule, NzMessageModule, NzDrawerModule, NzModalModule, NzNotificationModule,NzSwitchModule),
    provideAnimations(),
    provideRouter(
        appRoutes,
        withPreloading(PreloadSelective),
        withComponentInputBinding() // 开启路由参数绑定到组件的输入属性,ng16新增特性
    ),
    // 初始化配置
    {
      provide: APP_INITIALIZER,
      useFactory: (bootstrap: BootstrapService) => () => {
        return bootstrap.init();
      },
      deps: [BootstrapService],
      multi: true,
    },
    // 国际化
    {
      provide: NZ_I18N,
      useFactory: (localId: string) => {
        switch (localId) {
          case 'en':
            return en_US;
          default:
            return zh_CN;
        }
      },
      deps: [LOCALE_ID]
    },
    {provide: HTTP_INTERCEPTORS, useClass: GlobalInterceptor, multi: true},
    {
      provide: AMAP_CONFIG, useValue: {
        securityJsCode: '9a2396b90169c48885aXXXXX6',
        key: 'de07643eaabaXXXXXX29284'
      }
    }
  ]
};

{
  provide: AMAP_CONFIG, useValue: {
    securityJsCode: '9a2396b9XXX2c0c3eaf6fdb6',
    key: 'de07643XXXX5a5629284'
  }
}

这个就是

然后在你的BootstrapService 中添加启动 loadAMap

import {Inject, Injectable} from '@angular/core';
import {registerLocaleData} from "@angular/common";
import zh from "@angular/common/locales/zh";
import {NzIconService} from "ng-zorro-antd/icon";
import {load} from "@amap/amap-jsapi-loader";
import {AMAP_CONFIG, AMAPConfig} from "../config";
import {SkinService} from "./skin.service";

@Injectable({providedIn: 'root'})
export class BootstrapService {

  constructor(private nzIconService: NzIconService,
              private skinService: SkinService,
              @Inject(AMAP_CONFIG) private amapConfig: AMAPConfig) {
  }

  init(): void {
    // 注册本地化语言包
    registerLocaleData(zh);
    // 注册icon
    // this.nzIconService.addIconLiteral('outline:clear', '')
    // 初始化设置主题
    this.skinService.loadTheme(this.skinService.localTheme()).then();
    // 加载地图
    this.loadAMap()
  }

  loadAMap(): void {
    window['_AMapSecurityConfig'] = {
      securityJsCode: this.amapConfig.securityJsCode, // 安全密钥
    };
    load({
      "key": this.amapConfig.key,
      "version": "2.0",   // 指定要加载的 JSAPI 的版本,缺省时默认为 1.4.15
      "plugins": [
        'AMap.Geolocation',
        'AMap.PolygonEditor',
        'AMap.PlaceSearch',
        'AMap.AutoComplete',
        'AMap.Polyline',
        'AMap.Geocoder'
      ], // 需要使用的的插件列表,如比例尺'AMap.Scale'等
      "AMapUI": {// 是否加载 AMapUI,缺省不加载
        "version": '1.1',// AMapUI 缺省 1.1
        "plugins": [
          'overlay/SimpleMarker'
        ],
      },
      "Loca": { // 是否加载 Loca, 缺省不加载
        "version": '2.0'  // Loca 版本,缺省 1.3.2
      },
    }).then((AMap) => {
      window['AMap'] = AMap
    }).catch(e => {
      console.log(e);
    })
  }
}

成品展示:

在这里插入图片描述

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MarianMT进行文本数据增强

B战学习视频&#xff1a;使用MarianMT进行文本数据增强 pip install transformers4.1.1 sentencepiece0.1.94 pip install mosestokenizer1.1.0from transformers import MarianMTModel,MarianTokenizer初始化模型&#xff0c;将英语翻译成罗曼语, target_model_nameHelsink…

力扣爆刷第148天之贪心算法五连刷(区间合并)

力扣爆刷第148天之贪心算法五连刷&#xff08;区间合并&#xff09; 文章目录 力扣爆刷第148天之贪心算法五连刷&#xff08;区间合并&#xff09;一、406. 根据身高重建队列二、452. 用最少数量的箭引爆气球三、435. 无重叠区间四、763. 划分字母区间五、56. 合并区间六、738.…

Android开机动画压缩包zip,自制开机动画(基于Android10.0.0-r41)

文章目录 Android开机动画压缩包zip&#xff0c;自制开机动画1.Android加载压缩包原理2.自制开机动画 Android开机动画压缩包zip&#xff0c;自制开机动画 1.Android加载压缩包原理 这里有个md文件我们看下 核心部分, 首先要创建一个文件叫做desc.txt&#xff0c;这是规定的…

Facebook开户|Facebook广告设计与测试优化

早上好家人们~今天Zoey给大家伙带来的是Facebook广告设计与测试优化&#xff0c;需要的家人们看过来啦&#xff01; 一、避免复杂用图和过多的文字 根据Facebook的数据显示&#xff0c;用户平均浏览一个贴文的时间在手机上仅花1.7秒、在电脑上则为2.5秒。因此&#xff0c;广告…

Dockershim 与 Containerd:两种容器运行时的故事

在不断发展的容器化世界中&#xff0c;两个关键组件经常被混淆&#xff1a;Dockershim 和 containerd。虽然它们在管理容器方面都发挥着重要作用&#xff0c;但它们的用途却截然不同。本文深入探讨了它们的功能&#xff0c;深入探讨了 Dockershim 和 containerd 之间的区别。 揭…

亚马逊运营黑科技,自养号测评技术打造产品权重

关于亚马逊运营&#xff0c;常规运营投入的成本过于高&#xff0c;而且广告投入效果也微乎其微&#xff0c;这也是为什么大多数卖家选择自养号测评的最主要原因。自养号测评技术&#xff0c;是一种用于提升产品权重和知名度的策略。以下是对该技术的详细解析&#xff1a; 一、…

CSS 常用的三种居中定位布局

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“”。 一、三种常用布局 1.子绝父相 margin 居中 注意&#xff1a;父级相对定位&#xff0c;子级绝对定位&#xff0c;并且子级margin-left&#xff0c;margin-top是负值&#xff0c;为宽度、高度的一半。 /** …

Java 中的 Map 集合:入门篇

在 Java 编程中&#xff0c;Map 是用于存储键值对。它提供了快速的查找和检索功能&#xff0c;是处理大量数据的理想选择。 本文将深入介绍 Java 中的 Map 集合&#xff0c;包括其基本概念、常见实现类、典型用法以及一些常见问题的解决方案。 1. Map 的基本概念 Map 是一种键…

电脑响度均衡是什么?它如何开启?

什么是响度均衡 响度均衡&#xff08;Loudness Equalization&#xff09;是一种音频处理技术&#xff0c;旨在平衡音频信号的响度水平&#xff0c;使得不同音源在播放时具有相似的响度感受。简单来说&#xff0c;它可以让用户在播放不同音轨或音频内容时&#xff0c;不需要频繁…

Echarts柱状图数据太多,自定义长度之后,自适应浏览器缩放

不知道是不是最优解&#xff0c;但是当前解决了我遇到的问题&#xff0c;如有更好的方法&#xff0c;希望看到这篇文章的同学可以不吝指导一番&#xff0c;非常感谢 1、问题描述&#xff1a; 因Ecahrts柱状图数据有时多有时少&#xff0c;所以在数据达到一定程度之后&#xff…

20240606在Toybrick的TB-RK3588开发板的Android12下确认HDMI的驱动

20240606在Toybrick的TB-RK3588开发板的Android12下确认HDMI的驱动 2024/6/6 9:48 【原文是在RK3328的Android7.1下写的。我将它升级成为RK3588的Android12了】 RK平台主要采用 FB 和 DRM 两种显示框架。与此相对应&#xff0c; HDMI 也有两套驱动。 FB&#xff1a; LINUX 3.10…

分表策略,你真的分对了?

垂直分表方案 表的记录并不多&#xff0c;但是字段却很长&#xff0c;表占用空间很大&#xff0c;检索表的时候需要执行大量的IO&#xff0c;严重降低了性能。这时需要把大的字段拆分到另一个表&#xff0c;并且该表与原表是一对一的关系。 为什么垂直拆分之后查询性能就变快…

Django里的Form组件

Form组件提供 自动生成HTML标签和半自动读取关联数据 (“半自动”是指还得需要自己手写输入数据进来)表单验证和错误提示 要想创建并使用该组件&#xff0c;操作步骤如下&#xff1a; 在 views.py 里创建类 # 在 views.py 文件里from django import formsclass AssetForm(fo…

HDFS文件块损坏处理方案

1、问题概述 flume采集文本文件存储到hdfs中hive的ods层目录,并在hive中通过msck repair table刷新元数据,加载文本文件。报错如下: 2、问题分析 文件块BP-531411289-172.31.57.12-1539657748238出现了未知异常,导致namenode不能获取该文件块的信息,该文件块是由flume采…

JeecgBoot/SpringBoot升级Nacos(2.0.4到2.2.3)启动报错

错误如下&#xff1a; 报这种错误基本就很头大了&#xff0c;是框架不兼容的问题&#xff0c;自己找很难找到解决方法。 解决方案是把SpringBoot框架版本调高。 修改前&#xff1a; <parent><groupId>org.springframework.boot</groupId><artifactId&g…

如何在 Mac 上玩 Windows 游戏:Parallels Desktop 玩转秘籍

引言 作为一名热爱游戏的 Mac 用户&#xff0c;你可能曾为 Mac 系统的有限游戏选择感到困扰。然而&#xff0c;通过 Parallels Desktop 虚拟机软件&#xff0c;你可以在 Mac 上轻松畅玩多款 Windows 游戏&#xff0c;尽情体验游戏的乐趣。 为什么选择 Parallels Desktop&…

刷机维修进阶教程-----魅族18系列 魅族21系列机型修复基带 改写参数等 通用新机型操作 步骤解析

在前面几期博文中解析了一些老款机型修复基带 修复各项参数以及改写参数的步骤解析。通常这些步骤可以用于各种问题导致的基带丢失 串码丢失以及有些参数修复或者一些特殊场合需要改写参数的需求。今天对于一些新机型操作以上需求做一些步骤解析,明白其操作原理。可以通用于一…

LeetCode 26删除有序数组中的重复项

去重题&#xff0c;双指针&#xff0c;&#xff0c;因为题干说原地删除&#xff0c;且nums其余元素不重要。一个cur记录当前不重复的数应该插在第几位了&#xff0c;for循环里的i相当于是第二个指针&#xff08;右指针&#xff09;&#xff0c;遍历数组来找不重复的元素 class …

C#WPF数字大屏项目实战12--动态获取设备数据

1、如何获取设备实时数据 现在大屏上的数据都是静态的数据或后台构造的来源数据&#xff0c;在实际项目中现场数据应该来自现场的实时数据&#xff0c;这些数据有些是来自现场设备的动态数据&#xff0c;有些是来自其他系统推送的&#xff0c;有些需要主动查询其他业务&#xf…

基于Arduino的简易磁悬浮装置原理图和源代码分享

磁悬浮装置原理 大家可能都玩过这种磁悬浮玩具&#xff0c;它们的工作原理与此类似。 首先&#xff0c;让我们了解一下这个原理&#xff0c;其实非常简单。它主要依赖于磁力对悬浮物体的控制。基本原理如下&#xff1a;在浮子的正下方放置一个霍尔传感器。当传感器检测到浮子向…