在H5中使用腾讯地图,实现定位,距离计算,实时搜索,地址逆解析
1、创建应用
在腾讯位置服务控制台中,创建应用,填入名称、描述、产品类型、已经域名白名单
一定要配置域名白名单,否则无法使用
2、下载微信sdk包
-
解压后放入根目录
-
将
qqmap-wx-jssdk.js
替换成以下代码
注意:这个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>