Bootstrap

在H5中使用腾讯地图,实现定位,距离计算,实时搜索,地址逆解析

1、创建应用

腾讯位置服务控制台中,创建应用,填入名称、描述、产品类型、已经域名白名单
在这里插入图片描述

一定要配置域名白名单,否则无法使用

2、下载微信sdk包

注意:这个sdk是为小程序设计的,所以在vue的h5中使用会有跨域问题,所以结合vue-jsonp对这个sdk做了一下修改,把sdk内使用的wx对象重写,替换了里面的request方法,使用vue-jsonp完成跨域,可以直接放入h5中使用

var vm = ''

/**
 * 这里是重写部分
 */
var wx = {
    request(obj){
        obj.data.output = 'jsonp'
        vm.$jsonp(obj.url,obj.data)
        .then(json => {
                if(json.status == 0){
                    obj.success(json)
                }else {
                    obj.fail(json)
                }
            })
            .catch(err => {
                obj.fail(err)
            })
    }
}

var ERROR_CONF = {
    KEY_ERR: 311,
    KEY_ERR_MSG: 'key格式错误',
    PARAM_ERR: 310,
    PARAM_ERR_MSG: '请求参数信息有误',
    SYSTEM_ERR: 600,
    SYSTEM_ERR_MSG: '系统错误',
    WX_ERR_CODE: 1000,
    WX_OK_CODE: 200
};
var BASE_URL = 'https://apis.map.qq.com/ws/';
var URL_SEARCH = BASE_URL + 'place/v1/search';
var URL_SUGGESTION = BASE_URL + 'place/v1/suggestion';
var URL_GET_GEOCODER = BASE_URL + 'geocoder/v1/';
var URL_CITY_LIST = BASE_URL + 'district/v1/list';
var URL_AREA_LIST = BASE_URL + 'district/v1/getchildren';
var URL_DISTANCE = BASE_URL + 'distance/v1/';
var URL_DIRECTION = BASE_URL + 'direction/v1/';
var MODE = {
  driving: 'driving',
  transit: 'transit'
};
var EARTH_RADIUS = 6378136.49;
var Utils = {
  /**
  * md5加密方法
  * 版权所有©2011 Sebastian Tschan,https://blueimp.net
  */
  safeAdd(x, y) {
    var lsw = (x & 0xffff) + (y & 0xffff);
    var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
    return (msw << 16) | (lsw & 0xffff);
  },
  bitRotateLeft(num, cnt) {
    return (num << cnt) | (num >>> (32 - cnt));
  },
  md5cmn(q, a, b, x, s, t) {
    return this.safeAdd(this.bitRotateLeft(this.safeAdd(this.safeAdd(a, q), this.safeAdd(x, t)), s), b);
  },
  md5ff(a, b, c, d, x, s, t) {
    return this.md5cmn((b & c) | (~b & d), a, b, x, s, t);
  },
  md5gg(a, b, c, d, x, s, t) {
    return this.md5cmn((b & d) | (c & ~d), a, b, x, s, t);
  },
  md5hh(a, b, c, d, x, s, t) {
    return this.md5cmn(b ^ c ^ d, a, b, x, s, t);
  },
  md5ii(a, b, c, d, x, s, t) {
    return this.md5cmn(c ^ (b | ~d), a, b, x, s, t);
  },
  binlMD5(x, len) {
    /* append padding */
    x[len >> 5] |= 0x80 << (len % 32);
    x[((len + 64) >>> 9 << 4) + 14] = len;

    var i;
    var olda;
    var oldb;
    var oldc;
    var oldd;
    var a = 1732584193;
    var b = -271733879;
    var c = -1732584194;
    var d = 271733878;

    for (i = 0; i < x.length; i += 16) {
      olda = a;
      oldb = b;
      oldc = c;
      oldd = d;

      a = this.md5ff(a, b, c, d, x[i], 7, -680876936);
      d = this.md5ff(d, a, b, c, x[i + 1], 12, -389564586);
      c = this.md5ff(c, d, a, b, x[i + 2], 17, 606105819);
      b = this.md5ff(b, c, d, a, x[i + 3], 22, -1044525330);
      a = this.md5ff(a, b, c, d, x[i + 4], 7, -176418897);
      d = this.md5ff(d, a, b, c, x[i + 5], 12, 1200080426);
      c = this.md5ff(c, d, a, b, x[i + 6], 17, -1473231341);
      b = this.md5ff(b, c, d, a, x[i + 7], 22, -45705983);
      a = this.md5ff(a, b, c, d, x[i + 8], 7, 1770035416);
      d = this.md5ff(d, a, b, c, x[i + 9], 12, -1958414417);
      c = this.md5ff(c, d, a, b, x[i + 10], 17, -42063);
      b = this.md5ff(b, c, d, a, x[i + 11], 22, -1990404162);
      a = this.md5ff(a, b, c, d, x[i + 12], 7, 1804603682);
      d = this.md5ff(d, a, b, c, x[i + 13], 12, -40341101);
      c = this.md5ff(c, d, a, b, x[i + 14], 17, -1502002290);
      b = this.md5ff(b, c, d, a, x[i + 15], 22, 1236535329);

      a = this.md5gg(a, b, c, d, x[i + 1], 5, -165796510);
      d = this.md5gg(d, a, b, c, x[i + 6], 9, -1069501632);
      c = this.md5gg(c, d, a, b, x[i + 11], 14, 643717713);
      b = this.md5gg(b, c, d, a, x[i], 20, -373897302);
      a = this.md5gg(a, b, c, d, x[i + 5], 5, -701558691);
      d = this.md5gg(d, a, b, c, x[i + 10], 9, 38016083);
      c = this.md5gg(c, d, a, b, x[i + 15], 14, -660478335);
      b = this.md5gg(b, c, d, a, x[i + 4], 20, -405537848);
      a = this.md5gg(a, b, c, d, x[i + 9], 5, 568446438);
      d = this.md5gg(d, a, b, c, x[i + 14], 9, -1019803690);
      c = this.md5gg(c, d, a, b, x[i + 3], 14, -187363961);
      b = this.md5gg(b, c, d, a, x[i + 8], 20, 1163531501);
      a = this.md5gg(a, b, c, d, x[i + 13], 5, -1444681467);
      d = this.md5gg(d, a, b, c, x[i + 2], 9, -51403784);
      c = this.md5gg(c, d, a, b, x[i + 7], 14, 1735328473);
      b = this.md5gg(b, c, d, a, x[i + 12], 20, -1926607734);

      a = this.md5hh(a, b, c, d, x[i + 5], 4, -378558);
      d = this.md5hh(d, a, b, c, x[i + 8], 11, -2022574463);
      c = this.md5hh(c, d, a, b, x[i + 11], 16, 1839030562);
      b = this.md5hh(b, c, d, a, x[i + 14], 23, -35309556);
      a = this.md5hh(a, b, c, d, x[i + 1], 4, -1530992060);
      d = this.md5hh(d, a, b, c, x[i + 4], 11, 1272893353);
      c = this.md5hh(c, d, a, b, x[i + 7], 16, -155497632);
      b = this.md5hh(b, c, d, a, x[i + 10], 23, -1094730640);
      a = this.md5hh(a, b, c, d, x[i + 13], 4, 681279174);
      d = this.md5hh(d, a, b, c, x[i], 11, -358537222);
      c = this.md5hh(c, d, a, b, x[i + 3], 16, -722521979);
      b = this.md5hh(b, c, d, a, x[i + 6], 23, 76029189);
      a = this.md5hh(a, b, c, d, x[i + 9], 4, -640364487);
      d = this.md5hh(d, a, b, c, x[i + 12], 11, -421815835);
      c = this.md5hh(c, d, a, b, x[i + 15], 16, 530742520);
      b = this.md5hh(b, c, d, a, x[i + 2], 23, -995338651);

      a = this.md5ii(a, b, c, d, x[i], 6, -198630844);
      d = this.md5ii(d, a, b, c, x[i + 7], 10, 1126891415);
      c = this.md5ii(c, d, a, b, x[i + 14], 15, -1416354905);
      b = this.md5ii(b, c, d, a, x[i + 5], 21, -57434055);
      a = this.md5ii(a, b, c, d, x[i + 12], 6, 1700485571);
      d = this.md5ii(d, a, b, c, x[i + 3], 10, -1894986606);
      c = this.md5ii(c, d, a, b, x[i + 10], 15, -1051523);
      b = this.md5ii(b, c, d, a, x[i + 1], 21, -2054922799);
      a = this.md5ii(a, b, c, d, x[i + 8], 6, 1873313359);
      d = this.md5ii(d, a, b, c, x[i + 15], 10, -30611744);
      c = this.md5ii(c, d, a, b, x[i + 6], 15, -1560198380);
      b = this.md5ii(b, c, d, a, x[i + 13], 21, 1309151649);
      a = this.md5ii(a, b, c, d, x[i + 4], 6, -145523070);
      d = this.md5ii(d, a, b, c, x[i + 11], 10, -1120210379);
      c = this.md5ii(c, d, a, b, x[i + 2], 15, 718787259);
      b = this.md5ii(b, c, d, a, x[i + 9], 21, -343485551);

      a = this.safeAdd(a, olda);
      b = this.safeAdd(b, oldb);
      c = this.safeAdd(c, oldc);
      d = this.safeAdd(d, oldd);
    }
    return [a, b, c, d];
  },
  binl2rstr(input) {
    var i;
    var output = '';
    var length32 = input.length * 32;
    for (i = 0; i < length32; i += 8) {
      output += String.fromCharCode((input[i >> 5] >>> (i % 32)) & 0xff);
    }
    return output;
  },
  rstr2binl(input) {
    var i;
    var output = [];
    output[(input.length >> 2) - 1] = undefined;
    for (i = 0; i < output.length; i += 1) {
      output[i] = 0;
    }
    var length8 = input.length * 8;
    for (i = 0; i < length8; i += 8) {
      output[i >> 5] |= (input.charCodeAt(i / 8) & 0xff) << (i % 32);
    }
    return output;
  },
  rstrMD5(s) {
    return this.binl2rstr(this.binlMD5(this.rstr2binl(s), s.length * 8));
  },
  rstrHMACMD5(key, data) {
    var i;
    var bkey = this.rstr2binl(key);
    var ipad = [];
    var opad = [];
    var hash;
    ipad[15] = opad[15] = undefined;
    if (bkey.length > 16) {
      bkey = this.binlMD5(bkey, key.length * 8);
    }
    for (i = 0; i < 16; i += 1) {
      ipad[i] = bkey[i] ^ 0x36363636;
      opad[i] = bkey[i] ^ 0x5c5c5c5c;
    }
    hash = this.binlMD5(ipad.concat(this.rstr2binl(data)), 512 + data.length * 8);
    return this.binl2rstr(this.binlMD5(opad.concat(hash), 512 + 128));
  },
  rstr2hex(input) {
    var hexTab = '0123456789abcdef';
    var output = '';
    var x;
    var i;
    for (i = 0; i < input.length; i += 1) {
      x = input.charCodeAt(i);
      output += hexTab.charAt((x >>> 4) & 0x0f) + hexTab.charAt(x & 0x0f);
    }
    return output;
  },
  str2rstrUTF8(input) {
    return unescape(encodeURIComponent(input));
  },
  rawMD5(s) {
    return this.rstrMD5(this.str2rstrUTF8(s));
  },
  hexMD5(s) {
    return this.rstr2hex(this.rawMD5(s));
  },
  rawHMACMD5(k, d) {
    return this.rstrHMACMD5(this.str2rstrUTF8(k), str2rstrUTF8(d));
  },
  hexHMACMD5(k, d) {
    return this.rstr2hex(this.rawHMACMD5(k, d));
  },

  md5(string, key, raw) {
    if (!key) {
      if (!raw) {
        return this.hexMD5(string);
      }
      return this.rawMD5(string);
    }
    if (!raw) {
      return this.hexHMACMD5(key, string);
    }
    return this.rawHMACMD5(key, string);
  },
  /**
   * 得到md5加密后的sig参数
   * @param {Object} requestParam 接口参数
   * @param {String} sk签名字符串
   * @param {String} featrue 方法名
   * @return 返回加密后的sig参数
   */
  getSig(requestParam, sk, feature, mode) {
    var sig = null;
    var requestArr = [];
    Object.keys(requestParam).sort().forEach(function(key){
      requestArr.push(key + '=' + requestParam[key]);
    });
    if (feature == 'search') {
      sig = '/ws/place/v1/search?' + requestArr.join('&') + sk;
    }
    if (feature == 'suggest') {
      sig = '/ws/place/v1/suggestion?' + requestArr.join('&') + sk;
    }
    if (feature == 'reverseGeocoder') {
      sig = '/ws/geocoder/v1/?' + requestArr.join('&') + sk;
    }
    if (feature == 'geocoder') {
      sig = '/ws/geocoder/v1/?' + requestArr.join('&') + sk;
    }
    if (feature == 'getCityList') {
      sig = '/ws/district/v1/list?' + requestArr.join('&') + sk;
    }
    if (feature == 'getDistrictByCityId') {
      sig = '/ws/district/v1/getchildren?' + requestArr.join('&') + sk;
    }
    if (feature == 'calculateDistance') {
      sig = '/ws/distance/v1/?' + requestArr.join('&') + sk;
    }
    if (feature == 'direction') {
      sig = '/ws/direction/v1/' + mode + '?' + requestArr.join('&') + sk;
    }
    sig = this.md5(sig);
    return sig;
  },
    /**
     * 得到终点query字符串
     * @param {Array|String} 检索数据
     */
    location2query(data) {
        if (typeof data == 'string') {
            return data;
        }
        var query = '';
        for (var i = 0; i < data.length; i++) {
            var d = data[i];
            if (!!query) {
                query += ';';
            }
            if (d.location) {
                query = query + d.location.lat + ',' + d.location.lng;
            }
            if (d.latitude && d.longitude) {
                query = query + d.latitude + ',' + d.longitude;
            }
        }
        return query;
    },

    /**
     * 计算角度
     */
    rad(d) {
      return d * Math.PI / 180.0;
    },  
    /**
     * 处理终点location数组
     * @return 返回终点数组
     */
    getEndLocation(location){
      var to = location.split(';');
      var endLocation = [];
      for (var i = 0; i < to.length; i++) {
        endLocation.push({
          lat: parseFloat(to[i].split(',')[0]),
          lng: parseFloat(to[i].split(',')[1])
        })
      }
      return endLocation;
    },

    /**
     * 计算两点间直线距离
     * @param a 表示纬度差
     * @param b 表示经度差
     * @return 返回的是距离,单位m
     */
    getDistance(latFrom, lngFrom, latTo, lngTo) {
      var radLatFrom = this.rad(latFrom);
      var radLatTo = this.rad(latTo);
      var a = radLatFrom - radLatTo;
      var b = this.rad(lngFrom) - this.rad(lngTo);
      var distance = 2 * Math.asin(Math.sqrt(Math.pow(Math.sin(a / 2), 2) + Math.cos(radLatFrom) * Math.cos(radLatTo) * Math.pow(Math.sin(b / 2), 2)));
      distance = distance * EARTH_RADIUS;
      distance = Math.round(distance * 10000) / 10000;
      return parseFloat(distance.toFixed(0));
    },
    /**
     * 使用微信接口进行定位
     */
    getWXLocation(success, fail, complete) {
        uni.getLocation({
            type: 'gcj02',
            success: success,
            fail: fail,
            complete: complete
        });
    },

    /**
     * 获取location参数
     */
    getLocationParam(location) {
        if (typeof location == 'string') {
            var locationArr = location.split(',');
            if (locationArr.length === 2) {
                location = {
                    latitude: location.split(',')[0],
                    longitude: location.split(',')[1]
                };
            } else {
                location = {};
            }
        }
        return location;
    },

    /**
     * 回调函数默认处理
     */
    polyfillParam(param) {
        param.success = param.success || function () { };
        param.fail = param.fail || function () { };
        param.complete = param.complete || function () { };
    },

    /**
     * 验证param对应的key值是否为空
     * 
     * @param {Object} param 接口参数
     * @param {String} key 对应参数的key
     */
    checkParamKeyEmpty(param, key) {
        if (!param[key]) {
            var errconf = this.buildErrorConfig(ERROR_CONF.PARAM_ERR, ERROR_CONF.PARAM_ERR_MSG + key +'参数格式有误');
            param.fail(errconf);
            param.complete(errconf);
            return true;
        }
        return false;
    },

    /**
     * 验证参数中是否存在检索词keyword
     * 
     * @param {Object} param 接口参数
     */
    checkKeyword(param){
        return !this.checkParamKeyEmpty(param, 'keyword');
    },

    /**
     * 验证location值
     * 
     * @param {Object} param 接口参数
     */
    checkLocation(param) {
        var location = this.getLocationParam(param.location);
        if (!location || !location.latitude || !location.longitude) {
            var errconf = this.buildErrorConfig(ERROR_CONF.PARAM_ERR, ERROR_CONF.PARAM_ERR_MSG + ' location参数格式有误');
            param.fail(errconf);
            param.complete(errconf);
            return false;
        }
        return true;
    },

    /**
     * 构造错误数据结构
     * @param {Number} errCode 错误码
     * @param {Number} errMsg 错误描述
     */
    buildErrorConfig(errCode, errMsg) {
        return {
            status: errCode,
            message: errMsg
        };
    },

    /**
     * 
     * 数据处理函数
     * 根据传入参数不同处理不同数据
     * @param {String} feature 功能名称
     * search 地点搜索
     * suggest关键词提示
     * reverseGeocoder逆地址解析
     * geocoder地址解析
     * getCityList获取城市列表:父集
     * getDistrictByCityId获取区县列表:子集
     * calculateDistance距离计算
     * @param {Object} param 接口参数
     * @param {Object} data 数据
     */
    handleData(param,data,feature){
      if (feature == 'search') {
        var searchResult = data.data;
        var searchSimplify = [];
        for (var i = 0; i < searchResult.length; i++) {
          searchSimplify.push({
            id: searchResult[i].id || null,
            title: searchResult[i].title || null,
            latitude: searchResult[i].location && searchResult[i].location.lat || null,
            longitude: searchResult[i].location && searchResult[i].location.lng || null,
            address: searchResult[i].address || null,
            category: searchResult[i].category || null,
            tel: searchResult[i].tel || null,
            adcode: searchResult[i].ad_info && searchResult[i].ad_info.adcode || null,
            city: searchResult[i].ad_info && searchResult[i].ad_info.city || null,
            district: searchResult[i].ad_info && searchResult[i].ad_info.district || null,
            province: searchResult[i].ad_info && searchResult[i].ad_info.province || null
          })
        }
        param.success(data, {
          searchResult: searchResult,
          searchSimplify: searchSimplify
        })
      } else if (feature == 'suggest') {
        var suggestResult = data.data;
        var suggestSimplify = [];
        for (var i = 0; i < suggestResult.length; i++) {
          suggestSimplify.push({
            adcode: suggestResult[i].adcode || null,
            address: suggestResult[i].address || null,
            category: suggestResult[i].category || null,
            city: suggestResult[i].city || null,
            district: suggestResult[i].district || null,
            id: suggestResult[i].id || null,
            latitude: suggestResult[i].location && suggestResult[i].location.lat || null,
            longitude: suggestResult[i].location && suggestResult[i].location.lng || null,
            province: suggestResult[i].province || null,
            title: suggestResult[i].title || null,
            type: suggestResult[i].type || null
          })
        }
        param.success(data, {
          suggestResult: suggestResult,
          suggestSimplify: suggestSimplify
          })
      } else if (feature == 'reverseGeocoder') {
        var reverseGeocoderResult = data.result;
        var reverseGeocoderSimplify = {
          address: reverseGeocoderResult.address || null,
          latitude: reverseGeocoderResult.location && reverseGeocoderResult.location.lat || null,
          longitude: reverseGeocoderResult.location && reverseGeocoderResult.location.lng || null,
          adcode: reverseGeocoderResult.ad_info && reverseGeocoderResult.ad_info.adcode || null,
          city: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.city || null,
          district: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.district || null,
          nation: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.nation || null,
          province: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.province || null,
          street: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.street || null,
          street_number: reverseGeocoderResult.address_component && reverseGeocoderResult.address_component.street_number || null,
          recommend: reverseGeocoderResult.formatted_addresses && reverseGeocoderResult.formatted_addresses.recommend || null,
          rough: reverseGeocoderResult.formatted_addresses && reverseGeocoderResult.formatted_addresses.rough || null
        };
        if (reverseGeocoderResult.pois) {//判断是否返回周边poi
          var pois = reverseGeocoderResult.pois;
          var poisSimplify = [];
          for (var i = 0;i < pois.length;i++) {
            poisSimplify.push({
              id: pois[i].id || null,
              title: pois[i].title || null,
              latitude: pois[i].location && pois[i].location.lat || null,
              longitude: pois[i].location && pois[i].location.lng || null,
              address: pois[i].address || null,
              category: pois[i].category || null,
              adcode: pois[i].ad_info && pois[i].ad_info.adcode || null,
              city: pois[i].ad_info && pois[i].ad_info.city || null,
              district: pois[i].ad_info && pois[i].ad_info.district || null,
              province: pois[i].ad_info && pois[i].ad_info.province || null
            })
          }
          param.success(data,{
            reverseGeocoderResult: reverseGeocoderResult,
            reverseGeocoderSimplify: reverseGeocoderSimplify,
            pois: pois,
            poisSimplify: poisSimplify
          })
        } else {
          param.success(data, {
            reverseGeocoderResult: reverseGeocoderResult,
            reverseGeocoderSimplify: reverseGeocoderSimplify
          })
        }
      } else if (feature == 'geocoder') {
        var geocoderResult = data.result;
        var geocoderSimplify = {
          title: geocoderResult.title || null,
          latitude: geocoderResult.location && geocoderResult.location.lat || null,
          longitude: geocoderResult.location && geocoderResult.location.lng || null,
          adcode: geocoderResult.ad_info && geocoderResult.ad_info.adcode || null,
          province: geocoderResult.address_components && geocoderResult.address_components.province || null,
          city: geocoderResult.address_components && geocoderResult.address_components.city || null,
          district: geocoderResult.address_components && geocoderResult.address_components.district || null,
          street: geocoderResult.address_components && geocoderResult.address_components.street || null,
          street_number: geocoderResult.address_components && geocoderResult.address_components.street_number || null,
          level: geocoderResult.level || null
        };
        param.success(data,{
          geocoderResult: geocoderResult,
          geocoderSimplify: geocoderSimplify
        });
      } else if (feature == 'getCityList') {
        var provinceResult = data.result[0];
        var cityResult = data.result[1];
        var districtResult = data.result[2];
        param.success(data,{
          provinceResult: provinceResult,
          cityResult: cityResult,
          districtResult: districtResult
        });
      } else if (feature == 'getDistrictByCityId') {
        var districtByCity = data.result[0];
        param.success(data, districtByCity);
      } else if (feature == 'calculateDistance') {
        var calculateDistanceResult = data.result.elements;  
        var distance = [];
        for (var i = 0; i < calculateDistanceResult.length; i++){
          distance.push(calculateDistanceResult[i].distance);
        }   
        param.success(data, {
          calculateDistanceResult: calculateDistanceResult,
          distance: distance
          });
      } else if (feature == 'direction') {
        var direction = data.result.routes;
        param.success(data,direction);
      } else {
        param.success(data);
      }
    },

    /**
     * 构造微信请求参数,公共属性处理
     * 
     * @param {Object} param 接口参数
     * @param {Object} param 配置项
     * @param {String} feature 方法名
     */
    buildWxRequestConfig(param, options, feature) {
        var that = this;
        options.header = { "content-type": "application/json" };
        options.method = 'GET';
        options.success = function (res) {
            var data = res;
            if (data.status === 0) {
              that.handleData(param, data, feature);
            } else {
                param.fail(data);
            }
        };
        options.fail = function (res) {
            res.statusCode = ERROR_CONF.WX_ERR_CODE;
            param.fail(that.buildErrorConfig(ERROR_CONF.WX_ERR_CODE, res.errMsg));
        };
        options.complete = function (res) {
            var statusCode = +res.statusCode;
            switch(statusCode) {
                case ERROR_CONF.WX_ERR_CODE: {
                    param.complete(that.buildErrorConfig(ERROR_CONF.WX_ERR_CODE, res.errMsg));
                    break;
                }
                case ERROR_CONF.WX_OK_CODE: {
                    var data = res.data;
                    if (data.status === 0) {
                        param.complete(data);
                    } else {
                        param.complete(that.buildErrorConfig(data.status, data.message));
                    }
                    break;
                }
                default:{
                    param.complete(that.buildErrorConfig(ERROR_CONF.SYSTEM_ERR, ERROR_CONF.SYSTEM_ERR_MSG));
                }

            }
        };
        return options;
    },

    /**
     * 处理用户参数是否传入坐标进行不同的处理
     */
    locationProcess(param, locationsuccess, locationfail, locationcomplete) {
        var that = this;
        locationfail = locationfail || function (res) {
            res.statusCode = ERROR_CONF.WX_ERR_CODE;
            param.fail(that.buildErrorConfig(ERROR_CONF.WX_ERR_CODE, res.errMsg));
        };
        locationcomplete = locationcomplete || function (res) {
            if (res.statusCode == ERROR_CONF.WX_ERR_CODE) {
                param.complete(that.buildErrorConfig(ERROR_CONF.WX_ERR_CODE, res.errMsg));
            }
        };
        if (!param.location) {
            that.getWXLocation(locationsuccess, locationfail, locationcomplete);
        } else if (that.checkLocation(param)) {
            var location = Utils.getLocationParam(param.location);
            locationsuccess(location);
        }
    }
};


