vue3使用腾讯地图选择地点

在这里插入图片描述

<template>
    <div>
        <el-dialog width="60%" :before-close="cancel" v-model="props.visible">
            <span>
                <el-autocomplete v-model="addressKeyword" placeholder="请输入地址,例如临沂人民广场" clearable
                    style="margin-bottom:20px;width:100%;" :fetch-suggestions="querySearch" @select="handleSelect">
                    <template #append>
                        <el-button icon="el-icon-search" @click="getAddressKeyword"></el-button>
                    </template>
                </el-autocomplete>
                <div id="container" style="width:100%;height:600px;"></div>
            </span>
            <template #footer>
                <span class="dialog-footer">
                    <el-button @click="cancel">取 消</el-button>
                    <el-button type="primary" @click="confirm">确 定</el-button>
                </span>
            </template>
        </el-dialog>
    </div>
</template>
  
<script setup>
import { ref, reactive, onMounted, nextTick } from "vue";
import md5 from 'js-md5';
import { jsonp } from 'vue-jsonp'

const markersArray = ref([]);
const restaurants = ref([]);
const map = ref(null);
const getAddress = ref(null);
const getAddCode = ref(null);
const addressKeyword = ref('');
const shopInfo = reactive({
    lng: '',
    lat: '',
    address: ''
});

const props = defineProps({
    visible: Boolean,
    lng: String,
    lat: String
})

onMounted(() => {
    nextTick(() => {
        init()
    })
})
var markerLayer
// 初始化地图
const init = () => {
    // 定义地图中心点坐标
    var center = new TMap.LatLng(35.05151, 118.34787)
    // 定义map变量,调用 TMap.Map() 构造函数创建地图
    map.value = new TMap.Map(document.getElementById('container'), {
        center: center, // 设置地图中心点坐标
        zoom: 13, // 设置地图缩放级别
        pitch: 0, // 设置俯仰角
        rotation: 0 // 设置地图旋转角度
    })
    // 获取点击后的地址
    map.value.on('click', function (event) {
        shopInfo.address = event.poi.name
        // 获取点击后的地图坐标并设置Marker
        shopInfo.lng = event.latLng.lng
        shopInfo.lat = event.latLng.lat
        if (markerLayer) {
            markerLayer.setGeometries([])
        }
        markerLayer = new TMap.MultiMarker({
            map: map.value,
            styles: {
                // 点标记样式
                marker: new TMap.MarkerStyle({
                    width: 20, // 样式宽
                    height: 30, // 样式高
                    anchor: { x: 10, y: 30 }, // 描点位置
                    src: 'https://mapapi.qq.com/web/lbs/javascriptGL/demo/img/markerDefault.png' // 标记路径
                })
            }
        })
        markerLayer.add({
            id: shopInfo.lat,
            position: new TMap.LatLng(shopInfo.lat, shopInfo.lng)
        })
    })
    if (props.lat && props.lng) {
        if (markerLayer) {
            markerLayer.setGeometries([])
        }
        markerLayer = new TMap.MultiMarker({
            map: map.value,
            styles: {
                // 点标记样式
                marker: new TMap.MarkerStyle({
                    width: 20, // 样式宽
                    height: 30, // 样式高
                    anchor: { x: 10, y: 30 }, // 描点位置
                    src: 'https://mapapi.qq.com/web/lbs/javascriptGL/demo/img/markerDefault.png' // 标记路径
                })
            }
        })
        markerLayer.updateGeometries({
            styleId: "marker",
            id: props.lat,
            position: new TMap.LatLng(props.lat, props.lng)
        })
        map.value.setCenter(new TMap.LatLng(props.lat, props.lng))
    }
}
const querySearch = (queryString, cb) => {
    var results = queryString ? restaurants.value.filter(createFilter(queryString)) : restaurants.value
    // 调用 callback 返回建议列表的数据
    cb(results)
}
const createFilter = (queryString) => {
    return (restaurant) => {
        return (restaurant.value.toLowerCase().indexOf(queryString.toLowerCase()) === 0)
    }
}
const handleSelect = (item) => {
    // 腾讯地图文档: https://lbs.qq.com/faq/serverFaq/webServiceKey
    // csdn参考:https://blog.csdn.net/tiankonlan/article/details/126363935
    let str = `/ws/geocoder/v1/?address=${item || addressKeyword.value}&callback=jsonpCallback&key=${import.meta.env.VITE_APP_TMapKey}&output=jsonp${import.meta.env.VITE_APP_TMapSig}`
    let sig = md5(str);
    sig = encodeURI(sig)

    let getData = {
        address: item || addressKeyword.value,
        callbackQuery: 'callback',
        callbackName: 'jsonpCallback',
        output: 'jsonp',
        key: import.meta.env.VITE_APP_TMapKey,
        sig: sig
    }

    jsonp('https://apis.map.qq.com/ws/geocoder/v1/', getData).then(response => {
        if (response.message === '查询无结果') {
            alert('查询无结果')
            return false
        }
        shopInfo.lng = response.result.location.lng
        shopInfo.lat = response.result.location.lat
        shopInfo.address = item
        if (markerLayer) {
            markerLayer.setGeometries([])
        }
        markerLayer = new TMap.MultiMarker({
            map: map.value,
            styles: {
                // 点标记样式
                marker: new TMap.MarkerStyle({
                    width: 20, // 样式宽
                    height: 30, // 样式高
                    anchor: { x: 10, y: 30 }, // 描点位置
                    src: 'https://mapapi.qq.com/web/lbs/javascriptGL/demo/img/markerDefault.png' // 标记路径
                })
            }
        })
        map.value.setCenter(new TMap.LatLng(shopInfo.lat, shopInfo.lng))
        markerLayer.updateGeometries({
            id: shopInfo.lat,
            position: new TMap.LatLng(shopInfo.lat, shopInfo.lng)
        })
    }).catch(function (error) {
        console.log(error)
    })
}
const getAddressKeyword = () => {
    handleSelect(addressKeyword.value)
}
const emit = defineEmits(['map-confirm', 'cancel'])
/***
 * 确认
 */