class QQMapWX {

    /**
     * 构造函数
     * 
     * @param {Object} options 接口参数,key 为必选参数
     */
    constructor(options) {
        if (!options.key) {
            throw Error('key值不能为空');
        }
		// console.log('构造函数111',options)
        vm = options.vm
        this.key = options.key;
    };

    /**
     * POI周边检索
     *
     * @param {Object} options 接口参数对象
     * 
     * 参数对象结构可以参考
     * @see http://lbs.qq.com/webservice_v1/guide-search.html
     */
    search(options) {
        var that = this;
        options = options || {};

        Utils.polyfillParam(options);

        if (!Utils.checkKeyword(options)) {
            return;
        }

        var requestParam = {
            keyword: options.keyword,
            orderby: options.orderby || '_distance',
            page_size: options.page_size || 10,
            page_index: options.page_index || 1,
            output: 'json',
            key: that.key
        };

        if (options.address_format) {
            requestParam.address_format = options.address_format;
        }

        if (options.filter) {
            requestParam.filter = options.filter;
        }

        var distance = options.distance || "1000";
        var auto_extend = options.auto_extend || 1;
        var region = null;
        var rectangle = null;

        //判断城市限定参数
        if (options.region) {
          region = options.region;
        }

        //矩形限定坐标(暂时只支持字符串格式)
        if (options.rectangle) {
          rectangle = options.rectangle;
        }

        var locationsuccess = function (result) {        
          if (region && !rectangle) {
            //城市限定参数拼接
            requestParam.boundary = "region(" + region + "," + auto_extend + "," + result.latitude + "," + result.longitude + ")";
            if (options.sig) {
              requestParam.sig = Utils.getSig(requestParam, options.sig, 'search');
            }
          } else if (rectangle && !region) {
            //矩形搜索
            requestParam.boundary = "rectangle(" + rectangle + ")";
            if (options.sig) {
              requestParam.sig = Utils.getSig(requestParam, options.sig, 'search');
            }
            } else {
              requestParam.boundary = "nearby(" + result.latitude + "," + result.longitude + "," + distance + "," + auto_extend + ")";
            if (options.sig) {
              requestParam.sig = Utils.getSig(requestParam, options.sig, 'search');
            }
            }            
            wx.request(Utils.buildWxRequestConfig(options, {
                url: URL_SEARCH,
                data: requestParam
            }, 'search'));
        };
        Utils.locationProcess(options, locationsuccess);
    };

    /**
     * sug模糊检索
     *
     * @param {Object} options 接口参数对象
     * 
     * 参数对象结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-suggestion.html
     */
    getSuggestion(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);

        if (!Utils.checkKeyword(options)) {
            return;
        }

        var requestParam = {
            keyword: options.keyword,
            region: options.region || '全国',
            region_fix: options.region_fix || 0,
            policy: options.policy || 0,
            page_size: options.page_size || 10,//控制显示条数
            page_index: options.page_index || 1,//控制页数
            get_subpois : options.get_subpois || 0,//返回子地点
            output: 'json',
            key: that.key
        };
        //长地址
        if (options.address_format) {
          requestParam.address_format = options.address_format;
        }
        //过滤
        if (options.filter) {
          requestParam.filter = options.filter;
        }
        //排序
        if (options.location) {
          var locationsuccess = function (result) {
            requestParam.location = result.latitude + ',' + result.longitude;
            if (options.sig) {
              requestParam.sig = Utils.getSig(requestParam, options.sig, 'suggest');
            }
            wx.request(Utils.buildWxRequestConfig(options, {
              url: URL_SUGGESTION,
              data: requestParam
            }, "suggest"));      
          };
          Utils.locationProcess(options, locationsuccess);
        } else {
          if (options.sig) {
            requestParam.sig = Utils.getSig(requestParam, options.sig, 'suggest');
          }
          wx.request(Utils.buildWxRequestConfig(options, {
            url: URL_SUGGESTION,
            data: requestParam
          }, "suggest"));      
        }        
    };

    /**
     * 逆地址解析
     *
     * @param {Object} options 接口参数对象
     * 
     * 请求参数结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-gcoder.html
     */
    reverseGeocoder(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);
        var requestParam = {
            coord_type: options.coord_type || 5,
            get_poi: options.get_poi || 0,
            output: 'json',
            key: that.key
        };
        if (options.poi_options) {
            requestParam.poi_options = options.poi_options
        }

        var locationsuccess = function (result) {
            requestParam.location = result.latitude + ',' + result.longitude;
          if (options.sig) {
            requestParam.sig = Utils.getSig(requestParam, options.sig, 'reverseGeocoder');
          }
            wx.request(Utils.buildWxRequestConfig(options, {
                url: URL_GET_GEOCODER,
                data: requestParam
            }, 'reverseGeocoder'));
        };
        Utils.locationProcess(options, locationsuccess);
    };

    /**
     * 地址解析
     *
     * @param {Object} options 接口参数对象
     * 
     * 请求参数结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-geocoder.html
     */
    geocoder(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);

        if (Utils.checkParamKeyEmpty(options, 'address')) {
            return;
        }

        var requestParam = {
            address: options.address,
            output: 'json',
            key: that.key
        };

        //城市限定
        if (options.region) {
          requestParam.region = options.region;
        }

        if (options.sig) {
          requestParam.sig = Utils.getSig(requestParam, options.sig, 'geocoder');
        }

        wx.request(Utils.buildWxRequestConfig(options, {
            url: URL_GET_GEOCODER,
            data: requestParam
        },'geocoder'));
    };


    /**
     * 获取城市列表
     *
     * @param {Object} options 接口参数对象
     * 
     * 请求参数结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-region.html
     */
    getCityList(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);
        var requestParam = {
            output: 'json',
            key: that.key
        };

        if (options.sig) {
          requestParam.sig = Utils.getSig(requestParam, options.sig, 'getCityList');
        }

        wx.request(Utils.buildWxRequestConfig(options, {
            url: URL_CITY_LIST,
            data: requestParam
        },'getCityList'));
    };

    /**
     * 获取对应城市ID的区县列表
     *
     * @param {Object} options 接口参数对象
     * 
     * 请求参数结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-region.html
     */
    getDistrictByCityId(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);

        if (Utils.checkParamKeyEmpty(options, 'id')) {
            return;
        }

        var requestParam = {
            id: options.id || '',
            output: 'json',
            key: that.key
        };

        if (options.sig) {
          requestParam.sig = Utils.getSig(requestParam, options.sig, 'getDistrictByCityId');
        }

        wx.request(Utils.buildWxRequestConfig(options, {
            url: URL_AREA_LIST,
            data: requestParam
        },'getDistrictByCityId'));
    };

    /**
     * 用于单起点到多终点的路线距离(非直线距离)计算:
     * 支持两种距离计算方式:步行和驾车。
     * 起点到终点最大限制直线距离10公里。
     *
     * 新增直线距离计算。
     * 
     * @param {Object} options 接口参数对象
     * 
     * 请求参数结构可以参考
     * http://lbs.qq.com/webservice_v1/guide-distance.html
     */
    calculateDistance(options) {
        var that = this;
        options = options || {};
        Utils.polyfillParam(options);

        if (Utils.checkParamKeyEmpty(options, 'to')) {
            return;
        }

        var requestParam = {
            mode: options.mode || 'walking',
            to: Utils.location2query(options.to),
            output: 'json',
            key: that.key
        };

        if (options.from) {
          options.location = options.from;
        }

        //计算直线距离
        if(requestParam.mode == 'straight'){        
          var locationsuccess = function (result) {
            var locationTo = Utils.getEndLocation(requestParam.to);//处理终点坐标
            var data = {
              message:"query ok",
              result:{
                elements:[]
              },
              status:0
            };
            for (var i = 0; i < locationTo.length; i++) {
              data.result.elements.push({//将坐标存入
                distance: Utils.getDistance(result.latitude, result.longitude, locationTo[i].lat, locationTo[i].lng),
                duration:0,
                from:{
                  lat: result.latitude,
                  lng:result.longitude
                },
                to:{
                  lat: locationTo[i].lat,
                  lng: locationTo[i].lng
                }
              });            
            }
            var calculateResult = data.result.elements;
            var distanceResult = [];
            for (var i = 0; i < calculateResult.length; i++) {
              distanceResult.push(calculateResult[i].distance);
            }  
            return options.success(data,{
              calculateResult: calculateResult,
              distanceResult: distanceResult
            });
          };
          
          Utils.locationProcess(options, locationsuccess);
        } else {
          var locationsuccess = function (result) {
            requestParam.from = result.latitude + ',' + result.longitude;
            if (options.sig) {
              requestParam.sig = Utils.getSig(requestParam, options.sig, 'calculateDistance');
            }
            wx.request(Utils.buildWxRequestConfig(options, {
              url: URL_DISTANCE,
              data: requestParam
            },'calculateDistance'));
          };

          Utils.locationProcess(options, locationsuccess);
        }      
    };

  /**
   * 路线规划:
   * 
   * @param {Object} options 接口参数对象
   * 
   * 请求参数结构可以参考
   * https://lbs.qq.com/webservice_v1/guide-road.html
   */
  direction(options) {
    var that = this;
    options = options || {};
    Utils.polyfillParam(options);

    if (Utils.checkParamKeyEmpty(options, 'to')) {
      return;
    }

    var requestParam = {
      output: 'json',
      key: that.key
    };

    //to格式处理
    if (typeof options.to == 'string') {
      requestParam.to = options.to;
    } else {
      requestParam.to = options.to.latitude + ',' + options.to.longitude;
    }
    //初始化局部请求域名
    var SET_URL_DIRECTION = null;
    //设置默认mode属性
    options.mode = options.mode || MODE.driving;

    //设置请求域名
    SET_URL_DIRECTION = URL_DIRECTION + options.mode;

    if (options.from) {
      options.location = options.from;
    }

    if (options.mode == MODE.driving) {
      if (options.from_poi) {
        requestParam.from_poi = options.from_poi;
      }
      if (options.heading) {
        requestParam.heading = options.heading;
      }
      if (options.speed) {
        requestParam.speed = options.speed;
      }
      if (options.accuracy) {
        requestParam.accuracy = options.accuracy;
      }
      if (options.road_type) {
        requestParam.road_type = options.road_type;
      }
      if (options.to_poi) {
        requestParam.to_poi = options.to_poi;
      }
      if (options.from_track) {
        requestParam.from_track = options.from_track;
      }
      if (options.waypoints) {
        requestParam.waypoints = options.waypoints;
      }
      if (options.policy) {
        requestParam.policy = options.policy;
      }
      if (options.plate_number) {
        requestParam.plate_number = options.plate_number;
      }
    }

    if (options.mode == MODE.transit) {
      if (options.departure_time) {
        requestParam.departure_time = options.departure_time;
      }
      if (options.policy) {
        requestParam.policy = options.policy;
      }
    } 

    var locationsuccess = function (result) {
      requestParam.from = result.latitude + ',' + result.longitude;
      if (options.sig) {
        requestParam.sig = Utils.getSig(requestParam, options.sig, 'direction',options.mode);
      }
      wx.request(Utils.buildWxRequestConfig(options, {
        url: SET_URL_DIRECTION,
        data: requestParam
      }, 'direction'));
    };

    Utils.locationProcess(options, locationsuccess);
  }
};