const confirm = () => {
    let data = JSON.parse(JSON.stringify(shopInfo))
    emit('map-confirm', data)
}
/***
 * 取消
 */
const cancel = () => {
    emit('cancel')
}
</script>
  
<style lang="scss" scoped>
.serachinput {
    width: 300px;
    box-sizing: border-box;
    padding: 9px;
    border: 1px solid #dddee1;
    line-height: 20px;
    font-size: 16px;
    height: 38px;
    color: #333;
    position: relative;
    border-radius: 4px;
    -webkit-box-shadow: #666 0px 0px 10px;
    -moz-box-shadow: #666 0px 0px 10px;
    box-shadow: #666 0px 0px 10px;
}

:deep(.el-dialog__header) {
    border-bottom: 0 !important;
}
</style>
  
  
<think>好的,我现在需要帮助用户在Vue3项目中集成腾讯地图API并实现地点搜索功能。首先,我需要回顾用户提供的引用内容,看看有没有相关的步骤或代码示例。 根据引用[1]和[2],用户需要在index.html中通过script标签引入腾讯地图的JS文件,并且替换成自己申请的key。引用[4]也提到类似的方法,使用script标签引入API库。这说明第一步肯定是申请key,然后在HTML文件中引入脚本。 接下来,用户想要实现地点搜索功能,可能需要使用腾讯地图的搜索组件或API。引用[1]提到地图选点组件,引用[4]则提到了地点查询和获取经纬度。所以,我需要考虑如何在Vue3组件中初始化地图,并添加搜索框和事件处理。 然后,我需要考虑如何将腾讯地图的API与Vue3结合使用。因为Vue3是响应式的,可能需要使用ref或reactive来管理状态,比如搜索关键字、搜索结果列表等。同时,要注意异步加载的问题,确保地图脚本加载完成后再初始化地图实例。 另外,引用[3]和[4]都提到在public/index.html中引入脚本,这说明用户可能需要检查引入的脚本是否正确,尤其是key的替换部分。常见的错误可能包括key未正确申请,或者脚本URL中的参数有误。 可能的步骤应该是:申请key→引入脚本→创建地图容器→初始化地图→添加搜索控件→处理搜索结果→展示结果。同时,用户可能需要处理地图加载的异步问题,使用生命周期钩子如onMounted来确保DOM加载完成后再执行地图初始化。 还需要考虑错误处理,比如地图加载失败的情况,或者用户输入无效时的提示。此外,样式方面,地图容器需要设置宽度和高度,否则可能无法显示。 总结下来,步骤应该是:申请腾讯地图key,在index.html引入JS文件,创建Vue组件,在组件中初始化地图,添加搜索功能,绑定事件监听,处理搜索结果的展示和选择。同时要注意异步加载和生命周期管理,确保地图API可用后再执行相关操作。</think>### 在 Vue3 中集成腾讯地图 API 并实现地点搜索的步骤 #### 1. **申请腾讯地图开发者 Key** - 访问[腾讯位置服务官网](https://lbs.qq.com/),注册账号并创建应用,获取 API Key。 #### 2. **在项目中引入腾讯地图 JS 文件** 在 `index.html` 的 `<head>` 标签内添加以下脚本(替换 `你的key` 为实际申请的 Key): ```html <script charset="utf-8" src="https://map.qq.com/api/gljs?v=1.exp&key=你的key"></script> ``` [^1][^2][^4] #### 3. **创建地图容器组件** 在 Vue3 组件中定义地图容器(需设置宽高): ```html <template> <div id="map-container" style="width: 100%; height: 400px;"></div> <input v-model="searchKeyword" @keyup.enter="searchLocation" /> <ul> <li v-for="item in searchResults" @click="selectLocation(item)"> {{ item.title }} </li> </ul> </template> ``` #### 4. **初始化地图与搜索功能** ```javascript import { ref, onMounted } from &#39;vue&#39;; export default { setup() { const mapInstance = ref(null); const searchKeyword = ref(&#39;&#39;); const searchResults = ref([]); onMounted(() => { // 异步加载地图 if (window.TMap) { initMap(); } else { window.initMap = initMap; // 避免重复加载 } }); const initMap = () => { mapInstance.value = new TMap.Map(&#39;map-container&#39;, { center: new TMap.LatLng(39.984104, 116.307503), zoom: 11 }); // 添加搜索控件 const searchService = new TMap.service.Search({ pageSize: 10, pageIndex: 1 }); }; const searchLocation = () => { if (!searchKeyword.value) return; new TMap.service.Search().search({ keyword: searchKeyword.value, success: (res) => { searchResults.value = res.data; }, fail: (err) => console.error(&#39;搜索失败:&#39;, err) }); }; const selectLocation = (item) => { mapInstance.value.setCenter(new TMap.LatLng(item.location.lat, item.location.lng)); }; return { searchKeyword, searchResults, searchLocation, selectLocation }; } }; ``` #### 5. **关键配置说明** - **地图中心点**:通过 `new TMap.LatLng(lat, lng)` 设置初始坐标 - **搜索服务**:使用 `TMap.service.Search()` 调用地点搜索 API - **事件绑定**:通过 `@keyup.enter` 触发搜索,`@click` 选择定位 #### 6. **常见问题处理** - **地图不显示**:检查容器宽高是否设置、Key 是否正确、网络是否正常 - **搜索无结果**:确认关键字符合腾讯地图数据规范(如包含城市名) - **API 未加载**:检查 `index.html` 脚本引入顺序和 Key 替换 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值