module.exports = QQMapWX;

3、安装vue-jsonp

  • npm install vue-jsonp

  • main.js中引入

import VueJsonp from 'vue-jsonp'   
Vue.use(VueJsonp) 

4、使用

  • 在需要使用的界面中导入第二步重写的QQMapWX文件,(不能挂在main.js中,会报错)
  • 根据文档实现需求,文档地址: 微信小程序sdk开发文档
  • 下面是基于uniapp的例子

获取当前位置信息,逆解析地址

文档地址

<script>
	import QQMapWX from '../../libs/qqmap-wx-jssdk.js'
	export default {
		data() {
			return {
				qqMap: new QQMapWX({
					key: 申请的key,//在控制台可查看
					vm: this
				}),
			}
		},
		onLoad() {
			this.getLocation()
		},
		methods: {
			getLocation() {
				uni.showLoading({
					title: '定位中...'
				})
				this.qqMap.reverseGeocoder({
					success: res => {
						//res.result是位置信息
						let local=res.result
						uni.hideLoading()
						return (local)
					},
					fail: err => {
						uni.showToast({
							title: '获取位置信息失败,原因是' + err.message,
							icon: 'none'
						})
					},
					complete: (err) => {
						uni.hideLoading()
					}
				})
			}
		},

	}
</script>

路线规划,距离计算

文档地址

			getDirection(from, end, type) {
				this.qqMap.direction({
					mode: 'driving',
					from: from,
					to: end,
					success: res => {
						// 绘制
						res = res.result.routes[0]
						// 绘制规划路径
						if (type == 'draw') {
							// var ret = res;
							var coors = res.polyline,
								pl = [];
							//坐标解压(返回的点串坐标,通过前向差分进行压缩)
							var kr = 1000000;
							for (var i = 2; i < coors.length; i++) {
								coors[i] = Number(coors[i - 2]) + Number(coors[i]) / kr;
							}
							//将解压后的坐标放入点串数组pl中
							for (var i = 0; i < coors.length; i += 2) {
								pl.push({
									latitude: coors[i],
									longitude: coors[i + 1]
								})
							}
							//设置polyline属性,将路线显示出来,将解压坐标第一个数据作为起点
							let polyline = [{
								points: pl,
								color: '#0080FF',
								width: 4,
								arrowLine: true,
							}]
							Object.assign(this.map, {
								polyline: polyline,
								lat: pl[0].latitude,
								lng: pl[0].longitude
							})
							// console.log('规划路线成功',this.map)
						} else {
							//计算预计到达时间和距离
							uni.showLoading()
							this.info =
								`司机距您${(res.distance/1000000).toFixed(2)}公里,预计还有${Math.ceil( res.duration/60)}分钟到达`
							uni.hideLoading()
						}
					},
					complete: res => {
						console.log('路线规划完成', res)
					}
				})
			}

实时搜索

文档地址

  • 效果图
    在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

  • 代码部分

    • 整体代码
<template>
	<view class="container">
		<me-search :fixed='true' :leftText='keydata.region' placeholder='输入出发地' :searchVal='keydata.keyword' @confirm='confirm'
		 @leftClick='leftClick' @cancel='searhCancel'></me-search>
		<view class="main">

			<!-- 地图 -->
			<map id="myMap" style="width: 100vw; height: 54vh;" :latitude="lat" :longitude="lng" :markers="map.markers"
			 :show-location="true" @markertap='markertap' v-show="!isInput"></map>
			<!-- {{isInput}} -->
			<view class="addr_list" :style="isInput?'max-height:90vh':'max-height:60vh'">
				<view class="addr_item" v-for="(item,index) in datalist" :key='index' @click="choose_addr(item)">
					<text class="iconfont icon-location1"></text>
					<view class="">
						<text>{{item.title}}</text>
						<text style="height: 24rpx;line-height: 1.2;">{{item.address}}</text>
					</view>
					<!-- <text class="iconfont icon-cross"></text> -->
				</view>
			</view>
			<!-- <button class="btn_squre_670" @click="clear" v-if="datalist.length>0">
				<text class="iconfont icon-delete2 ">清除全部历史记录</text>

			</button> -->
			<uni-popup type="bottom" ref='select_popup'>
				<view class="city">
					<text v-for="item in citySome" :key='item.id' @click="selectBtn(item)">
						{{item.name}}
					</text>
				</view>
			</uni-popup>
			<uni-popup ref='modal' type="dialog" :maskClick="false">
				<me-modal title='提示' :showPopup='true' :content="content" @btn_son='modalClick' :oneline='true' @close_son='close'></me-modal>
			</uni-popup>
		</view>
		<!-- <image src="../../static/image/司机.png" mode=""></image> -->
	</view>
</template>

<script>
let LOCATION_INFO={
	address: "四川省成都市双流区藏卫路北四段107号",
	city: "成都市",
	district: "双流区",
	lat: 30.571677,
	lng: 103.925857,
	nation: "中国",
	province: "四川省",
	street: "藏卫路北四段",
	street_number: "藏卫路北四段107号",
}
	import {
		mapState
	} from 'vuex'
	import QQMapWX from '../../libs/qqmap-wx-jssdk.js'
	export default {
		data() {
			return {
				qqMap: new QQMapWX({
					key: this.public_data.key,
					vm: this
				}),
				isInput: false,
				lat: '',
				lng: '',
				location_info: LOCATION_INFO,
				content: LOCATION_INFO.city ? '检测到您当前在' + LOCATION_INFO.address +
					',确认使用该位置?' : '获取定位失败,请手动选择位置',
				datalist: [],
				// 搜地址表单
				keydata: {
					region: LOCATION_INFO.city,

					keyword: LOCATION_INFO.street || '市中心'
				},
				mapdata: {
					get_poi: 1,
					location: LOCATION_INFO.lat + ',' + LOCATION_INFO.lng || ''
				},
				// 传回页面对象
				formdata: {
					city_name: ''
				},
				city: LOCATION_INFO.city, //没有定位就用后台数据
				map: {},
			};
		},
		computed: mapState(['locationInfo', 'citySome']),
	
		created() {
			this.getCurrentLocation(this.mapdata)
			Object.assign(this.formdata, {
				start_location: LOCATION_INFO.street_number,
				start_latitude: LOCATION_INFO.lat,
				start_longitude: LOCATION_INFO.lng,
				city_name: LOCATION_INFO.city,
			})
		},
		mounted() {
			this.$refs.modal.open()
		},
		onHide() {
			this.$refs.modal.close()
		},
		methods: {
			searhCancel() {
				uni.removeStorageSync('CREATE_DATA')
				uni.navigateBack({
					delta: 1
				})
			},
			// 检测城市id是列表中的那个位置
			cityId(e) {
				let b = 0
				this.citySome.map(item => {
					if (item.name == e) {
						b = item.id
					}
				})
				return b
			},
			markertap(e) {
				console.log('mark', e.detail)
				this.datalist.map(item => {
					if (item.id == e.detail.markerId) {
						let city_id = this.cityId(item.ad_info.city)
						console.log(8888, item)
						Object.assign(this.formdata, {
							start_location: item.address+item.title,
							start_latitude: item.location.lat,
							start_longitude: item.location.lng,
							city_name: item.ad_info.city,
						})
						this.content=`确定选择${item.title}(${item.address})?`
						this.$refs.modal.open()
						return
					}
				})
			},
			getCurrentLocation(data) {
				// var that = this;
				this.qqMap.reverseGeocoder({
					...data, //是否返回周边POI列表:1.返回;0不返回(默认),非必须参数
					success: (res) => {
						let locationData = res.result
						var a = locationData.address_component; //单独
						var latitude = locationData.location.lat //locationStr.split(',')[0]
						var longitude = locationData.location.lng; //locationStr.split(',')[1]
						let mks = [];
						this.lat = latitude
						this.lng = longitude
						for (var i = 0; i < locationData.pois.length; i++) {
							mks.push({ // 获取返回结果,放到mks数组中
								title: locationData.pois[i].title,
								id: locationData.pois[i].id,
								latitude: locationData.pois[i].location.lat,
								longitude: locationData.pois[i].location.lng,
								// iconPath: './resources/placeholder.png', //图标路径
								width: 20,
								height: 20,
								label: {
									// content: locationData.pois[i].title,
									color: '#000',
									borderRadius: 10,
									bgColor: '#fff',
									padding: 3
								}
							})
						}
						//根据地址解析在地图上标记解析地址位置
						// this.datalist=
						Object.assign(this.map, {
							...locationData.location,
							markers: mks,
						})
						this.datalist = locationData.pois

						this.isInput = false
					}
				})
			},
			close() {
				this.$refs.modal.close()
			},
			modalClick(e) {
				if (e.value) {
					uni.showLoading({})
					if (this.content.indexOf('当前在') > -1) {
						// 进入该页面定位
						let local = this.location_info
						let city_id = 0
						uni.getStorage({
							key: 'CREATE_DATA',
							complete: (res) => {
								res = res.data || {}
								Object.assign(res, {
									city_id: this.cityId(local.city),
									start_latitude: local.lat,
									start_longitude: local.lng,
									start_location: local.address,
									city_name: local.city,
								})
								uni.setStorageSync('CREATE_DATA', res)
								uni.navigateTo({
									url: '/pages/addr/end'
								})
							}
						})
						this.close()
						console.log(1111, this.formdata)
					} else if (this.content.indexOf('手动选择') > -1) {
						this.close()
						this.$refs.select_popup.open()
						// 点击marker地图选点确定
					} else {
						uni.getStorage({
							key: 'CREATE_DATA',
							complete: (res) => {
								let a = res.data || {}
								console.log('缓存了', a)
								Object.assign(a, this.formdata)
								uni.setStorageSync('CREATE_DATA', a)
								uni.navigateTo({
									url: '/pages/addr/end'
								})
								// console.log(err)
							}
						})
					}
					uni.hideLoading()
				} else {
					this.close()
				}
			},
			clear() {
				this.datalist.splice(0)
			},

			leftClick(e) {
				this.$refs.select_popup.open()
			},
			selectBtn(e) {
				if (e.name !== this.formdata.city_name) {
					this.datalist = []
				}
				// this.formdata.city_name = e.name //删除后会导致视图不更新
				this.keydata.region = e.name
				Object.assign(this.formdata, {
					city_id: e.id,
					city_name: e.name
				})
				this.isInput = true
				this.keydata.region = e.name
				this.getSuggestion(this.keydata)
				this.$refs.select_popup.close()
			},
			confirm(e) {
				this.isInput = true
				Object.assign(this.keydata, {
					keyword: e.value
				})
				this.getSuggestion(this.keydata)
			},

			getSuggestion(data) {
				// console.log('参数',...data)
				uni.showLoading({
					title: '搜索中'
				})
				this.qqMap.getSuggestion({
					...data,
					success: res => {
						this.datalist = res.data
						uni.hideLoading()
					},
					fail: err => {
						uni.showToast({
							title: err.message,
							icon: 'none'
						})
					},
					complete: (err) => {
						// this.isInput=false
						// console.log(err, data, '完成')
					}
				})
			},
			choose_addr(res) {
				// this.a = res.address
				console.log(res,this.isInput, 'choose')
				if (!this.isInput) {
					Object.assign(this.formdata, {
						start_location: res.address+res.title,
						start_latitude: res.location.lat,
						start_longitude: res.location.lng,
						city_name: res.city || res.ad_info.city,
					})
					this.content=`确定选择${res.title}(${res.address})?`
					console.log(this.formdata, '啊啊啊啊')
					setTimeout(() => {
						this.$refs.modal.open()
					})
				} else {
					this.qqMap.geocoder({
						address: res.address,
						success: gres => {
							let obj = {
								get_poi: 1,
								location: gres.result.location.lat + ',' + gres.result.location.lng
							}
							this.getCurrentLocation(obj)
							this.isInput = false
						},
						fail:err=>{
							console.log(err)
						},
						complete:error=>{
							console.log(error,'搜索错误')
						}
					})
				}
			},
			chooseMap(e) {
				// console.log(e)
				uni.chooseLocation({
					success(res) {
						console.log(res)
						console.log('位置名称:' + res.name);
						console.log('详细地址:' + res.address);
						console.log('纬度:' + res.latitude);
						console.log('经度:' + res.longitude);
						Object.assgin(this.map, {
							latitude: res.latitude,
							longitude: res.longitude,
							title: res.name
						})
					},
					complete(err) {
						console.log('结束')
						console.log(err)
					}
				});
			}
		}
	}
</script>
<style lang="scss">
	.container {
		height: 100%;

		.btn_squre_670 {
			font-weight: 400;
			background: #f3f3f3;
			color: #999;
			margin: 20rpx auto;
		}

		>.main {
			margin-top: 10vh;
		}

		.addr_list {
			max-height: 60vh;
			overflow-y: scroll;
			background: #f00;
		}

		.li_88 {
			background: #f3f3f3;
			text-align: center;
			color: #999;
			width: 702rpx;
			margin: 20rpx auto;
		}
	}
</style>

(2) me-search组件

<template>
	<view class="search" :style="{backgroundColor: bgColor,position:fixed?'fixed':'static',}">
		<view class="" @click="select">
			<text>{{leftText}}</text>
			<text class="iconfont icon-tubiaozhizuo-"></text>
		</view>
		<input :focus="showSync" :style="{backgroundColor:inputbgColor}" :placeholder="placeholder" placeholder-style="color:#ccc"
		 :maxlength="maxlength" @confirm="confirm" confirm-type="search" type="text" v-model="searchVal"/>
		<text v-if="clearButton" class="iconfont icon-shanchu" @click="clear"></text>
		<text class="" @click="cancel">
			{{cancelText}}
		</text>
	</view>
</template>

<script>
let LOCATION_INFO={
	address: "四川省成都市双流区藏卫路北四段107号",
	city: "成都市",
	district: "双流区",
	lat: 30.571677,
	lng: 103.925857,
	nation: "中国",
	province: "四川省",
	street: "藏卫路北四段",
	street_number: "藏卫路北四段107号",
}
	import {mapState} from 'vuex'
	export default {
		data() {
			return {
				showSync: false,
				searchVal:'',
			};
		},
		props: {	
			leftText: {
				type: String,
				default: LOCATION_INFO.city||'请选择',
			},
			cancelText: {
				type: String,
				default: '取消'
			},
			placeholder: {
				type: String,
				default: "请输入搜索内容"
			},
			maxlength: {
				type: [Number, String],
				default: 100
			},
			clearButton: {
				type: Boolean,
				default: true
			},
			bgColor: {
				type: String,
				default: "#fff"
			},
			inputbgColor: {
				type: String,
				default: '#fff'
			},
			fixed: {
				type: Boolean,
				default: false
			}
		},
		created() {
			// this.leftText = this.leftText
			// this.placeholder
		},
		computed:mapState(['locationInfo']),
		methods: {
			input(){
				this.$emit('input',{
					value:this.searchVal
				})
			},
			select() {
				this.$emit('leftClick',{
					value:true,
				})
			},
			clear() {
				this.searchVal = ""
			},
			cancel() {
				this.$emit("cancel");
			},
			confirm() {
				// #ifndef APP-PLUS
				uni.hideKeyboard();
				// #endif
				// #ifdef APP-PLUS
				plus.key.hideSoftKeybord()
				// #endif
				this.$emit("confirm", {
					value:this.searchVal
				})
			}
		}
	}
</script>

<style lang="scss">
	.search {
		width: 100%;
		background: #fff;
		height: 10vh;
		display: flex;
		justify-content: flex-start;
		align-items: center;
		box-shadow: 0 0 20rpx rgba(0,0,0,0.2);
		margin-bottom: 20rpx;
		z-index: 100;

		>view {
			width: 120rpx;
		}

		>input {
			width: 490rpx;
			padding: 0 20rpx;
			border-left: 1rpx solid #ccc;
			border-right: 1rpx solid #ccc;
		}

		>view {
			text-align: center;

			// >.iconfont {
			// 	font-size: 10rpx;
			// }
		}

		>text {
			color: #ccc;
		}

		>text:last-child {
			display: inline-block;
			width: 83rpx;
			text-align: center;
		}
		.iconfont{
			font-size: 20rpx;
		}
		>.iconfont {
			position: absolute;
			flex: 0 0 0%;
			right: 130rpx;
		}
	}
</style>

(3) me-model组件

<template>
	<!-- <uni-transition :styles="maskClass" :duration="duration" :show="showPopup" class="model"> -->
	<!-- <uni-popup ref="popup" type="dialog" class="model"> -->
	<view class="model" :styles="maskClass" v-if="showPopup">
		<view class="main">
			<view name='header' class="header">
				<text class="t_000_32">{{title}}</text>
				<text @click="close" class="iconfont icon-cross" v-if="iconDelete"></text>
			</view>
			<view class="body">
				<text class="t_ccc_28" :style="{color:color}" v-if="oneline">{{content}}</text>
				<view class="" v-else>
					<text>{{content_1}}</text>
					<text>{{content_2}}</text>
				</view>
				
			</view>
			<!-- <slot name='aa'>aaa</slot> -->
			<view class="footer">
				<button class="cancel" v-if="cancelShow" @click="btnClick(false)">{{cancelText}}</button>
				<button @click="btnClick(true)" class="confirm" v-if="confirmShow">{{confirmText}}</button>
			</view>
		</view>
		</view>
		<!-- <view class="main">
			<view v-slot="header">
				{{content}}
			</view>
			<view v-slot="body">body</view>
			<view v-slot="footer">
				footer
			</view>
		</view> -->
	<!-- </uni-transition> -->
	<!-- </uni-popup> -->
</template>

<script>
	export default {
		data() {
			return {
				duration: 1000,
				maskClass: {
					'backgroundColor': ' rgba(118, 117, 117, 0.501)'
					},
			};
		},
		props: {
			oneline: {
				type: Boolean,
				default: false
			},
			showTrans: {
				type: Boolean,
				default: false
			},
			title: {
				type: String,
				default: '提示'
			},
			iconDelete: {
				type: Boolean,
				default: true
			},
			content: {
				type: String,
				default:'内容'
			},
			color:{
				type:String,
				default:'#ccc',
			},
			content_1:{
				type:String,
				default:'第一排数字'
			},
			content_2:{
				type:String,
				default:'第二排数字'
			},
			cancelShow: {
				type: Boolean,
				default: true
			},
			cancelText: {
				type: String,
				default: '取消'
			},
			confirmShow: {
				type: Boolean,
				default: true
			},
			confirmText: {
				type: String,
				default: '确认'
			},
			showPopup: {
				type: Boolean,
				default: true
			}
		},
		methods: {
			close() {
				this.showTrans = false
				this.$nextTick(() => {
					this.$emit('close_son', false)
				}, 200)
			},
			btnClick(res) {
				console.log('子组1件', res)
				this.$emit('btn_son', {
					// value: res == 'ok' ? true: false
					value: res
				})
			},

		}
	}
</script>

<style lang="scss">
	.model {
		position: static;
		z-index: 999;
		// top: 0;
		>view {
			width: 660rpx;
			margin: 40vh auto;
			min-height: 340rpx;
			background: #ffffff;
			border-radius: 10rpx;
			position: relative;
			// text-align: center;
			display: flex;
			flex-flow: column nowrap;
			justify-content: space-between;
			align-items: center;
			>.header {
				line-height: 88rpx;
				height: 88rpx;
				>.iconfont {
					position: absolute;
					right: 30rpx;
					color: #999;
				}
			}

			>.body {
				width: 100%;
				text-align: center;
				>view{
					// background: #f00;
					height: 150rpx;
					display: flex;
					flex-flow: column nowrap;
					// justify-content: space-between;
					align-items: center;
					>text:first-child{
						font-size: 32rpx;
						margin-bottom: 40rpx;
					}
				}
				.t_ccc_28{
					    width: 80%;
					    display: block;
					    margin: 0 auto;
						color: #999;
				}
			}

			>.footer{
				display: flex;
				justify-content: space-around;
				width: 90%;

				button {
					flex: 0 0 45%;
					height: 88rpx;
					font-size: 32rpx;
					background: #f3f3f3;
					border-radius: 10rpx;
					margin: 20rpx 0;
					border: none;
				}

				.confirm {
					background: #019FE8;
					color: #fff;
				}
			}
		}
	}
</style>

参考文档:https://www.jianshu.com/p/4847480995c9

;