@ -0,0 +1,73 @@ |
||||
<style lang="scss"> |
||||
/* 注意要写在第一行,同时给style标签加入lang="scss"属性 */ |
||||
@import "@/uni_modules/uview-ui/index.scss"; |
||||
@import "@/static/css/iconfont.css"; |
||||
</style> |
||||
<script> |
||||
export default { |
||||
onLaunch: function(options) { |
||||
console.log('App Launch'); |
||||
var _this =this; |
||||
//系统信息 |
||||
uni.getSystemInfo({ |
||||
success:(res) => { |
||||
console.log(res); |
||||
_this.$user.session('app_system_info',res); |
||||
//检测当前平台,如果是安卓则启动安卓更新 |
||||
_this.$user.session('app_system',res.platform); |
||||
} |
||||
}); |
||||
|
||||
// #ifdef APP-PLUS |
||||
// 锁定屏幕方向 |
||||
plus.screen.lockOrientation('portrait-primary'); //锁定 |
||||
// #endif |
||||
}, |
||||
onShow: function() { |
||||
console.log('App Show') |
||||
var _this =this; |
||||
// #ifdef MP |
||||
//这里目前微信小程序和抖音功能基本一样 |
||||
uni.getProvider({ |
||||
service: 'oauth', |
||||
success: function (res) { |
||||
console.log(res.provider) |
||||
var provider = res.provider[0]; |
||||
|
||||
uni.login({ |
||||
provider:provider, |
||||
scopes:'auth_base', |
||||
success: function (rs) { |
||||
console.log(rs); |
||||
//发起网络请求 |
||||
var post = {loginCode: rs.code}; |
||||
_this.$api.post('ycl/user/wx-login', post,function(res){ |
||||
console.log(res); |
||||
_this.$user.session('openid',res.wxOpenid); |
||||
_this.$user.session('user_id',res.id); |
||||
if(!_this.$com.isNull(res.token)) |
||||
{ |
||||
_this.$user.session('token',res.token); |
||||
} |
||||
}); |
||||
}, |
||||
fail:function(rs){ |
||||
console.log('登录失败'+rs.errMsg); |
||||
} |
||||
}); |
||||
|
||||
|
||||
|
||||
|
||||
} |
||||
}); |
||||
|
||||
|
||||
// #endif |
||||
}, |
||||
onHide: function() { |
||||
console.log('App Hide') |
||||
} |
||||
} |
||||
</script> |
||||
|
@ -0,0 +1,90 @@ |
||||
|
||||
var access = { |
||||
|
||||
getAccess:function(k,userinfo){ |
||||
if (userinfo && userinfo.access) { |
||||
return userinfo.access.indexOf(k) != -1; |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
hasAccess:function(arr,userinfo) { |
||||
if (userinfo && userinfo.access) { |
||||
return userinfo.access.some(_ => arr.indexOf(_) > -1); |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
/** 是否一级商户 */ |
||||
isSuperType:function(userinfo){ |
||||
if (this.getAccess(1002,userinfo)) { |
||||
return userinfo.userType == 2 |
||||
} |
||||
if (this.getAccess(1003,userinfo) || this.getAccess(1004,userinfo)) { |
||||
return userinfo.userType == 1 |
||||
} |
||||
return false; |
||||
}, |
||||
/** 是否区级商户 */ |
||||
isAreaType:function(userinfo){ |
||||
if (this.getAccess(1002,userinfo)) { |
||||
return userinfo.userType == 1 |
||||
} |
||||
if (this.getAccess(1003,userinfo) || this.getAccess(1004,userinfo)) { |
||||
return userinfo.userType == 2 |
||||
} |
||||
return false; |
||||
}, |
||||
/** 是否员工 */ |
||||
isYuangongType:function(userinfo){ |
||||
if (this.getAccess(1002,userinfo)) { |
||||
return userinfo.userType == 3 |
||||
} |
||||
if (this.getAccess(1004,userinfo)) { |
||||
return userinfo.userType == 4 |
||||
} |
||||
return false; |
||||
}, |
||||
/** 是否三级 */ |
||||
isThirdType:function(userinfo){ |
||||
if (this.getAccess(1004,userinfo)) { |
||||
return userinfo.userType == 3 |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
|
||||
// 判断身份
|
||||
checkidentity:function(userinfo) { |
||||
var res = {} |
||||
if (this.isSuperType(userinfo)) { |
||||
res = { |
||||
levelTxt: '一级', |
||||
levelVal: 1 |
||||
} |
||||
} else if (this.isAreaType(userinfo)) { |
||||
res = { |
||||
levelTxt: '二级', |
||||
levelVal: 2 |
||||
} |
||||
} else if (this.isThirdType(userinfo)) { |
||||
res = { |
||||
levelTxt: '店长', |
||||
levelVal: 3 |
||||
} |
||||
/* } else if (this.isYuangongType(userinfo)) { |
||||
res = { |
||||
levelTxt: '员工', |
||||
levelVal: 3 |
||||
} */ |
||||
} else { |
||||
res = { |
||||
levelTxt: '用户', |
||||
levelVal: 4 |
||||
} |
||||
} |
||||
return res |
||||
} |
||||
} |
||||
|
||||
module.exports = access |
@ -0,0 +1,200 @@ |
||||
const md5 = require('./md5.js'); |
||||
const ENV = require('./env.js'); |
||||
var api = { |
||||
|
||||
/** |
||||
* post 请求数据
|
||||
*
|
||||
* @param string urlName url地址 |
||||
* @param {object} data 数据对象 |
||||
* @param function thenFun then回调 |
||||
* @param function catchFun catch回调 |
||||
*/ |
||||
post:function(urlName, data, thenFun,catchFun) { |
||||
//数据组合
|
||||
var newData = Object.assign(api.setSign(data), ENV.appData()); |
||||
uni.$u.http.post(urlName, newData).then(res => { |
||||
if(thenFun) thenFun(res); |
||||
}).catch((rs) =>{ |
||||
if(catchFun) catchFun(rs); |
||||
}); |
||||
}, |
||||
|
||||
/** |
||||
* get 请求数据
|
||||
*
|
||||
* @param string urlName url地址 |
||||
* @param {object} data 数据对象 |
||||
* @param function thenFun then回调 |
||||
* @param function catchFun catch回调 |
||||
*/ |
||||
get:function(urlName, data, thenFun,catchFun) { |
||||
//数据组合
|
||||
var newData = Object.assign(api.setSign(data), ENV.appData()); |
||||
uni.$u.http.get(urlName,{params:newData}).then(res => { |
||||
if(thenFun) thenFun(res); |
||||
}).catch((rs) =>{ |
||||
if(catchFun) catchFun(rs); |
||||
}); |
||||
}, |
||||
|
||||
/** |
||||
* put 请求数据
|
||||
*
|
||||
* @param string urlName url地址 |
||||
* @param {object} data 数据对象 |
||||
* @param function thenFun then回调 |
||||
* @param function catchFun catch回调 |
||||
*/ |
||||
put:function(urlName, data, thenFun,catchFun) { |
||||
//数据组合
|
||||
var newData = Object.assign(api.setSign(data), ENV.appData()); |
||||
uni.$u.http.put(urlName, newData).then(res => { |
||||
if(thenFun) thenFun(res); |
||||
}).catch((rs) =>{ |
||||
if(catchFun) catchFun(rs); |
||||
}); |
||||
}, |
||||
/** |
||||
* delete 请求数据
|
||||
*
|
||||
* @param string urlName url地址 |
||||
* @param {object} data 数据对象 |
||||
* @param function thenFun then回调 |
||||
* @param function catchFun catch回调 |
||||
*/ |
||||
delete:function(urlName, data, thenFun,catchFun) { |
||||
//数据组合
|
||||
var newData = Object.assign(api.setSign(data), ENV.appData()); |
||||
uni.$u.http.delete(urlName, newData).then(res => { |
||||
if(thenFun) thenFun(res); |
||||
}).catch((rs) =>{ |
||||
if(catchFun) catchFun(rs); |
||||
}); |
||||
}, |
||||
/** |
||||
* upload 请求数据
|
||||
*
|
||||
* @param string urlName url地址 |
||||
* @param {object} data 数据对象 {filePath:'',fileType:'image'} |
||||
* @param function thenFun then回调 |
||||
* @param function catchFun catch回调 |
||||
*/ |
||||
upload:function(urlName, data, thenFun,catchFun){ |
||||
var newData = data; |
||||
uni.$u.http.upload(urlName, { |
||||
params: {}, /* 会加在url上 */ |
||||
// #ifdef MP-ALIPAY
|
||||
fileType: newData.fileType, // 仅支付宝小程序,且必填。image/video/audio
|
||||
// #endif
|
||||
filePath: newData.filePath, // 要上传文件资源的路径。
|
||||
// 注:如果局部custom与全局custom有同名属性,则后面的属性会覆盖前面的属性,相当于Object.assign(全局,局部)
|
||||
custom: {auth:true,toast:true,}, // 可以加一些自定义参数,在拦截器等地方使用。比如这里我加了一个auth,可在拦截器里拿到,如果true就传token
|
||||
name: 'files', // 文件对应的 key , 开发者在服务器端通过这个 key 可以获取到文件二进制内容
|
||||
// #ifdef H5 || APP-PLUS
|
||||
timeout: 60000, // H5(HBuilderX 2.9.9+)、APP(HBuilderX 2.9.9+)
|
||||
// #endif
|
||||
header: { |
||||
"Content-Type": "multipart/form-data" |
||||
}, /* 会与全局header合并,如有同名属性,局部覆盖全局 */ |
||||
formData: {}, // HTTP 请求中其他额外的 form data
|
||||
// 返回当前请求的task, options。请勿在此处修改options。非必填
|
||||
getTask: (task, options) => { |
||||
task.onProgressUpdate((res) => { |
||||
console.log('上传进度' + res.progress); |
||||
console.log('已经上传的数据长度' + res.totalBytesSent); |
||||
console.log('预期需要上传的数据总长度' + res.totalBytesExpectedToSend); |
||||
}); |
||||
}, |
||||
}).then(res => { |
||||
if(thenFun) thenFun(res); |
||||
}).catch((rs) =>{ |
||||
if(catchFun) catchFun(rs); |
||||
}); |
||||
}, |
||||
|
||||
|
||||
setSign:function(data) { |
||||
const date = parseInt(new Date().getTime() / 1000); |
||||
const str = date + 'xingzhi&&2021'; |
||||
data.i_app_timestamp = date; |
||||
data.i_app_key = md5(str); |
||||
//判断登录 openid 和 unionid 的来源
|
||||
// #ifdef APP-PLUS
|
||||
data.i_app_platform = 'APP-PLUS';
|
||||
// #endif
|
||||
|
||||
// #ifdef MP
|
||||
//微信小程序
|
||||
// #ifdef MP-WEIXIN
|
||||
data.i_app_platform = 'MP-WEIXIN';
|
||||
// #endif
|
||||
|
||||
//支付宝小程序
|
||||
// #ifdef MP-ALIPAY
|
||||
data.i_app_platform = 'MP-ALIPAY';
|
||||
// #endif
|
||||
|
||||
//百度小程序
|
||||
// #ifdef MP-BAIDU
|
||||
data.i_app_platform = 'MP-BAIDU';
|
||||
// #endif
|
||||
|
||||
//头条小程序
|
||||
// #ifdef MP-TOUTIAO
|
||||
data.i_app_platform = 'MP-TOUTIAO';
|
||||
// #endif
|
||||
|
||||
//QQ小程序
|
||||
// #ifdef MP-QQ
|
||||
data.i_app_platform = 'MP-QQ';
|
||||
// #endif
|
||||
|
||||
//360小程序
|
||||
// #ifdef MP-360
|
||||
data.i_app_platform = 'MP-360';
|
||||
// #endif
|
||||
|
||||
|
||||
|
||||
// #endif
|
||||
|
||||
// #ifdef H5
|
||||
data.i_app_platform = 'H5';
|
||||
// #endif
|
||||
//语言包
|
||||
data.i_app_lang ='zh'; |
||||
//系统
|
||||
data.i_app_system = uni.getStorageSync('app_system'); |
||||
|
||||
//这里是token 如果登录每个接口都需要这个
|
||||
const token = uni.getStorageSync('token'); |
||||
if(token.length > 0) |
||||
{ |
||||
data.token = token; |
||||
}
|
||||
return data; |
||||
}, |
||||
/** |
||||
* 返回域名 |
||||
*
|
||||
*/ |
||||
domain:function() { |
||||
return ENV.CdnUrl; |
||||
}, |
||||
/** |
||||
* 返回接口地址 |
||||
*
|
||||
*/ |
||||
api:function() { |
||||
return ENV.ApiUrl; |
||||
}, |
||||
/** |
||||
* 返回接口环境 |
||||
*
|
||||
*/ |
||||
env:function(){ |
||||
return ENV.EnvShow; |
||||
}, |
||||
} |
||||
module.exports = api; |
@ -0,0 +1,268 @@ |
||||
const u_check = uni.$u.test; |
||||
var com = { |
||||
|
||||
/**************uview集成********************************/ |
||||
/** |
||||
* 验证检测 |
||||
*
|
||||
* @param array || obj || ... value 需要验证的值(可以是 array || obj || ... ) |
||||
* @param string type 需要验证的类型(可以是 array || obj || ...) |
||||
*/ |
||||
check:function(value,type) |
||||
{ |
||||
switch(type) { |
||||
//数组
|
||||
case 'array': |
||||
return u_check.array(value); |
||||
break; |
||||
//json字符串
|
||||
case 'json': |
||||
return u_check.jsonString(value); |
||||
break; |
||||
//object对象
|
||||
case 'object': |
||||
return u_check.object(value); |
||||
break;
|
||||
//email 邮箱号
|
||||
case 'email': |
||||
return u_check.email(value); |
||||
break;
|
||||
//mobile 手机号
|
||||
case 'mobile': |
||||
return u_check.mobile(value); |
||||
break; |
||||
//url 地址
|
||||
case 'url': |
||||
return u_check.url(value); |
||||
break;
|
||||
//empty 是否为空
|
||||
case 'empty': |
||||
return u_check.isEmpty(value); |
||||
break;
|
||||
//注解
|
||||
//总的来说,年月日之间可以用"/"或者"-"分隔(不能用中文分隔),时分秒之间用":"分隔,数值不能超出范围,如月份不能为13,则检验成功,否则失败。
|
||||
//date 日期时间
|
||||
case 'date': |
||||
return u_check.date(value); |
||||
break; |
||||
//number 是否十进制数值
|
||||
case 'number': |
||||
return u_check.number(value); |
||||
break;
|
||||
//digits 是否整数
|
||||
case 'digits': |
||||
return u_check.digits(value); |
||||
break;
|
||||
//idCard 是否身份证号
|
||||
case 'id_card': |
||||
return u_check.idCard(value); |
||||
break;
|
||||
//是否车牌号 (例:京A88888)
|
||||
case 'car_no': |
||||
return u_check.carNo(value); |
||||
break;
|
||||
//amount 金额字符串
|
||||
case 'amount': |
||||
return u_check.amount(value); |
||||
break; |
||||
//zh 是否汉字
|
||||
case 'zh': |
||||
return u_check.chinese(value); |
||||
break; |
||||
//en 是否字母 letter
|
||||
case 'en': |
||||
return u_check.letter(value); |
||||
break; |
||||
//str 是否字母或者数字
|
||||
case 'str': |
||||
return u_check.enOrNum(value); |
||||
break; |
||||
default: |
||||
return false; |
||||
}
|
||||
}, |
||||
//是否为空
|
||||
isNull:function(value) |
||||
{ |
||||
if(value != null || value != 'null') |
||||
{ |
||||
return u_check.isEmpty(value); |
||||
} |
||||
else |
||||
{ |
||||
return true; |
||||
} |
||||
}, |
||||
|
||||
/********************反馈信息****************************************************/ |
||||
alert: function(info) { |
||||
uni.showModal({ |
||||
title: '提示', |
||||
content: info, |
||||
showCancel: false |
||||
}); |
||||
}, |
||||
|
||||
alertDo: function(info, callback) { |
||||
uni.showModal({ |
||||
title: '提示', |
||||
content: info, |
||||
showCancel: false, |
||||
success() { |
||||
callback(); |
||||
} |
||||
}); |
||||
}, |
||||
confirm: function(info, callback, title) { |
||||
uni.showModal({ |
||||
title: title ? title : '提示', |
||||
content: info, |
||||
success: function(res) { |
||||
if (res.confirm) { |
||||
callback(); |
||||
} |
||||
} |
||||
}); |
||||
}, |
||||
|
||||
// 加载信息,带遮罩
|
||||
showLoading: function(title = '', mask = true) { |
||||
uni.showLoading({ |
||||
title, |
||||
mask |
||||
}); |
||||
},
|
||||
|
||||
// 隐藏遮罩
|
||||
hideLoading:function () { |
||||
uni.hideLoading(); |
||||
},
|
||||
|
||||
showSuccess: function(info) { |
||||
uni.showToast({ |
||||
title: info, |
||||
icon: 'success', |
||||
duration: 3000 |
||||
}); |
||||
}, |
||||
showError: function(info) { |
||||
uni.showToast({ |
||||
title: info, |
||||
icon: 'none', |
||||
duration: 3000 |
||||
}); |
||||
}, |
||||
showSuccessDo: function(info, time, callback) { |
||||
uni.showToast({ |
||||
title: info, |
||||
icon: 'success', |
||||
duration: time, |
||||
success: function() { |
||||
callback(); |
||||
} |
||||
}); |
||||
}, |
||||
/******************加减乘除****************************************************/ |
||||
/** |
||||
* 加法精度计算 |
||||
*
|
||||
* @param {Object} number 数字 |
||||
*/ |
||||
bcadd: function(a, b){ |
||||
var c, d, e; |
||||
try { |
||||
c = a.toString().split(".")[1].length |
||||
} catch(f) { |
||||
c = 0 |
||||
} |
||||
try { |
||||
d = b.toString().split(".")[1].length |
||||
} catch(f) { |
||||
d = 0 |
||||
} |
||||
return e = Math.pow(10, Math.max(c, d)),(this.bcmul(a, e) + this.bcmul(b, e)) / e |
||||
}, |
||||
|
||||
/** |
||||
* 减法精度计算 |
||||
*
|
||||
* @param {Object} number 数字 |
||||
*/ |
||||
bcsub: function(a, b){ |
||||
var c, d, e; |
||||
try { |
||||
c = a.toString().split(".")[1].length |
||||
} catch(f) { |
||||
c = 0 |
||||
} |
||||
try { |
||||
d = b.toString().split(".")[1].length |
||||
} catch(f) { |
||||
d = 0 |
||||
} |
||||
return e = Math.pow(10, Math.max(c, d)),(this.bcmul(a, e) - this.bcmul(b, e)) / e |
||||
}, |
||||
|
||||
/** |
||||
* 乘法精度计算 |
||||
*
|
||||
* @param {Object} number 数字 |
||||
*/ |
||||
bcmul: function(a, b){ |
||||
var c = 0, |
||||
d = a.toString(), |
||||
e = b.toString(); |
||||
try { |
||||
c += d.split(".")[1].length |
||||
} catch(f) {} |
||||
try { |
||||
c += e.split(".")[1].length |
||||
} catch(f) {} |
||||
return Number(d.replace(".", "")) * Number(e.replace(".", "")) / Math.pow(10, c) |
||||
}, |
||||
|
||||
/** |
||||
* 除法精度计算 |
||||
*
|
||||
* @param {Object} number 数字 |
||||
*/ |
||||
bcdiv: function(a, b){ |
||||
var c, d, e = 0, |
||||
f = 0; |
||||
try { |
||||
e = a.toString().split(".")[1].length |
||||
} catch(g) {} |
||||
try { |
||||
f = b.toString().split(".")[1].length |
||||
} catch(g) {} |
||||
return c = Number(a.toString().replace(".", "")),d = Number(b.toString().replace(".", "")),this.bcmul(c / d, Math.pow(10, f - e)) |
||||
}, |
||||
|
||||
/** |
||||
* 隐藏手机号 |
||||
* @param phone string 数值1 |
||||
*/ |
||||
hidephone:function (phone){ |
||||
var reg = /^(\d{3})\d{4}(\d{4})$/;
|
||||
return phone.replace(reg, "$1****$2") |
||||
}, |
||||
|
||||
removeObserver(arr) { |
||||
return arr.map((item) => { |
||||
if (Array.isArray(item)) return removeObserver(item) |
||||
else if (typeof item === 'object') { |
||||
const target = {} |
||||
for (let key in item) { |
||||
if (key !== '__ob__') { |
||||
target[key] = item[key] |
||||
} |
||||
} |
||||
return target |
||||
} |
||||
return item |
||||
}) |
||||
} |
||||
|
||||
} |
||||
|
||||
module.exports = com |
@ -0,0 +1,34 @@ |
||||
//api接口环境配置
|
||||
// production正式/development线上测试/local本地开发
|
||||
var NowENV = 'development';
|
||||
var ENV_API_URL = { |
||||
production:'https://netcharge.spacechong.com/', //正式环境
|
||||
development:'https://devnetcharge.spacechong.com/', //开发测试环境
|
||||
local:'http://192.168.10.123:8901/',//本地调试环境
|
||||
|
||||
} |
||||
//CdnUrl 环境配置[都默认为线上都行]
|
||||
var ENV_BASE_URL = { |
||||
production:'https://netcharge.spacechong.com', //正式环境
|
||||
development:'https://netcharge.spacechong.com', //开发测试环境
|
||||
local:'https://netcharge.spacechong.com',//本地调试环境
|
||||
} |
||||
|
||||
|
||||
|
||||
var ENV = { |
||||
EnvShow:NowENV,//当前环境
|
||||
ApiUrl:ENV_API_URL[NowENV], //正式
|
||||
CdnUrl:ENV_BASE_URL[NowENV], //正式
|
||||
|
||||
appData:function() { |
||||
var data = {}; //版本号
|
||||
data.i_app_version = '2.1.2'; |
||||
return data; |
||||
}, |
||||
QQmapkey:'2EBBZ-RAL33-TLM3A-3KVVR-V2LMO-3TFQ4', |
||||
|
||||
|
||||
} |
||||
|
||||
module.exports = ENV |
@ -0,0 +1,701 @@ |
||||
! function(e, n) { |
||||
"function" == typeof define && (define.amd || define.cmd) ? define(function() { |
||||
return n(e) |
||||
}) : n(e, !0) |
||||
}(window, function(o, e) { |
||||
if (!o.jWeixin) { |
||||
var n, c = { |
||||
config: "preVerifyJSAPI", |
||||
onMenuShareTimeline: "menu:share:timeline", |
||||
onMenuShareAppMessage: "menu:share:appmessage", |
||||
onMenuShareQQ: "menu:share:qq", |
||||
onMenuShareWeibo: "menu:share:weiboApp", |
||||
onMenuShareQZone: "menu:share:QZone", |
||||
previewImage: "imagePreview", |
||||
getLocation: "geoLocation", |
||||
openProductSpecificView: "openProductViewWithPid", |
||||
addCard: "batchAddCard", |
||||
openCard: "batchViewCard", |
||||
chooseWXPay: "getBrandWCPayRequest", |
||||
openEnterpriseRedPacket: "getRecevieBizHongBaoRequest", |
||||
startSearchBeacons: "startMonitoringBeacons", |
||||
stopSearchBeacons: "stopMonitoringBeacons", |
||||
onSearchBeacons: "onBeaconsInRange", |
||||
consumeAndShareCard: "consumedShareCard", |
||||
openAddress: "editAddress" |
||||
}, |
||||
a = function() { |
||||
var e = {}; |
||||
for (var n in c) e[c[n]] = n; |
||||
return e |
||||
}(), |
||||
i = o.document, |
||||
t = i.title, |
||||
r = navigator.userAgent.toLowerCase(), |
||||
s = navigator.platform.toLowerCase(), |
||||
d = !(!s.match("mac") && !s.match("win")), |
||||
u = -1 != r.indexOf("wxdebugger"), |
||||
l = -1 != r.indexOf("micromessenger"), |
||||
p = -1 != r.indexOf("android"), |
||||
f = -1 != r.indexOf("iphone") || -1 != r.indexOf("ipad"), |
||||
m = (n = r.match(/micromessenger\/(\d+\.\d+\.\d+)/) || r.match(/micromessenger\/(\d+\.\d+)/)) ? n[1] : "", |
||||
g = { |
||||
initStartTime: L(), |
||||
initEndTime: 0, |
||||
preVerifyStartTime: 0, |
||||
preVerifyEndTime: 0 |
||||
}, |
||||
h = { |
||||
version: 1, |
||||
appId: "", |
||||
initTime: 0, |
||||
preVerifyTime: 0, |
||||
networkType: "", |
||||
isPreVerifyOk: 1, |
||||
systemType: f ? 1 : p ? 2 : -1, |
||||
clientVersion: m, |
||||
url: encodeURIComponent(location.href) |
||||
}, |
||||
v = {}, |
||||
S = { |
||||
_completes: [] |
||||
}, |
||||
y = { |
||||
state: 0, |
||||
data: {} |
||||
}; |
||||
O(function() { |
||||
g.initEndTime = L() |
||||
}); |
||||
var I = !1, |
||||
_ = [], |
||||
w = { |
||||
config: function(e) { |
||||
B("config", v = e); |
||||
var t = !1 !== v.check; |
||||
O(function() { |
||||
if (t) M(c.config, { |
||||
verifyJsApiList: C(v.jsApiList), |
||||
verifyOpenTagList: C(v.openTagList) |
||||
}, function() { |
||||
S._complete = function(e) { |
||||
g.preVerifyEndTime = L(), y.state = 1, y.data = e |
||||
}, S.success = function(e) { |
||||
h.isPreVerifyOk = 0 |
||||
}, S.fail = function(e) { |
||||
S._fail ? S._fail(e) : y.state = -1 |
||||
}; |
||||
var t = S._completes; |
||||
return t.push(function() { |
||||
! function() { |
||||
if (!(d || u || v.debug || m < "6.0.2" || h.systemType < 0)) { |
||||
var i = new Image; |
||||
h.appId = v.appId, h.initTime = g.initEndTime - g.initStartTime, h.preVerifyTime = g.preVerifyEndTime - g.preVerifyStartTime, w.getNetworkType({ |
||||
isInnerInvoke: !0, |
||||
success: function(e) { |
||||
h.networkType = e.networkType; |
||||
var n = "https://open.weixin.qq.com/sdk/report?v=" + h.version + "&o=" + h.isPreVerifyOk + "&s=" + h.systemType + "&c=" + h.clientVersion + "&a=" + h.appId + "&n=" + h.networkType + "&i=" + h.initTime + "&p=" + h.preVerifyTime + "&u=" + h.url; |
||||
i.src = n |
||||
} |
||||
}) |
||||
} |
||||
}() |
||||
}), S.complete = function(e) { |
||||
for (var n = 0, i = t.length; n < i; ++n) t[n](); |
||||
S._completes = [] |
||||
}, S |
||||
}()), g.preVerifyStartTime = L(); |
||||
else { |
||||
y.state = 1; |
||||
for (var e = S._completes, n = 0, i = e.length; n < i; ++n) e[n](); |
||||
S._completes = [] |
||||
} |
||||
}), w.invoke || (w.invoke = function(e, n, i) { |
||||
o.WeixinJSBridge && WeixinJSBridge.invoke(e, x(n), i) |
||||
}, w.on = function(e, n) { |
||||
o.WeixinJSBridge && WeixinJSBridge.on(e, n) |
||||
}) |
||||
}, |
||||
ready: function(e) { |
||||
0 != y.state ? e() : (S._completes.push(e), !l && v.debug && e()) |
||||
}, |
||||
error: function(e) { |
||||
m < "6.0.2" || (-1 == y.state ? e(y.data) : S._fail = e) |
||||
}, |
||||
checkJsApi: function(e) { |
||||
M("checkJsApi", { |
||||
jsApiList: C(e.jsApiList) |
||||
}, (e._complete = function(e) { |
||||
if (p) { |
||||
var n = e.checkResult; |
||||
n && (e.checkResult = JSON.parse(n)) |
||||
} |
||||
e = function(e) { |
||||
var n = e.checkResult; |
||||
for (var i in n) { |
||||
var t = a[i]; |
||||
t && (n[t] = n[i], delete n[i]) |
||||
} |
||||
return e |
||||
}(e) |
||||
}, e)) |
||||
}, |
||||
onMenuShareTimeline: function(e) { |
||||
P(c.onMenuShareTimeline, { |
||||
complete: function() { |
||||
M("shareTimeline", { |
||||
title: e.title || t, |
||||
desc: e.title || t, |
||||
img_url: e.imgUrl || "", |
||||
link: e.link || location.href, |
||||
type: e.type || "link", |
||||
data_url: e.dataUrl || "" |
||||
}, e) |
||||
} |
||||
}, e) |
||||
}, |
||||
onMenuShareAppMessage: function(n) { |
||||
P(c.onMenuShareAppMessage, { |
||||
complete: function(e) { |
||||
"favorite" === e.scene ? M("sendAppMessage", { |
||||
title: n.title || t, |
||||
desc: n.desc || "", |
||||
link: n.link || location.href, |
||||
img_url: n.imgUrl || "", |
||||
type: n.type || "link", |
||||
data_url: n.dataUrl || "" |
||||
}) : M("sendAppMessage", { |
||||
title: n.title || t, |
||||
desc: n.desc || "", |
||||
link: n.link || location.href, |
||||
img_url: n.imgUrl || "", |
||||
type: n.type || "link", |
||||
data_url: n.dataUrl || "" |
||||
}, n) |
||||
} |
||||
}, n) |
||||
}, |
||||
onMenuShareQQ: function(e) { |
||||
P(c.onMenuShareQQ, { |
||||
complete: function() { |
||||
M("shareQQ", { |
||||
title: e.title || t, |
||||
desc: e.desc || "", |
||||
img_url: e.imgUrl || "", |
||||
link: e.link || location.href |
||||
}, e) |
||||
} |
||||
}, e) |
||||
}, |
||||
onMenuShareWeibo: function(e) { |
||||
P(c.onMenuShareWeibo, { |
||||
complete: function() { |
||||
M("shareWeiboApp", { |
||||
title: e.title || t, |
||||
desc: e.desc || "", |
||||
img_url: e.imgUrl || "", |
||||
link: e.link || location.href |
||||
}, e) |
||||
} |
||||
}, e) |
||||
}, |
||||
onMenuShareQZone: function(e) { |
||||
P(c.onMenuShareQZone, { |
||||
complete: function() { |
||||
M("shareQZone", { |
||||
title: e.title || t, |
||||
desc: e.desc || "", |
||||
img_url: e.imgUrl || "", |
||||
link: e.link || location.href |
||||
}, e) |
||||
} |
||||
}, e) |
||||
}, |
||||
updateTimelineShareData: function(e) { |
||||
M("updateTimelineShareData", { |
||||
title: e.title, |
||||
link: e.link, |
||||
imgUrl: e.imgUrl |
||||
}, e) |
||||
}, |
||||
updateAppMessageShareData: function(e) { |
||||
M("updateAppMessageShareData", { |
||||
title: e.title, |
||||
desc: e.desc, |
||||
link: e.link, |
||||
imgUrl: e.imgUrl |
||||
}, e) |
||||
}, |
||||
startRecord: function(e) { |
||||
M("startRecord", {}, e) |
||||
}, |
||||
stopRecord: function(e) { |
||||
M("stopRecord", {}, e) |
||||
}, |
||||
onVoiceRecordEnd: function(e) { |
||||
P("onVoiceRecordEnd", e) |
||||
}, |
||||
playVoice: function(e) { |
||||
M("playVoice", { |
||||
localId: e.localId |
||||
}, e) |
||||
}, |
||||
pauseVoice: function(e) { |
||||
M("pauseVoice", { |
||||
localId: e.localId |
||||
}, e) |
||||
}, |
||||
stopVoice: function(e) { |
||||
M("stopVoice", { |
||||
localId: e.localId |
||||
}, e) |
||||
}, |
||||
onVoicePlayEnd: function(e) { |
||||
P("onVoicePlayEnd", e) |
||||
}, |
||||
uploadVoice: function(e) { |
||||
M("uploadVoice", { |
||||
localId: e.localId, |
||||
isShowProgressTips: 0 == e.isShowProgressTips ? 0 : 1 |
||||
}, e) |
||||
}, |
||||
downloadVoice: function(e) { |
||||
M("downloadVoice", { |
||||
serverId: e.serverId, |
||||
isShowProgressTips: 0 == e.isShowProgressTips ? 0 : 1 |
||||
}, e) |
||||
}, |
||||
translateVoice: function(e) { |
||||
M("translateVoice", { |
||||
localId: e.localId, |
||||
isShowProgressTips: 0 == e.isShowProgressTips ? 0 : 1 |
||||
}, e) |
||||
}, |
||||
chooseImage: function(e) { |
||||
M("chooseImage", { |
||||
scene: "1|2", |
||||
count: e.count || 9, |
||||
sizeType: e.sizeType || ["original", "compressed"], |
||||
sourceType: e.sourceType || ["album", "camera"] |
||||
}, (e._complete = function(e) { |
||||
if (p) { |
||||
var n = e.localIds; |
||||
try { |
||||
n && (e.localIds = JSON.parse(n)) |
||||
} catch (e) {} |
||||
} |
||||
}, e)) |
||||
}, |
||||
getLocation: function(e) {}, |
||||
previewImage: function(e) { |
||||
M(c.previewImage, { |
||||
current: e.current, |
||||
urls: e.urls |
||||
}, e) |
||||
}, |
||||
uploadImage: function(e) { |
||||
M("uploadImage", { |
||||
localId: e.localId, |
||||
isShowProgressTips: 0 == e.isShowProgressTips ? 0 : 1 |
||||
}, e) |
||||
}, |
||||
downloadImage: function(e) { |
||||
M("downloadImage", { |
||||
serverId: e.serverId, |
||||
isShowProgressTips: 0 == e.isShowProgressTips ? 0 : 1 |
||||
}, e) |
||||
}, |
||||
getLocalImgData: function(e) { |
||||
!1 === I ? (I = !0, M("getLocalImgData", { |
||||
localId: e.localId |
||||
}, (e._complete = function(e) { |
||||
if (I = !1, 0 < _.length) { |
||||
var n = _.shift(); |
||||
wx.getLocalImgData(n) |
||||
} |
||||
}, e))) : _.push(e) |
||||
}, |
||||
getNetworkType: function(e) { |
||||
M("getNetworkType", {}, (e._complete = function(e) { |
||||
e = function(e) { |
||||
var n = e.errMsg; |
||||
e.errMsg = "getNetworkType:ok"; |
||||
var i = e.subtype; |
||||
if (delete e.subtype, i) e.networkType = i; |
||||
else { |
||||
var t = n.indexOf(":"), |
||||
o = n.substring(t + 1); |
||||
switch (o) { |
||||
case "wifi": |
||||
case "edge": |
||||
case "wwan": |
||||
e.networkType = o; |
||||
break; |
||||
default: |
||||
e.errMsg = "getNetworkType:fail" |
||||
} |
||||
} |
||||
return e |
||||
}(e) |
||||
}, e)) |
||||
}, |
||||
openLocation: function(e) { |
||||
M("openLocation", { |
||||
latitude: e.latitude, |
||||
longitude: e.longitude, |
||||
name: e.name || "", |
||||
address: e.address || "", |
||||
scale: e.scale || 28, |
||||
infoUrl: e.infoUrl || "" |
||||
}, e) |
||||
}, |
||||
getLocation: function(e) { |
||||
M(c.getLocation, { |
||||
type: (e = e || {}).type || "wgs84" |
||||
}, (e._complete = function(e) { |
||||
delete e.type |
||||
}, e)) |
||||
}, |
||||
hideOptionMenu: function(e) { |
||||
M("hideOptionMenu", {}, e) |
||||
}, |
||||
showOptionMenu: function(e) { |
||||
M("showOptionMenu", {}, e) |
||||
}, |
||||
closeWindow: function(e) { |
||||
M("closeWindow", {}, e = e || {}) |
||||
}, |
||||
hideMenuItems: function(e) { |
||||
M("hideMenuItems", { |
||||
menuList: e.menuList |
||||
}, e) |
||||
}, |
||||
showMenuItems: function(e) { |
||||
M("showMenuItems", { |
||||
menuList: e.menuList |
||||
}, e) |
||||
}, |
||||
hideAllNonBaseMenuItem: function(e) { |
||||
M("hideAllNonBaseMenuItem", {}, e) |
||||
}, |
||||
showAllNonBaseMenuItem: function(e) { |
||||
M("showAllNonBaseMenuItem", {}, e) |
||||
}, |
||||
scanQRCode: function(e) { |
||||
M("scanQRCode", { |
||||
needResult: (e = e || {}).needResult || 0, |
||||
scanType: e.scanType || ["qrCode", "barCode"] |
||||
}, (e._complete = function(e) { |
||||
if (f) { |
||||
var n = e.resultStr; |
||||
if (n) { |
||||
var i = JSON.parse(n); |
||||
e.resultStr = i && i.scan_code && i.scan_code.scan_result |
||||
} |
||||
} |
||||
}, e)) |
||||
}, |
||||
openAddress: function(e) { |
||||
M(c.openAddress, {}, (e._complete = function(e) { |
||||
e = function(e) { |
||||
return e.postalCode = e.addressPostalCode, delete e.addressPostalCode, e.provinceName = e.proviceFirstStageName, delete e.proviceFirstStageName, e.cityName = e.addressCitySecondStageName, delete e.addressCitySecondStageName, e.countryName = e.addressCountiesThirdStageName, delete e.addressCountiesThirdStageName, e.detailInfo = e.addressDetailInfo, delete e.addressDetailInfo, e |
||||
}(e) |
||||
}, e)) |
||||
}, |
||||
openProductSpecificView: function(e) { |
||||
M(c.openProductSpecificView, { |
||||
pid: e.productId, |
||||
view_type: e.viewType || 0, |
||||
ext_info: e.extInfo |
||||
}, e) |
||||
}, |
||||
addCard: function(e) { |
||||
for (var n = e.cardList, i = [], t = 0, o = n.length; t < o; ++t) { |
||||
var r = n[t], |
||||
a = { |
||||
card_id: r.cardId, |
||||
card_ext: r.cardExt |
||||
}; |
||||
i.push(a) |
||||
} |
||||
M(c.addCard, { |
||||
card_list: i |
||||
}, (e._complete = function(e) { |
||||
var n = e.card_list; |
||||
if (n) { |
||||
for (var i = 0, t = (n = JSON.parse(n)).length; i < t; ++i) { |
||||
var o = n[i]; |
||||
o.cardId = o.card_id, o.cardExt = o.card_ext, o.isSuccess = !!o.is_succ, delete o.card_id, delete o.card_ext, delete o.is_succ |
||||
} |
||||
e.cardList = n, delete e.card_list |
||||
} |
||||
}, e)) |
||||
}, |
||||
chooseCard: function(e) { |
||||
M("chooseCard", { |
||||
app_id: v.appId, |
||||
location_id: e.shopId || "", |
||||
sign_type: e.signType || "SHA1", |
||||
card_id: e.cardId || "", |
||||
card_type: e.cardType || "", |
||||
card_sign: e.cardSign, |
||||
time_stamp: e.timestamp + "", |
||||
nonce_str: e.nonceStr |
||||
}, (e._complete = function(e) { |
||||
e.cardList = e.choose_card_info, delete e.choose_card_info |
||||
}, e)) |
||||
}, |
||||
openCard: function(e) { |
||||
for (var n = e.cardList, i = [], t = 0, o = n.length; t < o; ++t) { |
||||
var r = n[t], |
||||
a = { |
||||
card_id: r.cardId, |
||||
code: r.code |
||||
}; |
||||
i.push(a) |
||||
} |
||||
M(c.openCard, { |
||||
card_list: i |
||||
}, e) |
||||
}, |
||||
consumeAndShareCard: function(e) { |
||||
M(c.consumeAndShareCard, { |
||||
consumedCardId: e.cardId, |
||||
consumedCode: e.code |
||||
}, e) |
||||
}, |
||||
chooseWXPay: function(e) { |
||||
M(c.chooseWXPay, V(e), e) |
||||
}, |
||||
openEnterpriseRedPacket: function(e) { |
||||
M(c.openEnterpriseRedPacket, V(e), e) |
||||
}, |
||||
startSearchBeacons: function(e) { |
||||
M(c.startSearchBeacons, { |
||||
ticket: e.ticket |
||||
}, e) |
||||
}, |
||||
stopSearchBeacons: function(e) { |
||||
M(c.stopSearchBeacons, {}, e) |
||||
}, |
||||
onSearchBeacons: function(e) { |
||||
P(c.onSearchBeacons, e) |
||||
}, |
||||
openEnterpriseChat: function(e) { |
||||
M("openEnterpriseChat", { |
||||
useridlist: e.userIds, |
||||
chatname: e.groupName |
||||
}, e) |
||||
}, |
||||
launchMiniProgram: function(e) { |
||||
M("launchMiniProgram", { |
||||
targetAppId: e.targetAppId, |
||||
path: function(e) { |
||||
if ("string" == typeof e && 0 < e.length) { |
||||
var n = e.split("?")[0], |
||||
i = e.split("?")[1]; |
||||
return n += ".html", void 0 !== i ? n + "?" + i : n |
||||
} |
||||
}(e.path), |
||||
envVersion: e.envVersion |
||||
}, e) |
||||
}, |
||||
openBusinessView: function(e) { |
||||
M("openBusinessView", { |
||||
businessType: e.businessType, |
||||
queryString: e.queryString || "", |
||||
envVersion: e.envVersion |
||||
}, (e._complete = function(n) { |
||||
if (p) { |
||||
var e = n.extraData; |
||||
if (e) try { |
||||
n.extraData = JSON.parse(e) |
||||
} catch (e) { |
||||
n.extraData = {} |
||||
} |
||||
} |
||||
}, e)) |
||||
}, |
||||
miniProgram: { |
||||
navigateBack: function(e) { |
||||
e = e || {}, O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "navigateBack", |
||||
arg: { |
||||
delta: e.delta || 1 |
||||
} |
||||
}, e) |
||||
}) |
||||
}, |
||||
navigateTo: function(e) { |
||||
O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "navigateTo", |
||||
arg: { |
||||
url: e.url |
||||
} |
||||
}, e) |
||||
}) |
||||
}, |
||||
redirectTo: function(e) { |
||||
O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "redirectTo", |
||||
arg: { |
||||
url: e.url |
||||
} |
||||
}, e) |
||||
}) |
||||
}, |
||||
switchTab: function(e) { |
||||
O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "switchTab", |
||||
arg: { |
||||
url: e.url |
||||
} |
||||
}, e) |
||||
}) |
||||
}, |
||||
reLaunch: function(e) { |
||||
O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "reLaunch", |
||||
arg: { |
||||
url: e.url |
||||
} |
||||
}, e) |
||||
}) |
||||
}, |
||||
postMessage: function(e) { |
||||
O(function() { |
||||
M("invokeMiniProgramAPI", { |
||||
name: "postMessage", |
||||
arg: e.data || {} |
||||
}, e) |
||||
}) |
||||
}, |
||||
getEnv: function(e) { |
||||
O(function() { |
||||
e({ |
||||
miniprogram: "miniprogram" === o.__wxjs_environment |
||||
}) |
||||
}) |
||||
} |
||||
} |
||||
}, |
||||
T = 1, |
||||
k = {}; |
||||
return i.addEventListener("error", function(e) { |
||||
if (!p) { |
||||
var n = e.target, |
||||
i = n.tagName, |
||||
t = n.src; |
||||
if ("IMG" == i || "VIDEO" == i || "AUDIO" == i || "SOURCE" == i) |
||||
if (-1 != t.indexOf("wxlocalresource://")) { |
||||
e.preventDefault(), e.stopPropagation(); |
||||
var o = n["wx-id"]; |
||||
if (o || (o = T++, n["wx-id"] = o), k[o]) return; |
||||
k[o] = !0, wx.ready(function() { |
||||
wx.getLocalImgData({ |
||||
localId: t, |
||||
success: function(e) { |
||||
n.src = e.localData |
||||
} |
||||
}) |
||||
}) |
||||
} |
||||
} |
||||
}, !0), i.addEventListener("load", function(e) { |
||||
if (!p) { |
||||
var n = e.target, |
||||
i = n.tagName; |
||||
n.src; |
||||
if ("IMG" == i || "VIDEO" == i || "AUDIO" == i || "SOURCE" == i) { |
||||
var t = n["wx-id"]; |
||||
t && (k[t] = !1) |
||||
} |
||||
} |
||||
}, !0), e && (o.wx = o.jWeixin = w), w |
||||
} |
||||
|
||||
function M(n, e, i) { |
||||
o.WeixinJSBridge ? WeixinJSBridge.invoke(n, x(e), function(e) { |
||||
A(n, e, i) |
||||
}) : B(n, i) |
||||
} |
||||
|
||||
function P(n, i, t) { |
||||
o.WeixinJSBridge ? WeixinJSBridge.on(n, function(e) { |
||||
t && t.trigger && t.trigger(e), A(n, e, i) |
||||
}) : B(n, t || i) |
||||
} |
||||
|
||||
function x(e) { |
||||
return (e = e || {}).appId = v.appId, e.verifyAppId = v.appId, e.verifySignType = "sha1", e.verifyTimestamp = v.timestamp + "", e.verifyNonceStr = v.nonceStr, e.verifySignature = v.signature, e |
||||
} |
||||
|
||||
function V(e) { |
||||
return { |
||||
timeStamp: e.timestamp + "", |
||||
nonceStr: e.nonceStr, |
||||
package: e.package, |
||||
paySign: e.paySign, |
||||
signType: e.signType || "SHA1" |
||||
} |
||||
} |
||||
|
||||
function A(e, n, i) { |
||||
"openEnterpriseChat" != e && "openBusinessView" !== e || (n.errCode = n.err_code), delete n.err_code, delete n.err_desc, delete n.err_detail; |
||||
var t = n.errMsg; |
||||
t || (t = n.err_msg, delete n.err_msg, t = function(e, n) { |
||||
var i = e, |
||||
t = a[i]; |
||||
t && (i = t); |
||||
var o = "ok"; |
||||
if (n) { |
||||
var r = n.indexOf(":"); |
||||
"confirm" == (o = n.substring(r + 1)) && (o = "ok"), "failed" == o && (o = "fail"), -1 != o.indexOf("failed_") && (o = o.substring(7)), -1 != o.indexOf("fail_") && (o = o.substring(5)), "access denied" != (o = (o = o.replace(/_/g, " ")).toLowerCase()) && "no permission to execute" != o || (o = "permission denied"), "config" == i && "function not exist" == o && (o = "ok"), "" == o && (o = "fail") |
||||
} |
||||
return n = i + ":" + o |
||||
}(e, t), n.errMsg = t), (i = i || {})._complete && (i._complete(n), delete i._complete), t = n.errMsg || "", v.debug && !i.isInnerInvoke && alert(JSON.stringify(n)); |
||||
var o = t.indexOf(":"); |
||||
switch (t.substring(o + 1)) { |
||||
case "ok": |
||||
i.success && i.success(n); |
||||
break; |
||||
case "cancel": |
||||
i.cancel && i.cancel(n); |
||||
break; |
||||
default: |
||||
i.fail && i.fail(n) |
||||
} |
||||
i.complete && i.complete(n) |
||||
} |
||||
|
||||
function C(e) { |
||||
if (e) { |
||||
for (var n = 0, i = e.length; n < i; ++n) { |
||||
var t = e[n], |
||||
o = c[t]; |
||||
o && (e[n] = o) |
||||
} |
||||
return e |
||||
} |
||||
} |
||||
|
||||
function B(e, n) { |
||||
if (!(!v.debug || n && n.isInnerInvoke)) { |
||||
var i = a[e]; |
||||
i && (e = i), n && n._complete && delete n._complete, console.log('"' + e + '",', n || "") |
||||
} |
||||
} |
||||
|
||||
function L() { |
||||
return (new Date).getTime() |
||||
} |
||||
|
||||
function O(e) { |
||||
l && (o.WeixinJSBridge ? e() : i.addEventListener && i.addEventListener("WeixinJSBridgeReady", e, !1)) |
||||
} |
||||
}); |
@ -0,0 +1,31 @@ |
||||
function AMapWX(a){this.key=a.key;this.requestConfig={key:a.key,s:"rsx",platform:"WXJS",appname:a.key,sdkversion:"1.2.0",logversion:"2.0"};this.MeRequestConfig={key:a.key,serviceName:"https://restapi.amap.com/rest/me"}} |
||||
AMapWX.prototype.getWxLocation=function(a,b){wx.getLocation({type:"gcj02",success:function(c){c=c.longitude+","+c.latitude;wx.setStorage({key:"userLocation",data:c});b(c)},fail:function(c){wx.getStorage({key:"userLocation",success:function(d){d.data&&b(d.data)}});a.fail({errCode:"0",errMsg:c.errMsg||""})}})}; |
||||
AMapWX.prototype.getMEKeywordsSearch=function(a){if(!a.options)return a.fail({errCode:"0",errMsg:"\u7f3a\u5c11\u5fc5\u8981\u53c2\u6570"});var b=a.options,c=this.MeRequestConfig,d={key:c.key,s:"rsx",platform:"WXJS",appname:a.key,sdkversion:"1.2.0",logversion:"2.0"};b.layerId&&(d.layerId=b.layerId);b.keywords&&(d.keywords=b.keywords);b.city&&(d.city=b.city);b.filter&&(d.filter=b.filter);b.sortrule&&(d.sortrule=b.sortrule);b.pageNum&&(d.pageNum=b.pageNum);b.pageSize&&(d.pageSize=b.pageSize);b.sig&&(d.sig= |
||||
b.sig);wx.request({url:c.serviceName+"/cpoint/datasearch/local",data:d,method:"GET",header:{"content-type":"application/json"},success:function(e){(e=e.data)&&e.status&&"1"===e.status&&0===e.code?a.success(e.data):a.fail({errCode:"0",errMsg:e})},fail:function(e){a.fail({errCode:"0",errMsg:e.errMsg||""})}})}; |
||||
AMapWX.prototype.getMEIdSearch=function(a){if(!a.options)return a.fail({errCode:"0",errMsg:"\u7f3a\u5c11\u5fc5\u8981\u53c2\u6570"});var b=a.options,c=this.MeRequestConfig,d={key:c.key,s:"rsx",platform:"WXJS",appname:a.key,sdkversion:"1.2.0",logversion:"2.0"};b.layerId&&(d.layerId=b.layerId);b.id&&(d.id=b.id);b.sig&&(d.sig=b.sig);wx.request({url:c.serviceName+"/cpoint/datasearch/id",data:d,method:"GET",header:{"content-type":"application/json"},success:function(e){(e=e.data)&&e.status&&"1"===e.status&& |
||||
0===e.code?a.success(e.data):a.fail({errCode:"0",errMsg:e})},fail:function(e){a.fail({errCode:"0",errMsg:e.errMsg||""})}})}; |
||||
AMapWX.prototype.getMEPolygonSearch=function(a){if(!a.options)return a.fail({errCode:"0",errMsg:"\u7f3a\u5c11\u5fc5\u8981\u53c2\u6570"});var b=a.options,c=this.MeRequestConfig,d={key:c.key,s:"rsx",platform:"WXJS",appname:a.key,sdkversion:"1.2.0",logversion:"2.0"};b.layerId&&(d.layerId=b.layerId);b.keywords&&(d.keywords=b.keywords);b.polygon&&(d.polygon=b.polygon);b.filter&&(d.filter=b.filter);b.sortrule&&(d.sortrule=b.sortrule);b.pageNum&&(d.pageNum=b.pageNum);b.pageSize&&(d.pageSize=b.pageSize); |
||||
b.sig&&(d.sig=b.sig);wx.request({url:c.serviceName+"/cpoint/datasearch/polygon",data:d,method:"GET",header:{"content-type":"application/json"},success:function(e){(e=e.data)&&e.status&&"1"===e.status&&0===e.code?a.success(e.data):a.fail({errCode:"0",errMsg:e})},fail:function(e){a.fail({errCode:"0",errMsg:e.errMsg||""})}})}; |
||||
AMapWX.prototype.getMEaroundSearch=function(a){if(!a.options)return a.fail({errCode:"0",errMsg:"\u7f3a\u5c11\u5fc5\u8981\u53c2\u6570"});var b=a.options,c=this.MeRequestConfig,d={key:c.key,s:"rsx",platform:"WXJS",appname:a.key,sdkversion:"1.2.0",logversion:"2.0"};b.layerId&&(d.layerId=b.layerId);b.keywords&&(d.keywords=b.keywords);b.center&&(d.center=b.center);b.radius&&(d.radius=b.radius);b.filter&&(d.filter=b.filter);b.sortrule&&(d.sortrule=b.sortrule);b.pageNum&&(d.pageNum=b.pageNum);b.pageSize&& |
||||
(d.pageSize=b.pageSize);b.sig&&(d.sig=b.sig);wx.request({url:c.serviceName+"/cpoint/datasearch/around",data:d,method:"GET",header:{"content-type":"application/json"},success:function(e){(e=e.data)&&e.status&&"1"===e.status&&0===e.code?a.success(e.data):a.fail({errCode:"0",errMsg:e})},fail:function(e){a.fail({errCode:"0",errMsg:e.errMsg||""})}})}; |
||||
AMapWX.prototype.getGeo=function(a){var b=this.requestConfig,c=a.options;b={key:this.key,extensions:"all",s:b.s,platform:b.platform,appname:this.key,sdkversion:b.sdkversion,logversion:b.logversion};c.address&&(b.address=c.address);c.city&&(b.city=c.city);c.batch&&(b.batch=c.batch);c.sig&&(b.sig=c.sig);wx.request({url:"https://restapi.amap.com/v3/geocode/geo",data:b,method:"GET",header:{"content-type":"application/json"},success:function(d){(d=d.data)&&d.status&&"1"===d.status?a.success(d):a.fail({errCode:"0", |
||||
errMsg:d})},fail:function(d){a.fail({errCode:"0",errMsg:d.errMsg||""})}})}; |
||||
AMapWX.prototype.getRegeo=function(a){function b(d){var e=c.requestConfig;wx.request({url:"https://restapi.amap.com/v3/geocode/regeo",data:{key:c.key,location:d,extensions:"all",s:e.s,platform:e.platform,appname:c.key,sdkversion:e.sdkversion,logversion:e.logversion},method:"GET",header:{"content-type":"application/json"},success:function(g){if(g.data.status&&"1"==g.data.status){g=g.data.regeocode;var h=g.addressComponent,f=[],k=g.roads[0].name+"\u9644\u8fd1",m=d.split(",")[0],n=d.split(",")[1];if(g.pois&& |
||||
g.pois[0]){k=g.pois[0].name+"\u9644\u8fd1";var l=g.pois[0].location;l&&(m=parseFloat(l.split(",")[0]),n=parseFloat(l.split(",")[1]))}h.provice&&f.push(h.provice);h.city&&f.push(h.city);h.district&&f.push(h.district);h.streetNumber&&h.streetNumber.street&&h.streetNumber.number?(f.push(h.streetNumber.street),f.push(h.streetNumber.number)):f.push(g.roads[0].name);f=f.join("");a.success([{iconPath:a.iconPath,width:a.iconWidth,height:a.iconHeight,name:f,desc:k,longitude:m,latitude:n,id:0,regeocodeData:g}])}else a.fail({errCode:g.data.infocode, |
||||
errMsg:g.data.info})},fail:function(g){a.fail({errCode:"0",errMsg:g.errMsg||""})}})}var c=this;a.location?b(a.location):c.getWxLocation(a,function(d){b(d)})}; |
||||
AMapWX.prototype.getWeather=function(a){function b(g){var h="base";a.type&&"forecast"==a.type&&(h="all");wx.request({url:"https://restapi.amap.com/v3/weather/weatherInfo",data:{key:d.key,city:g,extensions:h,s:e.s,platform:e.platform,appname:d.key,sdkversion:e.sdkversion,logversion:e.logversion},method:"GET",header:{"content-type":"application/json"},success:function(f){if(f.data.status&&"1"==f.data.status)if(f.data.lives){if((f=f.data.lives)&&0<f.length){f=f[0];var k={city:{text:"\u57ce\u5e02",data:f.city}, |
||||
weather:{text:"\u5929\u6c14",data:f.weather},temperature:{text:"\u6e29\u5ea6",data:f.temperature},winddirection:{text:"\u98ce\u5411",data:f.winddirection+"\u98ce"},windpower:{text:"\u98ce\u529b",data:f.windpower+"\u7ea7"},humidity:{text:"\u6e7f\u5ea6",data:f.humidity+"%"}};k.liveData=f;a.success(k)}}else f.data.forecasts&&f.data.forecasts[0]&&a.success({forecast:f.data.forecasts[0]});else a.fail({errCode:f.data.infocode,errMsg:f.data.info})},fail:function(f){a.fail({errCode:"0",errMsg:f.errMsg||""})}})} |
||||
function c(g){wx.request({url:"https://restapi.amap.com/v3/geocode/regeo",data:{key:d.key,location:g,extensions:"all",s:e.s,platform:e.platform,appname:d.key,sdkversion:e.sdkversion,logversion:e.logversion},method:"GET",header:{"content-type":"application/json"},success:function(h){if(h.data.status&&"1"==h.data.status){h=h.data.regeocode;if(h.addressComponent)var f=h.addressComponent.adcode;else h.aois&&0<h.aois.length&&(f=h.aois[0].adcode);b(f)}else a.fail({errCode:h.data.infocode,errMsg:h.data.info})}, |
||||
fail:function(h){a.fail({errCode:"0",errMsg:h.errMsg||""})}})}var d=this,e=d.requestConfig;a.city?b(a.city):d.getWxLocation(a,function(g){c(g)})}; |
||||
AMapWX.prototype.getPoiAround=function(a){function b(e){e={key:c.key,location:e,s:d.s,platform:d.platform,appname:c.key,sdkversion:d.sdkversion,logversion:d.logversion};a.querytypes&&(e.types=a.querytypes);a.querykeywords&&(e.keywords=a.querykeywords);wx.request({url:"https://restapi.amap.com/v3/place/around",data:e,method:"GET",header:{"content-type":"application/json"},success:function(g){if(g.data.status&&"1"==g.data.status){if((g=g.data)&&g.pois){for(var h=[],f=0;f<g.pois.length;f++){var k=0== |
||||
f?a.iconPathSelected:a.iconPath;h.push({latitude:parseFloat(g.pois[f].location.split(",")[1]),longitude:parseFloat(g.pois[f].location.split(",")[0]),iconPath:k,width:22,height:32,id:f,name:g.pois[f].name,address:g.pois[f].address})}a.success({markers:h,poisData:g.pois})}}else a.fail({errCode:g.data.infocode,errMsg:g.data.info})},fail:function(g){a.fail({errCode:"0",errMsg:g.errMsg||""})}})}var c=this,d=c.requestConfig;a.location?b(a.location):c.getWxLocation(a,function(e){b(e)})}; |
||||
AMapWX.prototype.getStaticmap=function(a){function b(e){c.push("location="+e);a.zoom&&c.push("zoom="+a.zoom);a.size&&c.push("size="+a.size);a.scale&&c.push("scale="+a.scale);a.markers&&c.push("markers="+a.markers);a.labels&&c.push("labels="+a.labels);a.paths&&c.push("paths="+a.paths);a.traffic&&c.push("traffic="+a.traffic);e="https://restapi.amap.com/v3/staticmap?"+c.join("&");a.success({url:e})}var c=[];c.push("key="+this.key);var d=this.requestConfig;c.push("s="+d.s);c.push("platform="+d.platform); |
||||
c.push("appname="+d.appname);c.push("sdkversion="+d.sdkversion);c.push("logversion="+d.logversion);a.location?b(a.location):this.getWxLocation(a,function(e){b(e)})}; |
||||
AMapWX.prototype.getInputtips=function(a){var b=Object.assign({},this.requestConfig);a.location&&(b.location=a.location);a.keywords&&(b.keywords=a.keywords);a.type&&(b.type=a.type);a.city&&(b.city=a.city);a.citylimit&&(b.citylimit=a.citylimit);wx.request({url:"https://restapi.amap.com/v3/assistant/inputtips",data:b,method:"GET",header:{"content-type":"application/json"},success:function(c){c&&c.data&&c.data.tips&&a.success({tips:c.data.tips})},fail:function(c){a.fail({errCode:"0",errMsg:c.errMsg|| |
||||
""})}})}; |
||||
AMapWX.prototype.getDrivingRoute=function(a){var b=Object.assign({},this.requestConfig);a.origin&&(b.origin=a.origin);a.destination&&(b.destination=a.destination);a.strategy&&(b.strategy=a.strategy);a.waypoints&&(b.waypoints=a.waypoints);a.avoidpolygons&&(b.avoidpolygons=a.avoidpolygons);a.avoidroad&&(b.avoidroad=a.avoidroad);wx.request({url:"https://restapi.amap.com/v3/direction/driving",data:b,method:"GET",header:{"content-type":"application/json"},success:function(c){c&&c.data&&c.data.route&&a.success({paths:c.data.route.paths, |
||||
taxi_cost:c.data.route.taxi_cost||""})},fail:function(c){a.fail({errCode:"0",errMsg:c.errMsg||""})}})}; |
||||
AMapWX.prototype.getWalkingRoute=function(a){var b=Object.assign({},this.requestConfig);a.origin&&(b.origin=a.origin);a.destination&&(b.destination=a.destination);wx.request({url:"https://restapi.amap.com/v3/direction/walking",data:b,method:"GET",header:{"content-type":"application/json"},success:function(c){c&&c.data&&c.data.route&&a.success({paths:c.data.route.paths})},fail:function(c){a.fail({errCode:"0",errMsg:c.errMsg||""})}})}; |
||||
AMapWX.prototype.getTransitRoute=function(a){var b=Object.assign({},this.requestConfig);a.origin&&(b.origin=a.origin);a.destination&&(b.destination=a.destination);a.strategy&&(b.strategy=a.strategy);a.city&&(b.city=a.city);a.cityd&&(b.cityd=a.cityd);wx.request({url:"https://restapi.amap.com/v3/direction/transit/integrated",data:b,method:"GET",header:{"content-type":"application/json"},success:function(c){c&&c.data&&c.data.route&&(c=c.data.route,a.success({distance:c.distance||"",taxi_cost:c.taxi_cost|| |
||||
"",transits:c.transits}))},fail:function(c){a.fail({errCode:"0",errMsg:c.errMsg||""})}})}; |
||||
AMapWX.prototype.getRidingRoute=function(a){var b=Object.assign({},this.requestConfig);a.origin&&(b.origin=a.origin);a.destination&&(b.destination=a.destination);wx.request({url:"https://restapi.amap.com/v3/direction/riding",data:b,method:"GET",header:{"content-type":"application/json"},success:function(c){c&&c.data&&c.data.route&&a.success({paths:c.data.route.paths})},fail:function(c){a.fail({errCode:"0",errMsg:c.errMsg||""})}})};module.exports.AMapWX=AMapWX; |
@ -0,0 +1,257 @@ |
||||
/* |
||||
* A JavaScript implementation of the RSA Data Security, Inc. MD5 Message |
||||
* Digest Algorithm, as defined in RFC 1321. |
||||
* Version 2.1 Copyright (C) Paul Johnston 1999 - 2002. |
||||
* Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet |
||||
* Distributed under the BSD License |
||||
* See http://pajhome.org.uk/crypt/md5 for more info.
|
||||
*/ |
||||
|
||||
/* |
||||
* Configurable variables. You may need to tweak these to be compatible with |
||||
* the server-side, but the defaults work in most cases. |
||||
*/ |
||||
var hexcase = 0; /* hex output format. 0 - lowercase; 1 - uppercase */ |
||||
var b64pad = ""; /* base-64 pad character. "=" for strict RFC compliance */ |
||||
var chrsz = 8; /* bits per input character. 8 - ASCII; 16 - Unicode */ |
||||
|
||||
/* |
||||
* These are the functions you'll usually want to call |
||||
* They take string arguments and return either hex or base-64 encoded strings |
||||
*/ |
||||
function hex_md5(s){ return binl2hex(core_md5(str2binl(s), s.length * chrsz));} |
||||
function b64_md5(s){ return binl2b64(core_md5(str2binl(s), s.length * chrsz));} |
||||
function str_md5(s){ return binl2str(core_md5(str2binl(s), s.length * chrsz));} |
||||
function hex_hmac_md5(key, data) { return binl2hex(core_hmac_md5(key, data)); } |
||||
function b64_hmac_md5(key, data) { return binl2b64(core_hmac_md5(key, data)); } |
||||
function str_hmac_md5(key, data) { return binl2str(core_hmac_md5(key, data)); } |
||||
|
||||
/* |
||||
* Perform a simple self-test to see if the VM is working |
||||
*/ |
||||
function md5_vm_test() |
||||
{ |
||||
return hex_md5("abc") == "900150983cd24fb0d6963f7d28e17f72"; |
||||
} |
||||
|
||||
/* |
||||
* Calculate the MD5 of an array of little-endian words, and a bit length |
||||
*/ |
||||
function core_md5(x, len) |
||||
{ |
||||
/* append padding */ |
||||
x[len >> 5] |= 0x80 << ((len) % 32); |
||||
x[(((len + 64) >>> 9) << 4) + 14] = len; |
||||
|
||||
var a = 1732584193; |
||||
var b = -271733879; |
||||
var c = -1732584194; |
||||
var d = 271733878; |
||||
|
||||
for(var i = 0; i < x.length; i += 16) |
||||
{ |
||||
var olda = a; |
||||
var oldb = b; |
||||
var oldc = c; |
||||
var oldd = d; |
||||
|
||||
a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936); |
||||
d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586); |
||||
c = md5_ff(c, d, a, b, x[i+ 2], 17, 606105819); |
||||
b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330); |
||||
a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897); |
||||
d = md5_ff(d, a, b, c, x[i+ 5], 12, 1200080426); |
||||
c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341); |
||||
b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983); |
||||
a = md5_ff(a, b, c, d, x[i+ 8], 7 , 1770035416); |
||||
d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417); |
||||
c = md5_ff(c, d, a, b, x[i+10], 17, -42063); |
||||
b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162); |
||||
a = md5_ff(a, b, c, d, x[i+12], 7 , 1804603682); |
||||
d = md5_ff(d, a, b, c, x[i+13], 12, -40341101); |
||||
c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290); |
||||
b = md5_ff(b, c, d, a, x[i+15], 22, 1236535329); |
||||
|
||||
a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510); |
||||
d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632); |
||||
c = md5_gg(c, d, a, b, x[i+11], 14, 643717713); |
||||
b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302); |
||||
a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691); |
||||
d = md5_gg(d, a, b, c, x[i+10], 9 , 38016083); |
||||
c = md5_gg(c, d, a, b, x[i+15], 14, -660478335); |
||||
b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848); |
||||
a = md5_gg(a, b, c, d, x[i+ 9], 5 , 568446438); |
||||
d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690); |
||||
c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961); |
||||
b = md5_gg(b, c, d, a, x[i+ 8], 20, 1163531501); |
||||
a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467); |
||||
d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784); |
||||
c = md5_gg(c, d, a, b, x[i+ 7], 14, 1735328473); |
||||
b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734); |
||||
|
||||
a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558); |
||||
d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463); |
||||
c = md5_hh(c, d, a, b, x[i+11], 16, 1839030562); |
||||
b = md5_hh(b, c, d, a, x[i+14], 23, -35309556); |
||||
a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060); |
||||
d = md5_hh(d, a, b, c, x[i+ 4], 11, 1272893353); |
||||
c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632); |
||||
b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640); |
||||
a = md5_hh(a, b, c, d, x[i+13], 4 , 681279174); |
||||
d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222); |
||||
c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979); |
||||
b = md5_hh(b, c, d, a, x[i+ 6], 23, 76029189); |
||||
a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487); |
||||
d = md5_hh(d, a, b, c, x[i+12], 11, -421815835); |
||||
c = md5_hh(c, d, a, b, x[i+15], 16, 530742520); |
||||
b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651); |
||||
|
||||
a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844); |
||||
d = md5_ii(d, a, b, c, x[i+ 7], 10, 1126891415); |
||||
c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905); |
||||
b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055); |
||||
a = md5_ii(a, b, c, d, x[i+12], 6 , 1700485571); |
||||
d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606); |
||||
c = md5_ii(c, d, a, b, x[i+10], 15, -1051523); |
||||
b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799); |
||||
a = md5_ii(a, b, c, d, x[i+ 8], 6 , 1873313359); |
||||
d = md5_ii(d, a, b, c, x[i+15], 10, -30611744); |
||||
c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380); |
||||
b = md5_ii(b, c, d, a, x[i+13], 21, 1309151649); |
||||
a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070); |
||||
d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379); |
||||
c = md5_ii(c, d, a, b, x[i+ 2], 15, 718787259); |
||||
b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551); |
||||
|
||||
a = safe_add(a, olda); |
||||
b = safe_add(b, oldb); |
||||
c = safe_add(c, oldc); |
||||
d = safe_add(d, oldd); |
||||
} |
||||
return Array(a, b, c, d); |
||||
|
||||
} |
||||
|
||||
/* |
||||
* These functions implement the four basic operations the algorithm uses. |
||||
*/ |
||||
function md5_cmn(q, a, b, x, s, t) |
||||
{ |
||||
return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b); |
||||
} |
||||
function md5_ff(a, b, c, d, x, s, t) |
||||
{ |
||||
return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t); |
||||
} |
||||
function md5_gg(a, b, c, d, x, s, t) |
||||
{ |
||||
return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t); |
||||
} |
||||
function md5_hh(a, b, c, d, x, s, t) |
||||
{ |
||||
return md5_cmn(b ^ c ^ d, a, b, x, s, t); |
||||
} |
||||
function md5_ii(a, b, c, d, x, s, t) |
||||
{ |
||||
return md5_cmn(c ^ (b | (~d)), a, b, x, s, t); |
||||
} |
||||
|
||||
/* |
||||
* Calculate the HMAC-MD5, of a key and some data |
||||
*/ |
||||
function core_hmac_md5(key, data) |
||||
{ |
||||
var bkey = str2binl(key); |
||||
if(bkey.length > 16) bkey = core_md5(bkey, key.length * chrsz); |
||||
|
||||
var ipad = Array(16), opad = Array(16); |
||||
for(var i = 0; i < 16; i++) |
||||
{ |
||||
ipad[i] = bkey[i] ^ 0x36363636; |
||||
opad[i] = bkey[i] ^ 0x5C5C5C5C; |
||||
} |
||||
|
||||
var hash = core_md5(ipad.concat(str2binl(data)), 512 + data.length * chrsz); |
||||
return core_md5(opad.concat(hash), 512 + 128); |
||||
} |
||||
|
||||
/* |
||||
* Add integers, wrapping at 2^32. This uses 16-bit operations internally |
||||
* to work around bugs in some JS interpreters. |
||||
*/ |
||||
function safe_add(x, y) |
||||
{ |
||||
var lsw = (x & 0xFFFF) + (y & 0xFFFF); |
||||
var msw = (x >> 16) + (y >> 16) + (lsw >> 16); |
||||
return (msw << 16) | (lsw & 0xFFFF); |
||||
} |
||||
|
||||
/* |
||||
* Bitwise rotate a 32-bit number to the left. |
||||
*/ |
||||
function bit_rol(num, cnt) |
||||
{ |
||||
return (num << cnt) | (num >>> (32 - cnt)); |
||||
} |
||||
|
||||
/* |
||||
* Convert a string to an array of little-endian words |
||||
* If chrsz is ASCII, characters >255 have their hi-byte silently ignored. |
||||
*/ |
||||
function str2binl(str) |
||||
{ |
||||
var bin = Array(); |
||||
var mask = (1 << chrsz) - 1; |
||||
for(var i = 0; i < str.length * chrsz; i += chrsz) |
||||
bin[i>>5] |= (str.charCodeAt(i / chrsz) & mask) << (i%32); |
||||
return bin; |
||||
} |
||||
|
||||
/* |
||||
* Convert an array of little-endian words to a string |
||||
*/ |
||||
function binl2str(bin) |
||||
{ |
||||
var str = ""; |
||||
var mask = (1 << chrsz) - 1; |
||||
for(var i = 0; i < bin.length * 32; i += chrsz) |
||||
str += String.fromCharCode((bin[i>>5] >>> (i % 32)) & mask); |
||||
return str; |
||||
} |
||||
|
||||
/* |
||||
* Convert an array of little-endian words to a hex string. |
||||
*/ |
||||
function binl2hex(binarray) |
||||
{ |
||||
var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef"; |
||||
var str = ""; |
||||
for(var i = 0; i < binarray.length * 4; i++) |
||||
{ |
||||
str += hex_tab.charAt((binarray[i>>2] >> ((i%4)*8+4)) & 0xF) + |
||||
hex_tab.charAt((binarray[i>>2] >> ((i%4)*8 )) & 0xF); |
||||
} |
||||
return str; |
||||
} |
||||
|
||||
/* |
||||
* Convert an array of little-endian words to a base-64 string |
||||
*/ |
||||
function binl2b64(binarray) |
||||
{ |
||||
var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; |
||||
var str = ""; |
||||
for(var i = 0; i < binarray.length * 4; i += 3) |
||||
{ |
||||
var triplet = (((binarray[i >> 2] >> 8 * ( i %4)) & 0xFF) << 16) |
||||
| (((binarray[i+1 >> 2] >> 8 * ((i+1)%4)) & 0xFF) << 8 ) |
||||
| ((binarray[i+2 >> 2] >> 8 * ((i+2)%4)) & 0xFF); |
||||
for(var j = 0; j < 4; j++) |
||||
{ |
||||
if(i * 8 + j * 6 > binarray.length * 32) str += b64pad; |
||||
else str += tab.charAt((triplet >> 6*(3-j)) & 0x3F); |
||||
} |
||||
} |
||||
return str; |
||||
} |
||||
module.exports = hex_md5; |
@ -0,0 +1,114 @@ |
||||
const ENV = require('./env.js'); |
||||
/* |
||||
{ |
||||
baseUrl: '', |
||||
header: {}, |
||||
method: 'GET', |
||||
dataType: 'json', |
||||
// #ifndef MP-ALIPAY || APP-PLUS
|
||||
responseType: 'text', |
||||
// #endif
|
||||
// 注:如果局部custom与全局custom有同名属性,则后面的属性会覆盖前面的属性,相当于Object.assign(全局,局部)
|
||||
custom: {}, // 全局自定义参数默认值
|
||||
// #ifdef MP-ALIPAY || MP-WEIXIN
|
||||
timeout: 30000, |
||||
// #endif
|
||||
// #ifdef APP-PLUS
|
||||
sslVerify: true, |
||||
// #endif
|
||||
// #ifdef H5
|
||||
// 跨域请求时是否携带凭证(cookies)仅H5支持(HBuilderX 2.6.15+)
|
||||
withCredentials: false, |
||||
// #endif
|
||||
// 局部优先级高于全局,返回当前请求的task,options。请勿在此处修改options。非必填
|
||||
// getTask: (task, options) => {
|
||||
// 相当于设置了请求超时时间500ms
|
||||
// setTimeout(() => {
|
||||
// task.abort()
|
||||
// }, 500)
|
||||
// }
|
||||
}*/ |
||||
// 此vm参数为页面的实例,可以通过它引用vuex中的变量
|
||||
module.exports = (vm) => { |
||||
|
||||
// 初始化请求配置
|
||||
uni.$u.http.setConfig((config) => { |
||||
|
||||
//console.log(config);
|
||||
/* config 为默认全局配置*/ |
||||
config.baseURL = ENV.ApiUrl; /* 根域名 */ |
||||
config.custom = { |
||||
auth:true, |
||||
toast:true, |
||||
}; |
||||
return config |
||||
}) |
||||
|
||||
// 请求拦截
|
||||
uni.$u.http.interceptors.request.use((config) => { // 可使用async await 做异步操作
|
||||
// 初始化请求拦截器时,会执行此方法,此时data为undefined,赋予默认{}
|
||||
config.data = config.data || {} |
||||
// 根据custom参数中配置的是否需要token,添加对应的请求头
|
||||
if(config?.custom?.auth) { |
||||
const token = uni.getStorageSync('token'); |
||||
if(token.length > 0) |
||||
{ |
||||
config.header.token = token; |
||||
} |
||||
} |
||||
return config
|
||||
}, config => { // 可使用async await 做异步操作
|
||||
return Promise.reject(config) |
||||
}) |
||||
|
||||
// 响应拦截
|
||||
uni.$u.http.interceptors.response.use((response) => { /* 对响应成功做点什么 可使用async await 做异步操作*/ |
||||
const data = response.data |
||||
|
||||
// 自定义参数
|
||||
const custom = response.config?.custom |
||||
if (data.result !== 1) {
|
||||
// 如果没有显式定义custom的toast参数为false的话,默认对报错进行toast弹出提示
|
||||
if (custom.toast !== false) { |
||||
uni.$u.toast(data.msg) |
||||
} |
||||
|
||||
// 如果需要catch返回,则进行reject
|
||||
if (custom?.catch) { |
||||
return Promise.reject(data) |
||||
} else { |
||||
// 否则返回一个pending中的promise,请求不会进入catch中
|
||||
return new Promise(() => { }) |
||||
} |
||||
} |
||||
return data.data === undefined ? {} : data.data |
||||
}, (response) => {
|
||||
//console.log(response);
|
||||
// 对响应错误做点什么 (statusCode !== 200)
|
||||
if(response.statusCode == 401) |
||||
{ |
||||
// 假设201为token失效,这里跳转登录
|
||||
uni.$u.toast('用户信息验证失败,请重新登录'); |
||||
setTimeout(() => { |
||||
// 此为uView的方法,详见路由相关文档
|
||||
//uni.$u.route('/pages/user/auth/login')
|
||||
}, 1500) |
||||
|
||||
} |
||||
else if(response.statusCode == 500) |
||||
{ |
||||
// 假设201为token失效,这里跳转登录
|
||||
uni.$u.toast('当前接口出错'); |
||||
|
||||
} |
||||
else |
||||
{ |
||||
|
||||
// 如果返回false,则会调用Promise的reject回调,
|
||||
// 并将进入this.$u.post(url).then().catch(res=>{})的catch回调中,res为服务端的返回值
|
||||
uni.$u.toast('当前接口状态码'+response.statusCode); |
||||
|
||||
} |
||||
return Promise.reject(response) |
||||
}) |
||||
} |
@ -0,0 +1,88 @@ |
||||
var user = { |
||||
|
||||
/** |
||||
* 本算法来源于简书开源代码,详见:https://www.jianshu.com/p/fdbf293d0a85
|
||||
* 全局唯一标识符(uuid,Globally Unique Identifier),也称作 uuid(Universally Unique IDentifier)
|
||||
* 一般用于多个组件之间,给它一个唯一的标识符,或者v-for循环的时候,如果使用数组的index可能会导致更新列表出现问题 |
||||
* 最可能的情况是左滑删除item或者对某条信息流"不喜欢"并去掉它的时候,会导致组件内的数据可能出现错乱 |
||||
* v-for的时候,推荐使用后端返回的id而不是循环的index |
||||
* @param {Number} len uuid的长度 |
||||
* @param {Boolean} firstU 将返回的首字母置为"u" |
||||
* @param {Nubmer} radix 生成uuid的基数(意味着返回的字符串都是这个基数),2-二进制,8-八进制,10-十进制,16-十六进制 |
||||
*/ |
||||
guid:function (len = 32, firstU = true, radix = null) { |
||||
let chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''); |
||||
let uuid = []; |
||||
radix = radix || chars.length; |
||||
|
||||
if (len) { |
||||
// 如果指定uuid长度,只是取随机的字符,0|x为位运算,能去掉x的小数位,返回整数位
|
||||
for (let i = 0; i < len; i++) uuid[i] = chars[0 | Math.random() * radix]; |
||||
} else { |
||||
let r; |
||||
// rfc4122标准要求返回的uuid中,某些位为固定的字符
|
||||
uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'; |
||||
uuid[14] = '4'; |
||||
|
||||
for (let i = 0; i < 36; i++) { |
||||
if (!uuid[i]) { |
||||
r = 0 | Math.random() * 16; |
||||
uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]; |
||||
} |
||||
} |
||||
} |
||||
// 移除第一个字符,并用u替代,因为第一个字符为数值时,该guuid不能用作id或者class
|
||||
if (firstU) { |
||||
uuid.shift(); |
||||
return 'u' + uuid.join(''); |
||||
} else { |
||||
return uuid.join(''); |
||||
} |
||||
}, |
||||
//设置用户本地信息
|
||||
|
||||
session: function(key, val) { |
||||
if (typeof(val) == 'undefined') { |
||||
var res = uni.getStorageSync(key); |
||||
return res; |
||||
} else { |
||||
uni.setStorageSync(key, val); |
||||
} |
||||
}, |
||||
//删除用户本地所有信息
|
||||
clearSession: function() { |
||||
var keys = uni.getStorageInfoSync().keys; |
||||
var session_id = uni.getStorageSync('session_id'); |
||||
for (var key in keys) { |
||||
uni.removeStorageSync(keys[key]); |
||||
} |
||||
uni.setStorageSync('session_id', session_id); |
||||
}, |
||||
|
||||
isLogin: function() { |
||||
var res = uni.getStorageSync('token'); |
||||
if (typeof(res) == 'undefined' || res == '') { |
||||
return false; |
||||
} else { |
||||
return res; |
||||
} |
||||
}, |
||||
|
||||
getUserInfo: function() { |
||||
var res = uni.getStorageSync('token'); |
||||
if (typeof(res) == 'undefined' || res == '') { |
||||
return false; |
||||
} else { |
||||
var userInfo = {}; |
||||
userInfo.nickName = uni.getStorageSync('nickName'); |
||||
userInfo.phone = uni.getStorageSync('phone'); |
||||
userInfo.userId = uni.getStorageSync('userId'); |
||||
userInfo.token = uni.getStorageSync('token'); |
||||
userInfo.avatar = uni.getStorageSync('avatar'); |
||||
userInfo.realName = uni.getStorageSync('realName'); |
||||
return userInfo; |
||||
} |
||||
} |
||||
} |
||||
|
||||
module.exports = user |
@ -0,0 +1,228 @@ |
||||
// /common/wechat.js
|
||||
//https://zhuanlan.zhihu.com/p/512790783?utm_id=0
|
||||
import Vue from "vue" |
||||
var jweixin = require('./jssdk.js') |
||||
export default { |
||||
// 调试模式
|
||||
debug: false, |
||||
// api列表
|
||||
jsApiList: [ |
||||
'updateAppMessageShareData', |
||||
'updateTimelineShareData', |
||||
'closeWindow', |
||||
'getLocation', |
||||
'openLocation', |
||||
'openAddress', |
||||
'scanQRCode', |
||||
'chooseImage', |
||||
'chooseWXPay' |
||||
], |
||||
// 判断是否在微信中
|
||||
isWechat: function() { |
||||
var ua = window.navigator.userAgent.toLowerCase() |
||||
return ua.match(/micromessenger/i) == 'micromessenger' ? true : false |
||||
}, |
||||
// 初始化sdk配置
|
||||
initJssdk: function(callback) { |
||||
var url = window.location.href; |
||||
var post = {url:url}; |
||||
if (this.isWechat()) { |
||||
Vue.prototype.$u.post('login/getApi', post).then(res => { |
||||
console.log('getApi : '); |
||||
console.log(res); |
||||
var share = JSON.parse(res.data); |
||||
console.log('getApi-share : '); |
||||
console.log(share); |
||||
jweixin.config({ |
||||
debug: share.debug || this.debug, |
||||
appId: share.appId, |
||||
timestamp:share.timestamp, |
||||
nonceStr: share.nonceStr, |
||||
signature: share.signature, |
||||
jsApiList: share.jsApiList || this.jsApiList |
||||
}) |
||||
if (typeof callback === 'function') { |
||||
console.log('getApi-callback : '); |
||||
console.log(share); |
||||
callback(share); |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
// 关闭页面事件
|
||||
closeWindow: function(callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function(wx) { |
||||
wx.closeWindow() |
||||
if (typeof callback === 'function') { |
||||
callback(jweixin) |
||||
} |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 微信分享
|
||||
share: function(data, callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
|
||||
jweixin.ready(function() { |
||||
|
||||
var shareData = { |
||||
title: data.title, |
||||
desc: data.desc, |
||||
link: window.location.href, |
||||
imgUrl: data.image, |
||||
success: function(res) { |
||||
callback(res) |
||||
|
||||
}, |
||||
cancel: function(res) { |
||||
callback(res) |
||||
} |
||||
} |
||||
jweixin.updateAppMessageShareData(shareData) |
||||
jweixin.updateTimelineShareData(shareData) |
||||
|
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 获取位置信息
|
||||
getLocation: function(callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.getLocation({ |
||||
type: 'gcj02', |
||||
success: function(res) { |
||||
callback(res) |
||||
}, |
||||
fail: function(err) { |
||||
callback(err) |
||||
} |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 查看位置信息
|
||||
openLocation: function(data, callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.openLocation({ |
||||
latitude: data.latitude, |
||||
longitude: data.longitude |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 获取微信收货地址
|
||||
openAddress: function(callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.openAddress({ |
||||
success: function(res) { |
||||
callback(res) |
||||
}, |
||||
fail: function(err) { |
||||
callback(err) |
||||
} |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 微信扫码
|
||||
scanQRCode: function(callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.scanQRCode({ |
||||
needResult: 1, // 0:微信处理|1:返回扫描结果
|
||||
scanType: ["qrCode", "barCode"], |
||||
success: function(res) { |
||||
let durl = /https:\/\/([^\/]+)\//i |
||||
let domain |
||||
res.resultStr.replace(durl, (e) => { |
||||
domain = e |
||||
}) |
||||
callback(res) |
||||
}, |
||||
fail: function(err) { |
||||
callback(err) |
||||
} |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 选择图片
|
||||
chooseImage: function(callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.chooseImage({ |
||||
count: 1, |
||||
sizeType: ['compressed'], |
||||
sourceType: ['album'], |
||||
success: function(res) { |
||||
callback(res) |
||||
} |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 微信支付
|
||||
wxpay: function(data, callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
jweixin.chooseWXPay({ |
||||
timestamp: data.timeStamp, |
||||
nonceStr: data.nonceStr, |
||||
package: data.package, |
||||
signType: data.signType, |
||||
paySign: data.paySign, |
||||
success: function(res) { |
||||
callback(res) |
||||
}, |
||||
cancel: function(res) { |
||||
callback(res) |
||||
}, |
||||
fail: function(err) { |
||||
callback(err) |
||||
} |
||||
}) |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
// 微信支付 另一种方式
|
||||
wxpayBridge: function(data, callback) { |
||||
if (this.isWechat()) { |
||||
this.initJssdk(function(init) { |
||||
jweixin.ready(function() { |
||||
WeixinJSBridge.invoke( |
||||
'getBrandWCPayRequest', { |
||||
"appId": data.appId, |
||||
"timeStamp": data.timeStamp, |
||||
"nonceStr": data.nonceStr, |
||||
"package": data.package, |
||||
"signType": "MD5", |
||||
"paySign": data.paySign |
||||
}, |
||||
function(res) { |
||||
callback(res) |
||||
} |
||||
) |
||||
}) |
||||
}) |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,20 @@ |
||||
<!DOCTYPE html> |
||||
<html lang="en"> |
||||
<head> |
||||
<meta charset="UTF-8" /> |
||||
<script> |
||||
var coverSupport = 'CSS' in window && typeof CSS.supports === 'function' && (CSS.supports('top: env(a)') || |
||||
CSS.supports('top: constant(a)')) |
||||
document.write( |
||||
'<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0' + |
||||
(coverSupport ? ', viewport-fit=cover' : '') + '" />') |
||||
</script> |
||||
<title></title> |
||||
<!--preload-links--> |
||||
<!--app-context--> |
||||
</head> |
||||
<body> |
||||
<div id="app"><!--app-html--></div> |
||||
<script type="module" src="/main.js"></script> |
||||
</body> |
||||
</html> |
@ -0,0 +1,23 @@ |
||||
{ |
||||
"app.name": "空间充", |
||||
"tabbar.home": "HOME", |
||||
"tabbar.list": "NEARBY", |
||||
"tabbar.user": "USER", |
||||
"page.list.index.title": "NEARBY", |
||||
|
||||
"page.user.index.title": "User Center", |
||||
"pages.user.setting.index.title":"Setting", |
||||
"pages.user.setting.index.setLang":"Language", |
||||
|
||||
|
||||
"locale.auto": "System", |
||||
"locale.en": "English", |
||||
"locale.zh-hans": "简体中文", |
||||
"locale.zh-hant": "繁体中文", |
||||
"locale.language-change-confirm": "Applying this setting will restart the app" |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
} |
@ -0,0 +1,8 @@ |
||||
import zhHans from './zh-Hans.json' |
||||
import zhHant from './zh-Hant.json' |
||||
import en from './en.json' |
||||
export default { |
||||
'zh-Hans': zhHans, |
||||
'zh-Hant': zhHant, |
||||
en |
||||
} |
@ -0,0 +1,36 @@ |
||||
{ |
||||
"common": { |
||||
"uni.app.quit": "Press again to quit the app", |
||||
"uni.async.error": "The connection to the server timed out, click the screen to retry", |
||||
"uni.showActionSheet.cancel": "Cancel", |
||||
"uni.showToast.unpaired": "Please note that showToast and hideToast must be paired", |
||||
"uni.showLoading.unpaired": "Please note that showLoading and hideLoading must be paired", |
||||
"uni.showModal.cancel": "Cancel", |
||||
"uni.showModal.confirm": "OK", |
||||
"uni.chooseImage.cancel": "Cancel", |
||||
"uni.chooseImage.sourceType.album": "Choose from Album", |
||||
"uni.chooseImage.sourceType.camera": "shoot", |
||||
"uni.chooseVideo.cancel": "Cancel", |
||||
"uni.chooseVideo.sourceType.album": "Choose from Album", |
||||
"uni.chooseVideo.sourceType.camera": "Camera", |
||||
"uni.previewImage.cancel": "Cancel", |
||||
"uni.previewImage.button.save": "Save Image", |
||||
"uni.previewImage.save.success": "Successful saving image to album", |
||||
"uni.previewImage.save.fail": "Failed to save image to album", |
||||
"uni.setClipboardData.success": "Content copied", |
||||
"uni.scanCode.title": "scan", |
||||
"uni.scanCode.album": "Album", |
||||
"uni.scanCode.fail": "Recognition failed", |
||||
"uni.scanCode.flash.on": "Light touch to light", |
||||
"uni.scanCode.flash.off": "Flick Off", |
||||
"uni.startSoterAuthentication.authContent": "Fingerprint identification...", |
||||
"uni.picker.done": "Done", |
||||
"uni.picker.cancel": "Cancel", |
||||
"uni.video.danmu": "Barrage", |
||||
"uni.video.volume": "Volume", |
||||
"uni.button.feedback.title": "Problem Feedback", |
||||
"uni.button.feedback.send": "send" |
||||
}, |
||||
"ios": {}, |
||||
"android": {} |
||||
} |
@ -0,0 +1,36 @@ |
||||
{ |
||||
"common": { |
||||
"uni.app.quit": "再按一次退出应用", |
||||
"uni.async.error": "连接服务器超时,点击屏幕重试", |
||||
"uni.showActionSheet.cancel": "取消", |
||||
"uni.showToast.unpaired": "请注意 showToast 与 hideToast 必须配对使用", |
||||
"uni.showLoading.unpaired": "请注意 showLoading 与 hideLoading 必须配对使用", |
||||
"uni.showModal.cancel": "取消", |
||||
"uni.showModal.confirm": "确定", |
||||
"uni.chooseImage.cancel": "取消", |
||||
"uni.chooseImage.sourceType.album": "从相册选择", |
||||
"uni.chooseImage.sourceType.camera": "拍摄", |
||||
"uni.chooseVideo.cancel": "取消", |
||||
"uni.chooseVideo.sourceType.album": "从相册选择", |
||||
"uni.chooseVideo.sourceType.camera": "拍摄", |
||||
"uni.previewImage.cancel": "取消", |
||||
"uni.previewImage.button.save": "保存图像", |
||||
"uni.previewImage.save.success": "保存图像到相册成功", |
||||
"uni.previewImage.save.fail": "保存图像到相册失败", |
||||
"uni.setClipboardData.success": "内容已复制", |
||||
"uni.scanCode.title": "扫码", |
||||
"uni.scanCode.album": "相册", |
||||
"uni.scanCode.fail": "识别失败", |
||||
"uni.scanCode.flash.on": "轻触照亮", |
||||
"uni.scanCode.flash.off": "轻触关闭", |
||||
"uni.startSoterAuthentication.authContent": "指纹识别中...", |
||||
"uni.picker.done": "完成", |
||||
"uni.picker.cancel": "取消", |
||||
"uni.video.danmu": "弹幕", |
||||
"uni.video.volume": "音量", |
||||
"uni.button.feedback.title": "问题反馈", |
||||
"uni.button.feedback.send": "发送" |
||||
}, |
||||
"ios": {}, |
||||
"android": {} |
||||
} |
@ -0,0 +1,36 @@ |
||||
{ |
||||
"common": { |
||||
"uni.app.quit": "再按一次退出應用", |
||||
"uni.async.error": "連接服務器超時,點擊屏幕重試", |
||||
"uni.showActionSheet.cancel": "取消", |
||||
"uni.showToast.unpaired": "請註意 showToast 與 hideToast 必須配對使用", |
||||
"uni.showLoading.unpaired": "請註意 showLoading 與 hideLoading 必須配對使用", |
||||
"uni.showModal.cancel": "取消", |
||||
"uni.showModal.confirm": "確定", |
||||
"uni.chooseImage.cancel": "取消", |
||||
"uni.chooseImage.sourceType.album": "從相冊選擇", |
||||
"uni.chooseImage.sourceType.camera": "拍攝", |
||||
"uni.chooseVideo.cancel": "取消", |
||||
"uni.chooseVideo.sourceType.album": "從相冊選擇", |
||||
"uni.chooseVideo.sourceType.camera": "拍攝", |
||||
"uni.previewImage.cancel": "取消", |
||||
"uni.previewImage.button.save": "保存圖像", |
||||
"uni.previewImage.save.success": "保存圖像到相冊成功", |
||||
"uni.previewImage.save.fail": "保存圖像到相冊失敗", |
||||
"uni.setClipboardData.success": "內容已復製", |
||||
"uni.scanCode.title": "掃碼", |
||||
"uni.scanCode.album": "相冊", |
||||
"uni.scanCode.fail": "識別失敗", |
||||
"uni.scanCode.flash.on": "輕觸照亮", |
||||
"uni.scanCode.flash.off": "輕觸關閉", |
||||
"uni.startSoterAuthentication.authContent": "指紋識別中...", |
||||
"uni.picker.done": "完成", |
||||
"uni.picker.cancel": "取消", |
||||
"uni.video.danmu": "彈幕", |
||||
"uni.video.volume": "音量", |
||||
"uni.button.feedback.title": "問題反饋", |
||||
"uni.button.feedback.send": "發送" |
||||
}, |
||||
"ios": {}, |
||||
"android": {} |
||||
} |
@ -0,0 +1,39 @@ |
||||
{ |
||||
"app.name": "合成照相馆", |
||||
"tabbar.home": "首页", |
||||
"tabbar.list": "附近", |
||||
"tabbar.user": "我的", |
||||
"page.list.index.title": "附近", |
||||
|
||||
"page.user.index.title": "用户中心", |
||||
"pages.user.setting.index.title":"设置", |
||||
"pages.user.setting.index.setLang":"多语言", |
||||
|
||||
"locale.auto": "系统", |
||||
"locale.en": "English", |
||||
"locale.zh-hans": "简体中文", |
||||
"locale.zh-hant": "繁体中文", |
||||
"locale.language-change-confirm": "应用此设置将重启App", |
||||
"locale.tips": "提示", |
||||
|
||||
"locale.login": "登录账号", |
||||
"locale.regist": "账号注册", |
||||
"locale.logout": "退出登录", |
||||
|
||||
"pages.user.auth.login.login":"登录页面", |
||||
"pages.user.auth.login.mobile":"手机号码", |
||||
"pages.user.auth.login.input_mobile":"请输入手机号码", |
||||
"pages.user.auth.login.mobile_must":"手机号码格式不正确", |
||||
|
||||
"pages.user.auth.login.email":"邮箱地址", |
||||
"pages.user.auth.login.input_email":"请输入Email地址", |
||||
"pages.user.auth.login.email_must":"Email格式不正确", |
||||
|
||||
"pages.user.auth.login.password":"账号密码", |
||||
"pages.user.auth.login.input_password":"请输入账号密码", |
||||
"pages.user.auth.login.password_must":"密码不小于6位", |
||||
|
||||
"pages.user.auth.regist.regist":"注册页面", |
||||
"pages.user.auth.regist.regist_success":"注册成功" |
||||
|
||||
} |
@ -0,0 +1,20 @@ |
||||
{ |
||||
"app.name": "空间充", |
||||
"tabbar.home": "首頁", |
||||
"tabbar.list": "附近", |
||||
"tabbar.user": "我的", |
||||
"page.list.index.title": "附近", |
||||
|
||||
"page.user.index.title": "用戶中心", |
||||
"pages.user.setting.index.title":"設置", |
||||
"pages.user.setting.index.setLang":"多語言", |
||||
|
||||
"locale.auto": "系統", |
||||
"locale.en": "English", |
||||
"locale.zh-hans": "简体中文", |
||||
"locale.zh-hant": "繁體中文", |
||||
"locale.language-change-confirm": "應用此設置將重啟App" |
||||
|
||||
|
||||
|
||||
} |
@ -0,0 +1,57 @@ |
||||
import App from './App' |
||||
import messages from './locale/index' |
||||
// main.js
|
||||
import uView from '@/uni_modules/uview-ui'; |
||||
Vue.use(uView); |
||||
|
||||
|
||||
let i18nConfig = { |
||||
locale: uni.getLocale(), |
||||
messages |
||||
} |
||||
// 引入请求封装,将app参数传递到配置中
|
||||
require('@/config/request.js')(app); //uview js
|
||||
//自己封装的部分
|
||||
import api from '@/config/api.js'; |
||||
Vue.prototype.$api = api; |
||||
import user from '@/config/user.js'; |
||||
Vue.prototype.$user = user; |
||||
import com from '@/config/com.js'; |
||||
Vue.prototype.$com = com; |
||||
|
||||
import access from '@/config/access/index.js'; |
||||
Vue.prototype.$access = access; |
||||
|
||||
// #ifdef H5
|
||||
import wechat from '@/config/wechat.js'
|
||||
Vue.prototype.$wechat =wechat;
|
||||
// #endif
|
||||
|
||||
|
||||
|
||||
// #ifndef VUE3
|
||||
import Vue from 'vue' |
||||
import VueI18n from 'vue-i18n' |
||||
Vue.use(VueI18n) |
||||
const i18n = new VueI18n(i18nConfig) |
||||
Vue.config.productionTip = false |
||||
App.mpType = 'app' |
||||
const app = new Vue({ |
||||
i18n, |
||||
...App |
||||
}) |
||||
app.$mount(); |
||||
// #endif
|
||||
|
||||
// #ifdef VUE3
|
||||
import { createSSRApp } from 'vue' |
||||
import { createI18n } from 'vue-i18n' |
||||
const i18n = createI18n(i18nConfig) |
||||
export function createApp() { |
||||
const app = createSSRApp(App) |
||||
app.use(i18n) |
||||
return { |
||||
app |
||||
} |
||||
} |
||||
// #endif
|
@ -0,0 +1,100 @@ |
||||
{ |
||||
"name" : "合成照相馆", |
||||
"appid" : "__UNI__304D8B8", |
||||
"description" : "", |
||||
"versionName" : "2.1.2", |
||||
"versionCode" : 212, |
||||
"transformPx" : false, |
||||
/* 5+App特有相关 */ |
||||
"app-plus" : { |
||||
"usingComponents" : true, |
||||
"nvueStyleCompiler" : "uni-app", |
||||
"compilerVersion" : 3, |
||||
"splashscreen" : { |
||||
"alwaysShowBeforeRender" : true, |
||||
"waiting" : true, |
||||
"autoclose" : true, |
||||
"delay" : 0 |
||||
}, |
||||
/* 模块配置 */ |
||||
"modules" : {}, |
||||
/* 应用发布信息 */ |
||||
"distribute" : { |
||||
/* android打包配置 */ |
||||
"android" : { |
||||
"permissions" : [ |
||||
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>", |
||||
"<uses-permission android:name=\"android.permission.VIBRATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.READ_LOGS\"/>", |
||||
"<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>", |
||||
"<uses-feature android:name=\"android.hardware.camera.autofocus\"/>", |
||||
"<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.CAMERA\"/>", |
||||
"<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>", |
||||
"<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>", |
||||
"<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>", |
||||
"<uses-feature android:name=\"android.hardware.camera\"/>", |
||||
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>" |
||||
] |
||||
}, |
||||
/* ios打包配置 */ |
||||
"ios" : {}, |
||||
/* SDK配置 */ |
||||
"sdkConfigs" : {} |
||||
} |
||||
}, |
||||
/* 快应用特有相关 */ |
||||
"quickapp" : {}, |
||||
/* 小程序特有相关 */ |
||||
"mp-weixin" : { |
||||
"appid" : "wx19306460077082e9", |
||||
"setting" : { |
||||
"urlCheck" : false, |
||||
"es6" : true, |
||||
"postcss" : true, |
||||
"minified" : true |
||||
}, |
||||
"usingComponents" : true, |
||||
"permission" : { |
||||
"scope.userLocation" : { |
||||
"desc" : "你的位置信息将用于小程序位置接口的效果展示" |
||||
} |
||||
}, |
||||
"requiredPrivateInfos" : [ "getLocation", "chooseLocation" ], |
||||
"libVersion" : "latest" |
||||
}, |
||||
"mp-alipay" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"mp-baidu" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"mp-toutiao" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"uniStatistics" : { |
||||
"enable" : false |
||||
}, |
||||
"vueVersion" : "2", |
||||
"locale" : "zh-Hans", |
||||
"fallbackLocale" : "zh-Hans", |
||||
"h5" : { |
||||
"title" : "空间充", |
||||
"router" : { |
||||
"mode" : "history", |
||||
"base" : "/h5/" |
||||
}, |
||||
"template" : "", |
||||
"devServer" : { |
||||
"https" : false |
||||
}, |
||||
"optimization" : { |
||||
"treeShaking" : { |
||||
"enable" : false |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,82 @@ |
||||
{ |
||||
"pages": [ //pages数组中第一项表示应用启动页,参考:https://uniapp.dcloud.io/collocation/pages |
||||
{ |
||||
"path": "pages/index/index", |
||||
"style": { |
||||
"navigationBarTitleText": "%app.name%", |
||||
"navigationStyle": "custom" |
||||
} |
||||
} |
||||
,{ |
||||
"path" : "pages/user/index", |
||||
"style" : |
||||
{ |
||||
"navigationBarTitleText": "%page.user.index.title%", |
||||
"enablePullDownRefresh": false |
||||
|
||||
} |
||||
|
||||
} |
||||
,{ |
||||
"path" : "pages/user/setting/index", |
||||
"style" : |
||||
{ |
||||
"navigationBarTitleText": "%pages.user.setting.index.setLang%", |
||||
"enablePullDownRefresh": false |
||||
} |
||||
|
||||
}, |
||||
{ |
||||
"path" : "pages/user/account/index", |
||||
"style" : |
||||
{ |
||||
"navigationBarTitleText" : "账户设置", |
||||
"enablePullDownRefresh" : false |
||||
} |
||||
}, |
||||
{ |
||||
"path" : "pages/web/index", |
||||
"style" : |
||||
{ |
||||
"navigationBarTitleText" : "外部页面", |
||||
"enablePullDownRefresh" : false |
||||
} |
||||
}, |
||||
{ |
||||
"path" : "pages/web/video", |
||||
"style" : |
||||
{ |
||||
"navigationBarTitleText" : "视频", |
||||
"enablePullDownRefresh" : false |
||||
} |
||||
} |
||||
|
||||
], |
||||
"subPackages": [], |
||||
"globalStyle": { |
||||
"navigationBarTextStyle": "black", |
||||
"navigationBarTitleText": "%app.name%", |
||||
"navigationBarBackgroundColor": "#FFFFFF", |
||||
"backgroundColor": "#FFFFFF" |
||||
}, |
||||
"tabBar": { |
||||
"color": "#AFB1DB", |
||||
"selectedColor": "#7A74F0", |
||||
"borderStyle": "black", |
||||
"backgroundColor": "#FFFFFF", |
||||
"list": [{ |
||||
"pagePath": "pages/index/index", |
||||
"iconPath": "static/img/common/home.png", |
||||
"selectedIconPath": "static/img/common/home_HL.png", |
||||
"text": "%tabbar.home%" |
||||
}, |
||||
{ |
||||
"pagePath": "pages/user/index", |
||||
"iconPath": "static/img/common/user.png", |
||||
"selectedIconPath": "static/img/common/user_HL.png", |
||||
"text": "%tabbar.user%" |
||||
} |
||||
] |
||||
}, |
||||
"uniIdRouter": {} |
||||
} |
@ -0,0 +1,482 @@ |
||||
<template> |
||||
<view class="container"> |
||||
|
||||
<view class="bannerBox"> |
||||
<u-swiper |
||||
:list="banner" |
||||
keyName="adImg" |
||||
:showTitle="false" |
||||
:autoplay="true" |
||||
circular |
||||
height="280" |
||||
radius="0" |
||||
@click="openAd" |
||||
></u-swiper> |
||||
</view> |
||||
|
||||
|
||||
<view class="contentBox"> |
||||
|
||||
<view class="navBox"> |
||||
<view class="navItem"> |
||||
<image src="/static/img/index/zjhc.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="navItem"> |
||||
<image src="/static/img/index/zjgs.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="navItem"> |
||||
<image src="/static/img/index/aixz.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="navItem"> |
||||
<image src="/static/img/index/kthf.png" mode="widthFix"></image> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="indexTitle"> |
||||
<image src="/static/img/index/title_zj.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="listBox"> |
||||
<view class="photoBox"> |
||||
<view class="photoTop"> |
||||
<view class="topLeft"> |
||||
<image src="/static/img/index/zj_icon.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="topCenter"> |
||||
<view class="title">一寸照</view> |
||||
<view class="desc">25×35mm | 413×579px</view> |
||||
|
||||
</view> |
||||
<view class="topRight"> |
||||
<view class="distance"> |
||||
<image src="/static/img/index/right.png" mode="widthFix"></image> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="photoBox"> |
||||
<view class="photoTop"> |
||||
<view class="topLeft"> |
||||
<image src="/static/img/index/zj_icon.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="topCenter"> |
||||
<view class="title">二寸照</view> |
||||
<view class="desc">25×35mm | 413×579px</view> |
||||
|
||||
</view> |
||||
<view class="topRight"> |
||||
<view class="distance"> |
||||
<image src="/static/img/index/right.png" mode="widthFix"></image> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
|
||||
|
||||
<view class="photoBox"> |
||||
<view class="photoTop"> |
||||
<view class="topLeft"> |
||||
<image src="/static/img/index/zj_icon.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="topCenter"> |
||||
<view class="title">大一寸照</view> |
||||
<view class="desc">25×35mm | 413×579px</view> |
||||
|
||||
</view> |
||||
<view class="topRight"> |
||||
<view class="distance"> |
||||
<image src="/static/img/index/right.png" mode="widthFix"></image> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="photoBox"> |
||||
<view class="photoTop"> |
||||
<view class="topLeft"> |
||||
<image src="/static/img/index/zj_icon.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="topCenter"> |
||||
<view class="title">小一寸照</view> |
||||
<view class="desc">25×35mm | 413×579px</view> |
||||
|
||||
</view> |
||||
<view class="topRight"> |
||||
<view class="distance"> |
||||
<image src="/static/img/index/right.png" mode="widthFix"></image> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
|
||||
</view> |
||||
<view class="indexTitle"> |
||||
<image src="/static/img/index/title_xz.png" mode="widthFix"></image> |
||||
</view> |
||||
|
||||
<view class="picList"> |
||||
<scroll-view class="scrollBox" scroll-x="true" @scroll="scroll1" scroll-left="0"> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
</scroll-view> |
||||
</view> |
||||
|
||||
<view class="indexTitle"> |
||||
<image src="/static/img/index/title_kt.png" mode="widthFix"></image> |
||||
</view> |
||||
|
||||
<view class="picList"> |
||||
<scroll-view class="scrollBox" scroll-x="true" @scroll="scroll2" scroll-left="0"> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
<view class="picBox"> |
||||
<view class="pic"> |
||||
<image src="/static/img/index/pic.png" mode="widthFix"></image> |
||||
</view> |
||||
<view class="pic_title">风格名称</view> |
||||
</view> |
||||
</scroll-view> |
||||
</view> |
||||
|
||||
|
||||
</view> |
||||
|
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
data() { |
||||
return { |
||||
banner: [ |
||||
{ |
||||
adImg:'/static/img/index/banner.png', |
||||
} |
||||
], |
||||
notice:'', |
||||
} |
||||
}, |
||||
computed:{ |
||||
|
||||
}, |
||||
onLoad() { |
||||
|
||||
this.getHomeInfo(); |
||||
|
||||
}, |
||||
onShareAppMessage() { |
||||
return { |
||||
title: '空间充-首页', |
||||
path: uni.$u.page(), |
||||
} |
||||
}, |
||||
onShareTimeline() { |
||||
return { |
||||
title: '空间充-首页', |
||||
path: uni.$u.page(), |
||||
} |
||||
}, |
||||
methods: { |
||||
//获取门店信息 |
||||
getHomeInfo(){ |
||||
var _this =this; |
||||
//post 请求例子 |
||||
_this.$api.post('ycl/home', {},function(res){ |
||||
console.log(res); |
||||
var d = res; |
||||
var banner = d.adList |
||||
banner.forEach((item, index) => { |
||||
item.title = item.adName; |
||||
}) |
||||
//_this.banner = banner; |
||||
//_this.notice = d.notice; |
||||
//_this.$user.session('servicePhone',d.servicePhone); |
||||
}); |
||||
}, |
||||
//滚动监控 |
||||
scroll1(e) { |
||||
console.log(e); |
||||
}, |
||||
scroll2(e) { |
||||
console.log(e); |
||||
}, |
||||
//扫码 |
||||
scanQrcode() |
||||
{ |
||||
var _this =this; |
||||
_this.$com.showLoading('识别中...'); |
||||
uni.scanCode({ |
||||
onlyFromCamera: false, |
||||
success: function (res) { |
||||
console.log(res); |
||||
console.log('条码类型:' + res.scanType); |
||||
console.log('条码内容:' + res.result); |
||||
if (res.scanType == 'QR_CODE') |
||||
{ |
||||
var url = res.result; |
||||
var codeTypeArr = ['deviceKey', 'deviceCode', 'qrcode', 'code']; //支持的条码参数 |
||||
var url_str = ''; |
||||
codeTypeArr.forEach(function(item) { |
||||
if(url.indexOf(item) != -1 ) |
||||
{ |
||||
var keyindex = item; |
||||
var deviceKey = _this.getParameterByName(url,item); |
||||
if(!_this.$com.isNull(deviceKey)) |
||||
{ |
||||
if(['qrcode', 'code'].includes(item)) { |
||||
keyindex = 'deviceCode'; |
||||
} |
||||
console.log(keyindex+'='+deviceKey); |
||||
url_str = `${keyindex}=${deviceKey}`; |
||||
return; |
||||
} |
||||
|
||||
} |
||||
}); |
||||
if(!_this.$com.isNull(url_str)) |
||||
{ |
||||
uni.$u.route({ |
||||
url: '/pages/order/order?'+url_str, |
||||
}); |
||||
} |
||||
else |
||||
{ |
||||
|
||||
_this.$com.alert('二维码格式有误'); |
||||
|
||||
} |
||||
} |
||||
else |
||||
{ |
||||
_this.$com.alert('暂不支持此种条码类型'); |
||||
} |
||||
}, |
||||
fail:function(err){ |
||||
_this.$com.alert('扫码失败,请重试'); |
||||
}, |
||||
complete:function(end){ |
||||
_this.$com.hideLoading(); |
||||
} |
||||
}); |
||||
}, |
||||
//获取url中是否有某个参数的值 |
||||
getParameterByName(url,name) { |
||||
name = name.replace(/[\[]/, "\\[").replace(/[\]]/, "\\]"); |
||||
var regexS = "[\\?&]" + name + "=([^&#]*)"; |
||||
var regex = new RegExp(regexS); |
||||
var results = regex.exec(url); |
||||
|
||||
if (results == null) return ""; |
||||
else return decodeURIComponent(results[1].replace(/\+/g, " ")); |
||||
}, |
||||
openAd(index) |
||||
{ |
||||
var _this =this; |
||||
var item = _this.banner[index]; |
||||
console.log(item); |
||||
if(!_this.$com.isNull(item.jumpUrl)) |
||||
{ |
||||
console.log(item.jumpUrl); |
||||
var url = ''; |
||||
switch(item.jumpType) |
||||
{ |
||||
case 0: |
||||
url=item.jumpUrl; |
||||
break; |
||||
|
||||
case 1: |
||||
url='/pages/web/index?title='+item.adName+'&url='+encodeURIComponent(item.jumpUrl); |
||||
break; |
||||
|
||||
case 2: |
||||
url='/pages/web/video?title='+item.adName+'&url='+encodeURIComponent(item.jumpUrl); |
||||
break; |
||||
|
||||
default: |
||||
url = ''; |
||||
} |
||||
if(!_this.$com.isNull(url)) |
||||
{ |
||||
uni.navigateTo({ |
||||
url:url, |
||||
}) |
||||
} |
||||
} |
||||
}, |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
.container{width: 100%;} |
||||
|
||||
.bannerBox{ |
||||
width: 100%; |
||||
background: #082436; |
||||
height: auto; |
||||
position: relative; |
||||
} |
||||
::v-deep .u-swiper{background-color: inherit !important;} |
||||
|
||||
.contentBox{ |
||||
width:746rpx; |
||||
height: 500rpx; |
||||
background: linear-gradient( 180deg, #DBD9FF 0%, #F5F6FB 100%); |
||||
border-radius: 28rpx 28rpx 0rpx 0rpx; |
||||
border: 2rpx solid #FFFFFF; |
||||
margin-top: -80rpx; |
||||
position: relative; |
||||
z-index: 2; |
||||
|
||||
|
||||
} |
||||
|
||||
.navBox{ width: 100%; height: auto; display: flex;} |
||||
.navBox .navItem{ width: 164rpx; height: 140rpx; margin: 40rpx 20rpx 20rpx 20rpx;} |
||||
.navBox .navItem image{ width: 100%; height: auto;} |
||||
.indexTitle {width: 730rpx; height: auto; margin: 20rpx auto; padding-left: 20rpx; } |
||||
.indexTitle image{ width: 226rpx; height: auto;} |
||||
|
||||
|
||||
.listBox{ width: 100%; height: auto;} |
||||
.photoBox{ |
||||
padding:20rpx 10rpx 10rpx 10rpx; |
||||
height: auto; |
||||
border-radius: 24rpx 24rpx 24rpx 24rpx; |
||||
opacity: 1; |
||||
background: #FFFFFF; |
||||
width: 670rpx; |
||||
margin: 15rpx auto; |
||||
} |
||||
.photoTop{width: 100%; height: auto; display: flex; |
||||
|
||||
} |
||||
.photoTop .topLeft{ |
||||
width: 100rpx; |
||||
height: 88rpx; |
||||
|
||||
} |
||||
.topLeft image{ width: 60rpx; height: auto; margin-left: 14rpx;} |
||||
.photoTop .topCenter{ |
||||
width: 400rpx; |
||||
height: auto; |
||||
padding-left: 18rpx; |
||||
|
||||
} |
||||
.topCenter .title{ |
||||
font-size: 30rpx; |
||||
font-weight: 400; |
||||
color:#333333; |
||||
line-height: 50rpx; |
||||
width: 100%; |
||||
overflow: hidden; //超出文本隐藏 |
||||
white-space: nowrap; //不换行,只显示一行 |
||||
text-overflow: ellipsis; //超出部分省略号显示 |
||||
} |
||||
.topCenter .desc{ |
||||
font-size: 24rpx; |
||||
color:#999999; |
||||
line-height: 30rpx; |
||||
width: 100%; |
||||
overflow: hidden; //超出文本隐藏 |
||||
white-space: nowrap; //不换行,只显示一行 |
||||
text-overflow: ellipsis; //超 |
||||
} |
||||
|
||||
|
||||
.photoTop .topRight{ |
||||
width: 160rpx; |
||||
height: auto; |
||||
text-align: right; |
||||
|
||||
} |
||||
.topRight .distance{ color: #90A1AA; font-size: 24rpx; font-family: PingFang SC-Regular; padding-top: 20rpx;} |
||||
.topRight .distance image{ width: 32rpx; height: 32rpx; margin-right: 24rpx;} |
||||
|
||||
|
||||
.picList{ width: 100%; height: 360rpx;} |
||||
|
||||
.scrollBox { |
||||
margin-top: 20rpx; |
||||
width: 100%; |
||||
height: 360rpx; |
||||
white-space: nowrap; |
||||
|
||||
} |
||||
|
||||
.picBox { |
||||
display: inline-block; |
||||
width:260rpx; |
||||
height: 360rpx; |
||||
text-align: center; |
||||
position: relative; |
||||
margin-left: 20rpx; |
||||
|
||||
} |
||||
.picBox .pic{ |
||||
position: absolute; |
||||
width:100%; |
||||
height: 360rpx; |
||||
|
||||
|
||||
} |
||||
.picBox .pic image{ |
||||
width:260rpx; |
||||
height: 360rpx; |
||||
border-radius:24rpx; |
||||
} |
||||
.picBox .pic_title{ |
||||
position: absolute; |
||||
width:240rpx; |
||||
height: 72rpx; |
||||
bottom: 0; |
||||
z-index: 2; |
||||
font-size: 24rpx; line-height: 72rpx; |
||||
padding-left: 20rpx; |
||||
color: #FFFFFF; |
||||
text-align: left; |
||||
overflow: hidden; //超出文本隐藏 |
||||
white-space: nowrap; //不换行,只显示一行 |
||||
text-overflow: ellipsis; //超 |
||||
} |
||||
|
||||
</style> |
@ -0,0 +1,275 @@ |
||||
<template> |
||||
<view class="container"> |
||||
|
||||
|
||||
<view class="form"> |
||||
<!--input box start--> |
||||
<view class='input_box'> |
||||
<view class='i_box_1'>头像</view> |
||||
<view class='i_box_2'></view> |
||||
<view class='i_box_3'> |
||||
|
||||
</view> |
||||
<view class='i_box_4'></view> |
||||
<view class='i_box_5'> |
||||
<view class="up"> |
||||
<view class="img"> |
||||
<u-upload |
||||
|
||||
:fileList="wxAvatar" |
||||
@afterRead="afterRead" |
||||
@delete="deletePic" |
||||
name="avatar" |
||||
width="26" |
||||
height="26" |
||||
:sizeType="['compressed']" |
||||
:deletable="true" |
||||
:maxCount="1" |
||||
:previewFullImage="false" |
||||
> |
||||
|
||||
</u-upload> |
||||
</view> |
||||
<view class="icon" @click="clearImg"> |
||||
<u-icon name="arrow-right" top="5"></u-icon> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
<!--input box end--> |
||||
|
||||
|
||||
<!--input box start--> |
||||
<view class='input_box'> |
||||
<view class='i_box_1'>昵称</view> |
||||
<view class='i_box_2'></view> |
||||
<view class='i_box_3x'> |
||||
<u--input |
||||
placeholder="请输入昵称" |
||||
inputAlign="right" |
||||
border="none" |
||||
color="#FFFFFF" |
||||
maxlength="10" |
||||
v-model="info.wxName" |
||||
></u--input> |
||||
</view> |
||||
|
||||
</view> |
||||
<!--input box end--> |
||||
|
||||
<!--input box start--> |
||||
<view class='input_box'> |
||||
<view class='i_box_1'>手机号</view> |
||||
<view class='i_box_2'></view> |
||||
<view class='i_box_3x'> |
||||
<u--input |
||||
v-if="!this.$com.isNull(info.phoneNumber)" |
||||
inputAlign="right" |
||||
border="none" |
||||
color="#FFFFFF" |
||||
maxlength="11" |
||||
v-model="info.phoneNumber" |
||||
:readonly="true" |
||||
></u--input> |
||||
<u-button v-else :text="info.phoneNumber || '点击获取手机号'" openType="getPhoneNumber" @getphonenumber="getPhoneNumber"></u-button> |
||||
</view> |
||||
|
||||
</view> |
||||
<!--input box end--> |
||||
|
||||
</view> |
||||
|
||||
<view class="btn-big"><u-button type="info" :loading="btn_loading" loadingText="加载中" :disabled="btn_disabled" text="保存" @click="add"></u-button></view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
export default { |
||||
data() { |
||||
return { |
||||
//form框 |
||||
info: { |
||||
wxName:'', |
||||
wxAvatar:'', |
||||
phoneNumber:'点击获取手机号', |
||||
|
||||
}, |
||||
wxAvatar: [], |
||||
//按钮设置 |
||||
btn_loading:false, |
||||
btn_disabled:false, |
||||
} |
||||
}, |
||||
onLoad(option) { |
||||
var _this = this; |
||||
_this.getUserInfo(); |
||||
|
||||
}, |
||||
methods: { |
||||
|
||||
//获取用户信息 |
||||
getUserInfo() |
||||
{ |
||||
var _this =this; |
||||
var post ={}; |
||||
_this.$api.post('ycl/user/wx-info',post,function(rs){ |
||||
console.log(rs); |
||||
_this.info = rs; |
||||
_this.wxAvatar =[{url:rs.wxAvatar}]; |
||||
}); |
||||
}, |
||||
// 删除图片 |
||||
deletePic(event) { |
||||
var _this = this; |
||||
_this.wxAvatar = []; |
||||
}, |
||||
// 新增图片 |
||||
afterRead(event) { |
||||
|
||||
var _this = this |
||||
console.log('event',event); |
||||
var edetail = event.file.thumb; |
||||
if (edetail.indexOf('http://tmp') != -1 || edetail.indexOf('wxfile://tmp') != -1) |
||||
{ |
||||
var FSM = uni.getFileSystemManager(); |
||||
FSM.readFile({ |
||||
filePath: edetail, |
||||
encoding: "base64", |
||||
success: function(data) { |
||||
_this.info.imgBase64 = "data:image/png;base64," + data.data |
||||
_this.wxAvatar = [{url:edetail}]; |
||||
}, |
||||
fail: function(err){ |
||||
console.log('http://tmp',err) |
||||
} |
||||
}) |
||||
|
||||
} |
||||
|
||||
}, |
||||
uploadFilePromise(url) { |
||||
|
||||
}, |
||||
loadData(){ |
||||
|
||||
}, |
||||
getPhoneNumber(e){ |
||||
var _this =this; |
||||
var detail = e.detail; |
||||
console.log(detail); |
||||
if (detail.errMsg === 'getPhoneNumber:ok') { |
||||
|
||||
_this.getPhoneApi(detail); |
||||
} else { |
||||
_this.$com.alert('授权失败') |
||||
} |
||||
}, |
||||
getPhoneApi(detail){ |
||||
var _this = this; |
||||
|
||||
var post ={ |
||||
|
||||
code: detail.code, |
||||
}; |
||||
_this.$api.post('ycl/user/wx-phone',post,function(rs){ |
||||
console.log(rs); |
||||
_this.info.phoneNumber = rs; |
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
|
||||
}, |
||||
|
||||
//提交表单 |
||||
add(){ |
||||
var _this =this; |
||||
var info = _this.info; |
||||
|
||||
if(_this.$com.isNull(info.wxName)) |
||||
{ |
||||
_this.$com.showError('请输入昵称'); |
||||
return; |
||||
} |
||||
if(_this.$com.isNull(info.wxAvatar) && _this.$com.isNull(info.imgBase64)) |
||||
{ |
||||
_this.$com.showError('请上传头像'); |
||||
return; |
||||
} |
||||
if(_this.$com.isNull(info.phoneNumber)) |
||||
{ |
||||
_this.$com.showError('手机号必须'); |
||||
return; |
||||
} |
||||
_this.btn_loading = true; |
||||
_this.btn_disabled = true; |
||||
var post = info; |
||||
_this.$api.put('ycl/user/wx-update',post,function(rs){ |
||||
console.log('更新token'); |
||||
_this.$user.session('token',rs.token); |
||||
_this.$com.alert('更新成功'); |
||||
setTimeout(function(){ |
||||
uni.$u.route({ |
||||
type:'navigateBack', |
||||
delta:1, |
||||
}); |
||||
},1000); |
||||
}); |
||||
//按钮还原 |
||||
_this.btn_loading = false; |
||||
_this.btn_disabled = false; |
||||
|
||||
}, |
||||
|
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
.container{padding:10rpx 20rpx;} |
||||
.form{ padding: 20rpx; |
||||
background: #082436; |
||||
border-radius: 24rpx 24rpx 24rpx 24rpx; |
||||
} |
||||
/*form基本元素*/ |
||||
/*input_box*/ |
||||
.input_box{ display: flex;flex-direction:row; height: 100rpx; line-height: 100rpx; |
||||
border-radius: 15rpx;box-shadow: 0 2rpx 2rpx rgba(0, 0, 0, 0.05); margin: 20rpx auto; color:#FFFFFF;} |
||||
|
||||
.i_box_1{ width: 25%;height: 100rpx; font-size: 28rpx; text-align: left; } |
||||
.i_box_2{width: 0.3%;height: 80rpx; margin-top: 10rpx;} |
||||
.i_box_3{ height: 100rpx;min-width: 49.4%; width: auto;font-size: 26rpx;} |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
/*特殊的 改变部分*/ |
||||
.i_box_3x{height: 100rpx;min-width: 74.7%; width: auto;font-size: 26rpx;} |
||||
::v-deep .u-input{ border:none !important; color:#FFFFFF !important; margin-top: 10rpx;} |
||||
.up{ width: 100%; height: auto; display: flex; margin-top: 20rpx;} |
||||
.up .img{width:80%; } |
||||
.up .img .u-upload{ margin-left: 80rpx;} |
||||
.up .icon{ width:20%;} |
||||
::v-deep .i_box_3x .u-button--info{ background-color: transparent !important; border: none !important; color: #FFFFFF !important; text-align: right !important;} |
||||
::v-deep .i_box_3x .u-button{display: block !important; padding:0rpx !important;} |
||||
::v-deep .u-upload__button{border-radius: 50% !important;} |
||||
::v-deep .u-upload__wrap__preview{border-radius: 50% !important;} |
||||
|
||||
|
||||
|
||||
.btn-big{ width:100%; |
||||
height: 92rpx; |
||||
line-height: 92rpx; |
||||
margin:30rpx auto; |
||||
display:block; |
||||
border-radius:56rpx; |
||||
background:#00F0E2; |
||||
font-size:28rpx; |
||||
color:#fff;text-align: center;} |
||||
|
||||
::v-deep .btn-big .u-button--info{ background-color: transparent !important; border: none !important; color: #FFFFFF !important; } |
||||
|
||||
|
||||
</style> |
@ -0,0 +1,176 @@ |
||||
<template> |
||||
<view class="container"> |
||||
|
||||
<view class="input_box"> |
||||
<text class="input_title">{{$t('pages.user.auth.login.mobile')}}</text> |
||||
<view class="input_content"> |
||||
<u--input |
||||
:placeholder="$t('pages.user.auth.login.input_mobile')" |
||||
prefixIcon="phone" |
||||
prefixIconStyle="font-size: 24rpx;color: #909399" |
||||
v-model="form.mobile" |
||||
@change="mobileChange" |
||||
></u--input> |
||||
</view> |
||||
</view> |
||||
|
||||
|
||||
|
||||
<view class="input_box"> |
||||
<text class="input_title">{{$t('pages.user.auth.login.password')}}</text> |
||||
<view class="input_content"> |
||||
<u--input |
||||
:placeholder="$t('pages.user.auth.login.input_password')" |
||||
prefixIcon="lock" |
||||
prefixIconStyle="font-size: 24rpx;color: #909399" |
||||
border="surround" |
||||
password |
||||
clearable |
||||
v-model="form.password" |
||||
@change="pwdChange" |
||||
></u--input> |
||||
</view> |
||||
</view> |
||||
|
||||
|
||||
|
||||
<view class="button_box"> |
||||
<view class="submit"> |
||||
<u-button type="primary" :plain="true" :hairline="true" :text="$t('locale.login')" size="normal" @click="login"></u-button> |
||||
</view> |
||||
</view> |
||||
<view class="footer"> |
||||
<!-- <text class="footer-text">找回密码</text> --> |
||||
<text class="footer-text" @click="regist">{{$t('locale.regist')}}</text> |
||||
<!-- <text class="footer-text">投诉建议</text> --> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
<script> |
||||
export default { |
||||
components: { |
||||
|
||||
}, |
||||
data() { |
||||
return { |
||||
|
||||
form:{ |
||||
mobile:'', |
||||
password:'', |
||||
}, |
||||
}; |
||||
}, |
||||
onShow() { |
||||
var _this =this; |
||||
|
||||
}, |
||||
methods:{ |
||||
//跳转到注册 |
||||
regist(){ |
||||
uni.$u.route({ |
||||
type:'redirectTo', |
||||
url: 'pages/user/auth/regist', |
||||
params: {} |
||||
}); |
||||
}, |
||||
//手机号 |
||||
mobileChange(e) |
||||
{ |
||||
var _this = this; |
||||
_this.form['mobile'] = e; |
||||
|
||||
}, |
||||
//密码 |
||||
pwdChange(e) |
||||
{ |
||||
var _this = this; |
||||
_this.form['password'] = e; |
||||
}, |
||||
login(){ |
||||
var _this =this; |
||||
var form = _this.form; |
||||
if(!_this.$com.check(form.mobile,'mobile')) |
||||
{ |
||||
uni.$u.toast(_this.$t('pages.user.auth.login.mobile_must')); |
||||
return false; |
||||
} |
||||
if(form.password.length < 6) |
||||
{ |
||||
uni.$u.toast(_this.$t('pages.user.auth.login.password_must')); |
||||
return false; |
||||
} |
||||
|
||||
var post = form; |
||||
_this.$api.post('user/log',post,function(rs){ |
||||
console.log(rs); |
||||
_this.$user.session('userInfo',rs.userinfo); |
||||
_this.$user.session('token',rs.token); |
||||
uni.$u.route({ |
||||
type:'navigateBack', |
||||
delta:2, |
||||
params: {} |
||||
}); |
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
}, |
||||
|
||||
}, |
||||
}; |
||||
</script> |
||||
|
||||
<style scoped lang="scss"> |
||||
/*输入框*/ |
||||
.input_box{ |
||||
width: 90%; |
||||
margin: 40rpx auto 40rpx auto; |
||||
height: auto; |
||||
} |
||||
.input_title{ |
||||
font-size: 30rpx; |
||||
color: #ABABAB; |
||||
height: 50rpx; |
||||
line-height: 50rpx; |
||||
margin: 10rpx auto; |
||||
display: block; |
||||
|
||||
} |
||||
|
||||
.button_box { |
||||
width: 90%; |
||||
margin: 40rpx auto 20rpx auto; |
||||
|
||||
} |
||||
|
||||
.submit { |
||||
width: 100%; |
||||
background-color: #4CAF50; |
||||
justify-content: center; |
||||
align-items: center; |
||||
border-radius: 10rpx; |
||||
} |
||||
|
||||
.submit-text { |
||||
color: white; |
||||
padding: 30rpx; |
||||
} |
||||
|
||||
.submit:active { |
||||
background-color: green; |
||||
opacity: 0.5; |
||||
} |
||||
|
||||
.footer { |
||||
flex-direction: row; |
||||
justify-content: center; |
||||
text-align: center; |
||||
align-items: center; |
||||
margin-top: 100rpx; |
||||
} |
||||
|
||||
.footer-text { |
||||
font-size: 28rpx; |
||||
color: #296db5; |
||||
padding: 30rpx; |
||||
} |
||||
</style> |
@ -0,0 +1,205 @@ |
||||
<template> |
||||
<view class="container"> |
||||
|
||||
<view class="input_box"> |
||||
<text class="input_title">{{$t('pages.user.auth.login.mobile')}}</text> |
||||
<view class="input_content"> |
||||
<u--input |
||||
:placeholder="$t('pages.user.auth.login.input_mobile')" |
||||
prefixIcon="phone" |
||||
prefixIconStyle="font-size: 24rpx;color: #909399" |
||||
type="number" |
||||
v-model="form.mobile" |
||||
@change="mobileChange" |
||||
></u--input> |
||||
</view> |
||||
</view> |
||||
|
||||
|
||||
|
||||
<view class="input_box"> |
||||
<text class="input_title">{{$t('pages.user.auth.login.password')}}</text> |
||||
<view class="input_content"> |
||||
<u--input |
||||
:placeholder="$t('pages.user.auth.login.input_password')" |
||||
prefixIcon="lock" |
||||
prefixIconStyle="font-size: 24rpx;color: #909399" |
||||
border="surround" |
||||
password |
||||
clearable |
||||
v-model="form.password" |
||||
@change="pwdChange" |
||||
></u--input> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="input_box"> |
||||
<text class="input_title">{{$t('pages.user.auth.login.email')}}</text> |
||||
<view class="input_content"> |
||||
<u--input |
||||
:placeholder="$t('pages.user.auth.login.input_email')" |
||||
prefixIcon="email" |
||||
prefixIconStyle="font-size: 24rpx;color: #909399" |
||||
v-model="form.email" |
||||
@change="emailChange" |
||||
></u--input> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="button_box"> |
||||
<view class="submit"> |
||||
<u-button type="primary" :plain="true" :hairline="true" :text="$t('locale.regist')" size="normal" @click="regist"></u-button> |
||||
</view> |
||||
</view> |
||||
<view class="footer"> |
||||
<!-- <text class="footer-text">找回密码</text> --> |
||||
<text class="footer-text" @click="login">{{$t('locale.login')}}</text> |
||||
<!-- <text class="footer-text">投诉建议</text> --> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
<script> |
||||
export default { |
||||
components: { |
||||
|
||||
}, |
||||
data() { |
||||
return { |
||||
form:{ |
||||
mobile:'', |
||||
password:'', |
||||
email:'', |
||||
}, |
||||
|
||||
}; |
||||
}, |
||||
onShow() { |
||||
var _this =this; |
||||
|
||||
}, |
||||
methods:{ |
||||
//跳转到登录 |
||||
login(){ |
||||
uni.$u.route({ |
||||
type:'redirectTo', |
||||
url: 'pages/user/auth/login', |
||||
params: {} |
||||
}); |
||||
}, |
||||
//手机号 |
||||
mobileChange(e) |
||||
{ |
||||
var _this = this; |
||||
_this.form['mobile'] = e; |
||||
|
||||
}, |
||||
//密码 |
||||
pwdChange(e) |
||||
{ |
||||
var _this = this; |
||||
_this.form['password'] = e; |
||||
}, |
||||
//email |
||||
emailChange(e) |
||||
{ |
||||
var _this = this; |
||||
_this.form['email'] = e; |
||||
}, |
||||
regist() |
||||
{ |
||||
var _this =this; |
||||
var form = _this.form; |
||||
if(!_this.$com.check(form.mobile,'mobile')) |
||||
{ |
||||
uni.$u.toast(_this.$t('pages.user.auth.login.mobile_must')); |
||||
return false; |
||||
} |
||||
if(form.password.length < 6) |
||||
{ |
||||
uni.$u.toast(_this.$t('pages.user.auth.login.password_must')); |
||||
return false; |
||||
} |
||||
if(!_this.$com.check(form.email,'email')) |
||||
{ |
||||
uni.$u.toast(_this.$t('pages.user.auth.login.email_must')); |
||||
return false; |
||||
} |
||||
|
||||
var post = form; |
||||
_this.$api.post('user/reg',post,function(rs){ |
||||
console.log(rs); |
||||
uni.showModal({ |
||||
title: _this.$t('locale.tips'), |
||||
content: _this.$t('pages.user.auth.regist.regist_success'), |
||||
success: function (res) { |
||||
if (res.confirm) { |
||||
console.log('用户点击确定'); |
||||
_this.login(); |
||||
} else if (res.cancel) { |
||||
console.log('用户点击取消'); |
||||
} |
||||
} |
||||
}); |
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
}, |
||||
}, |
||||
}; |
||||
</script> |
||||
|
||||
<style scoped lang="scss"> |
||||
/*输入框*/ |
||||
.input_box{ |
||||
width: 90%; |
||||
margin: 40rpx auto 40rpx auto; |
||||
height: auto; |
||||
} |
||||
.input_title{ |
||||
font-size: 30rpx; |
||||
color: #ABABAB; |
||||
height: 50rpx; |
||||
line-height: 50rpx; |
||||
margin: 10rpx auto; |
||||
display: block; |
||||
|
||||
} |
||||
|
||||
.button_box { |
||||
width: 90%; |
||||
margin: 40rpx auto 20rpx auto; |
||||
|
||||
} |
||||
|
||||
.submit { |
||||
width: 100%; |
||||
background-color: #4CAF50; |
||||
justify-content: center; |
||||
align-items: center; |
||||
border-radius: 10rpx; |
||||
} |
||||
|
||||
.submit-text { |
||||
color: white; |
||||
padding: 30rpx; |
||||
} |
||||
|
||||
.submit:active { |
||||
background-color: green; |
||||
opacity: 0.5; |
||||
} |
||||
|
||||
.footer { |
||||
flex-direction: row; |
||||
justify-content: center; |
||||
text-align: center; |
||||
align-items: center; |
||||
margin-top: 100rpx; |
||||
} |
||||
|
||||
.footer-text { |
||||
font-size: 28rpx; |
||||
color: #296db5; |
||||
padding: 30rpx; |
||||
} |
||||
</style> |
@ -0,0 +1,536 @@ |
||||
<template> |
||||
<view> |
||||
|
||||
<view class="header"> |
||||
<image class='background' src="/static/img/user/topbg.png" mode="aspectFill"></image> |
||||
<view class="userInfo"> |
||||
<view class="uleft"> |
||||
<u-image :src="userInfo.wxAvatar" width="60px" height="60px" radius="30px" @error="imgErr"></u-image> |
||||
</view> |
||||
<view class="ucenter"> |
||||
<view class="name"><text class="username">{{userInfo.wxName}}</text><text class="level">{{identityinfo.levelTxt}}</text></view> |
||||
</view> |
||||
<view class="uright"> |
||||
<view class="change" v-if="!this.$com.isNull(userInfo.phoneNumber)"> |
||||
<u-button text="切换" openType="getPhoneNumber" @getphonenumber="getPhoneNumber"></u-button> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="userItem"> |
||||
<view class="uItemBox"> |
||||
<view class="num">{{userInfo.totalAmount > 0 ? userInfo.totalAmount / 100 : '0.00'}}</view> |
||||
<view class="text">消费金额(元)</view> |
||||
</view> |
||||
<view class="uItemBox"> |
||||
<view class="num">{{userInfo.totalDur > 0 ? userInfo.totalDur : 0}}</view> |
||||
<view class="text">使用时长(分)</view> |
||||
</view> |
||||
</view> |
||||
|
||||
</view> |
||||
|
||||
<view class="orderBox"> |
||||
<view class="orderBoxTitle">我的订单</view> |
||||
<view class="orderItem"> |
||||
<view class="oItemBox" v-for="(item,key) in ordermenu" :key="key" @click="goTo(item.url)"> |
||||
<view class="image"> |
||||
<image :src="item.icon" mode="widthFix"></image> |
||||
</view> |
||||
<view class="order_text">{{item.label}}</view> |
||||
</view> |
||||
|
||||
</view> |
||||
</view> |
||||
|
||||
<view class="moneyBox" v-if="identityinfo.levelVal < 4"> |
||||
<view class="moneyBoxTitle">收入抽成</view> |
||||
<view class="moneyBoxItem"> |
||||
<view class="mItemBox"> |
||||
<view class="money"> |
||||
<text class="unit">¥</text><text class="num">{{userInfo.todayIncome > 0 ? userInfo.todayIncome / 100 :'0.00'}}</text> |
||||
</view> |
||||
<view class="order_text">今日收入</view> |
||||
</view> |
||||
<view class="mItemBox"> |
||||
<view class="money"> |
||||
<text class="unit">¥</text><text class="num">{{userInfo.totalIncome > 0 ? userInfo.totalIncome / 100 :'0.00'}}</text> |
||||
</view> |
||||
<view class="order_text">总收入</view> |
||||
</view> |
||||
|
||||
<view class="mItemBox"> |
||||
<view class="money"> |
||||
<text class="unit">¥</text><text class="num">{{userInfo.todayLevelIncome > 0 ? userInfo.todayLevelIncome / 100 :'0.00'}}</text> |
||||
</view> |
||||
<view class="order_text">今日抽成</view> |
||||
</view> |
||||
<view class="mItemBox"> |
||||
<view class="money"> |
||||
<text class="unit">¥</text><text class="num">{{userInfo.totalLevelIncome > 0 ? userInfo.totalLevelIncome / 100 :'0.00'}}</text> |
||||
</view> |
||||
<view class="order_text">总抽成</view> |
||||
</view> |
||||
|
||||
</view> |
||||
</view> |
||||
|
||||
|
||||
<view class="serviceBox"> |
||||
<view class="serviceBoxTitle">我的服务</view> |
||||
<view class="serviceItem"> |
||||
<view class="sItemBox" v-for="(item,key) in shopmenu" :key="key" v-if="item.level >= identityinfo.levelVal" @click="goTo(item.url)"> |
||||
<view class="image"> |
||||
<image :src="item.icon" mode="widthFix"></image> |
||||
</view> |
||||
<view class="order_text">{{item.label}}</view> |
||||
</view> |
||||
|
||||
|
||||
</view> |
||||
<view class="ad"> |
||||
<image src="@/static/img/user/banner.jpg" mode="widthFix" @click="openVideo"></image> |
||||
</view> |
||||
</view> |
||||
|
||||
<view class="serviceBox"> |
||||
<view class="serviceBoxTitle">我的工具</view> |
||||
<view class="serviceItem"> |
||||
<view class="sItemBox" v-for="(item,key) in setmenu" :key="key" @click="tools(item)"> |
||||
<view class="image"> |
||||
<image :src="item.icon" mode="widthFix"></image> |
||||
</view> |
||||
<view class="order_text">{{item.label}}</view> |
||||
</view> |
||||
|
||||
|
||||
|
||||
</view> |
||||
</view> |
||||
|
||||
<view class="nowENV" v-if="this.$api.env() == 'development' || this.$api.env() == 'local'"> |
||||
<u-tag v-if="this.$api.env() == 'development'" text="线上测试环境" type="warning" plain plainFill></u-tag> |
||||
<u-tag v-if="this.$api.env() == 'local'" text="本地开发环境" type="error" plain plainFill></u-tag> |
||||
</view> |
||||
|
||||
<view class="empty"> |
||||
</view> |
||||
|
||||
|
||||
<u-popup :show="video_show" :round="10" mode="center" :closeable="true" customStyle="{width:750rpx,height:auto}" @close="closeVideo"> |
||||
<view class="videoBox"> |
||||
<video id="myVideo" :src="video_src" @error="videoErrorCallback" :autoplay="false" controls></video> |
||||
</view> |
||||
</u-popup> |
||||
|
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
export default { |
||||
data() { |
||||
return { |
||||
isLogin:false, //登录 |
||||
userInfo:{ |
||||
wxAvatar:'/static/img/user/avatar.png', |
||||
wxName:'点击登录', |
||||
totalAmount:0.00, |
||||
totalDur:0, |
||||
}, |
||||
isShop:false, //店铺 |
||||
ordermenu: [ |
||||
{ label: '全部', icon: '/static/img/user/order_all.png', url: '/pages/order/list?tab=0'}, |
||||
{ label: '进行中', icon: '/static/img/user/order_ing.png', url: '/pages/order/list?tab=1'}, |
||||
{ label: '已完成', icon: '/static/img/user/order_finish.png', url: '/pages/order/list?tab=2'}, |
||||
], |
||||
shopmenu: [ |
||||
{ label: '店铺管理', icon: '/static/img/order/store.png', url: '/pages/user/shop/shopmanage', level: 3}, |
||||
{ label: '收入记录', icon: '/static/img/user/money_log.png', url: '/pages/user/mine/income', level: 3}, |
||||
{ label: '提现申请', icon: '/static/img/user/withdraw.png', url: '/pages/user/mine/withdraw', level: 3}, |
||||
{ label: '我的下级', icon: '/static/img/user/link.png', url: '/pages/user/mine/leveluser', level: 2}, |
||||
{ label: '查看码值', icon: '/static/img/user/code.png', url: '/pages/user/shop/lookcode', level: 3}, |
||||
{ label: '提现记录', icon: '/static/img/user/withdraw_log.png', url: '/pages/user/mine/withdrawrecord', level: 3}, |
||||
|
||||
], |
||||
setmenu: [ |
||||
{ label: '账户设置', icon: '/static/img/user/account.png', type: 'normal',url: '/pages/user/account/index'}, |
||||
{ label: '联系客服', icon: '/static/img/user/chat.png', type: 'service',url:''}, |
||||
{ label: '代理入驻', icon: '/static/img/user/settled.png', type: 'normal',url: '/pages/user/mine/settled'} |
||||
], |
||||
//等级 |
||||
identityinfo:'', |
||||
|
||||
//视频显示 |
||||
video_show:false, |
||||
video_src:'https://kjc.oss-cn-shenzhen.aliyuncs.com/video/demo.mp4', |
||||
|
||||
}; |
||||
}, |
||||
onReady: function (res) { |
||||
this.videoContext = uni.createVideoContext('myVideo') |
||||
}, |
||||
onShow() { |
||||
var _this =this; |
||||
var openid = _this.$user.session('openid'); |
||||
if(!_this.$com.isNull(openid)) |
||||
{ |
||||
_this.isLogin = true; |
||||
_this.getUserInfo(); |
||||
} |
||||
|
||||
|
||||
}, |
||||
onShareAppMessage() { |
||||
return { |
||||
title: '空间充-我的', |
||||
path: uni.$u.page(), |
||||
} |
||||
}, |
||||
onShareTimeline() { |
||||
return { |
||||
title: '空间充-我的', |
||||
path: uni.$u.page(), |
||||
} |
||||
}, |
||||
methods:{ |
||||
//头像加载失败 |
||||
imgErr(){ |
||||
this.userInfo.wxAvatar = '/static/img/user/avatar.png'; |
||||
}, |
||||
//获取用户信息 |
||||
getUserInfo() |
||||
{ |
||||
var _this =this; |
||||
var post ={}; |
||||
_this.$api.post('ycl/user/wx-info',post,function(rs){ |
||||
console.log(rs); |
||||
_this.userInfo = rs; |
||||
_this.$user.session('lifeData',rs); |
||||
var identityinfo = _this.$access.checkidentity(rs); |
||||
console.log(identityinfo); |
||||
_this.identityinfo = identityinfo; |
||||
_this.$user.session('identityinfo',identityinfo); |
||||
if(identityinfo.levelVal != 4) |
||||
{ |
||||
var setmenu = _this.setmenu; |
||||
var updatedSetmenu = setmenu.filter(item => item.label !== '代理入驻'); |
||||
_this.setmenu = updatedSetmenu; |
||||
} |
||||
|
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
}, |
||||
goTo(url) |
||||
{ |
||||
uni.$u.route({ |
||||
type:'navigateTo', |
||||
url: url, |
||||
}); |
||||
}, |
||||
tools(item) |
||||
{ |
||||
var _this =this; |
||||
if(item.type == 'service') |
||||
{ |
||||
var phone = _this.$user.session('servicePhone'); |
||||
uni.makePhoneCall({ |
||||
phoneNumber: phone, |
||||
success(res) { |
||||
console.log(res); |
||||
}, |
||||
fail(err) { |
||||
console.log(err); |
||||
}, |
||||
}); |
||||
} |
||||
else |
||||
{ |
||||
uni.$u.route({ |
||||
type:'navigateTo', |
||||
url: item.url, |
||||
}); |
||||
} |
||||
|
||||
}, |
||||
//打开视频 |
||||
openVideo(){ |
||||
var _this =this; |
||||
_this.video_show = true; |
||||
}, |
||||
videoErrorCallback(e) |
||||
{ |
||||
console.log(e); |
||||
}, |
||||
closeVideo(){ |
||||
var _this =this; |
||||
_this.video_show = false; |
||||
_this.videoContext.pause = true; |
||||
}, |
||||
|
||||
getPhoneNumber(e){ |
||||
var _this =this; |
||||
var detail = e.detail; |
||||
console.log(detail); |
||||
if (detail.errMsg === 'getPhoneNumber:ok') { |
||||
|
||||
_this.getPhoneApi(detail); |
||||
} else { |
||||
_this.$com.alert('授权失败') |
||||
} |
||||
}, |
||||
getPhoneApi(detail){ |
||||
var _this = this; |
||||
var openid = _this.$user.session('openid'); |
||||
var post ={ |
||||
|
||||
code: detail.code, |
||||
}; |
||||
_this.$api.post('ycl/user/wx-phone',post,function(rs){ |
||||
console.log(rs); |
||||
var phoneNumber = rs; |
||||
_this.userInfo.phoneNumber = phoneNumber; |
||||
if(_this.$com.isNull(openid)) |
||||
{ |
||||
_this.$com.alert('切换失败,请退出并重新进入小程序重试'); |
||||
} |
||||
// #ifdef MP |
||||
//这里目前微信小程序和抖音功能基本一样 |
||||
uni.getProvider({ |
||||
service: 'oauth', |
||||
success: function (res) { |
||||
console.log(res.provider) |
||||
var provider = res.provider[0]; |
||||
|
||||
uni.login({ |
||||
provider:provider, |
||||
scopes:'auth_base', |
||||
success: function (login) { |
||||
console.log(login); |
||||
//发起网络请求 |
||||
var xpost = { |
||||
loginCode: login.code, |
||||
phoneNumber:phoneNumber, |
||||
}; |
||||
_this.$api.post('ycl/user/wx-login', xpost,function(resx){ |
||||
console.log(resx); |
||||
_this.$user.session('openid',resx.wxOpenid); |
||||
_this.$user.session('user_id',resx.id); |
||||
if(!_this.$com.isNull(resx.token)) |
||||
{ |
||||
_this.$user.session('token',resx.token); |
||||
} |
||||
_this.isLogin = true; |
||||
_this.getUserInfo(); |
||||
}); |
||||
}, |
||||
fail:function(rs){ |
||||
console.log('登录失败'+rs.errMsg); |
||||
} |
||||
}); |
||||
|
||||
|
||||
|
||||
|
||||
} |
||||
}); |
||||
|
||||
|
||||
// #endif |
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
|
||||
}, |
||||
|
||||
}, |
||||
}; |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
page{ height:100%; width: 100%;} |
||||
.container{padding:10rpx 20rpx;} |
||||
.header{ width: 100%; height: 500rpx; overflow: hidden; position: relative;} |
||||
.background { |
||||
width: 100%; |
||||
height: 100%; |
||||
position:absolute; |
||||
background-size:100% 100%; |
||||
z-index: -1; |
||||
} |
||||
.userInfo{ width: 670rpx; height: auto; display: flex; margin: 100rpx auto 50rpx auto;} |
||||
.userInfo .uleft{ width: 120rpx; height: 120rpx;} |
||||
|
||||
.userInfo .ucenter{ width: 430rpx; height: 120rpx; line-height: 120rpx;} |
||||
.userInfo .ucenter .name{ width: 100%; height: 120rpx;} |
||||
.userInfo .ucenter .name .username{font-size: 36rpx; color:#FFFFFF; margin-left:20rpx;} |
||||
.userInfo .ucenter .name .level{width: auto; padding: 2rpx 10rpx; |
||||
height: 44rpx; |
||||
background: #095E59; text-align: center; font-size: 24rpx;color: #FFFFFF; |
||||
border-radius: 42rpx 42rpx 42rpx 42rpx; |
||||
border: 1rpx solid #00F0E2; margin-left: 20rpx;} |
||||
.userInfo .uright{ width: 120rpx; height: 120rpx;} |
||||
.userInfo .uright .change{width: 104rpx; |
||||
height: 50rpx; |
||||
background: #00F0E2; |
||||
border-radius: 42rpx 42rpx 42rpx 42rpx; color: #1E2629; font-size: 28rpx; line-height: 50rpx; text-align: center;} |
||||
::v-deep .change .u-button{display: block !important; padding:0rpx !important; background-color: inherit !important; height:auto !important; border: none !important; border-radius: 42rpx !important;} |
||||
.userItem{ |
||||
color: #FFFFFF; |
||||
width: 100%; |
||||
height: auto; |
||||
display: flex; |
||||
} |
||||
.uItemBox{ width:50%; text-align: center;} |
||||
.uItemBox .num{ |
||||
font-size: 40rpx; |
||||
font-family: DIN, DIN; |
||||
font-weight: bold; |
||||
} |
||||
.uItemBox .text{ |
||||
font-size: 24rpx; |
||||
font-family: PingFang SC, PingFang SC; |
||||
font-weight: 400; |
||||
line-height: 28rpx; |
||||
} |
||||
|
||||
/*订单部分*/ |
||||
|
||||
.orderBox{ |
||||
width: 670rpx; |
||||
height: auto; |
||||
background: #082436; |
||||
border-radius: 24rpx 24rpx 24rpx 24rpx; |
||||
opacity: 1; |
||||
margin: 0rpx auto; |
||||
padding: 20rpx; |
||||
} |
||||
.orderBoxTitle{ |
||||
font-size: 28rpx; |
||||
font-weight: bold; |
||||
color:#FFFFFF; |
||||
line-height: 60rpx; |
||||
} |
||||
|
||||
.orderItem{ |
||||
width: 100%; height: auto; |
||||
display: flex; |
||||
text-align: center; |
||||
} |
||||
.oItemBox{ |
||||
width: 33.33%; |
||||
|
||||
} |
||||
.oItemBox .image{ width: 100%; text-align: center; margin-top: 20rpx;} |
||||
.oItemBox .image image{width: 64rpx; height: auto; margin: 0rpx auto; } |
||||
.oItemBox .order_text{ |
||||
text-align: center; |
||||
font-size: 24rpx; |
||||
font-weight: 400; |
||||
line-height: 30rpx; |
||||
color: #FFFFFF; |
||||
} |
||||
|
||||
|
||||
/*收入部分*/ |
||||
|
||||
.moneyBox{ |
||||
width: 670rpx; |
||||
height: auto; |
||||
background: #082436; |
||||
border-radius: 24rpx 24rpx 24rpx 24rpx; |
||||
opacity: 1; |
||||
margin: 20rpx auto; |
||||
padding: 20rpx; |
||||
} |
||||
.moneyBoxTitle{ |
||||
font-size: 28rpx; |
||||
font-weight: bold; |
||||
color:#FFFFFF; |
||||
line-height: 40rpx; |
||||
} |
||||
|
||||
.moneyBoxItem{ |
||||
margin-top: 20rpx; |
||||
width: 100%; height: auto; |
||||
display: flex; |
||||
text-align: center; |
||||
gap:20rpx; |
||||
} |
||||
.mItemBox{ |
||||
width: 152rpx; |
||||
height: 128rpx; |
||||
background: linear-gradient(180deg, rgba(49,156,255,0) 0%, rgba(0,255,240,0.5) 100%); |
||||
border-radius: 16rpx 16rpx 16rpx 16rpx; |
||||
opacity: 1; |
||||
border: 1rpx solid #00F0E2; |
||||
|
||||
|
||||
} |
||||
.mItemBox .money{ width: 100%; text-align: center; color: #FFFFFF; line-height: 80rpx;} |
||||
.mItemBox .money .unit{ font-size: 24rpx; |
||||
font-family: PingFang SC, PingFang SC; |
||||
font-weight: bold;} |
||||
.mItemBox .money .num{ font-size: 30rpx; |
||||
font-family: DIN, DIN; |
||||
font-weight: 400; |
||||
} |
||||
.mItemBox .order_text{ |
||||
text-align: center; |
||||
font-size: 24rpx; |
||||
font-weight: 400; |
||||
line-height: 30rpx; |
||||
color: #FFECAA; |
||||
} |
||||
|
||||
|
||||
|
||||
|
||||
/*服务*/ |
||||
|
||||
.serviceBox{ |
||||
width: 670rpx; |
||||
height: auto; |
||||
background: #082436; |
||||
border-radius: 24rpx 24rpx 24rpx 24rpx; |
||||
opacity: 1; |
||||
margin: 20rpx auto; |
||||
padding: 20rpx; |
||||
} |
||||
.serviceBoxTitle{ |
||||
font-size: 28rpx; |
||||
font-weight: bold; |
||||
color:#FFFFFF; |
||||
line-height: 60rpx; |
||||
} |
||||
|
||||
.serviceItem{ |
||||
width: 100%; height: auto; |
||||
display: flex; |
||||
text-align: center; |
||||
flex-wrap:wrap; |
||||
} |
||||
.sItemBox{ |
||||
width: 25%; |
||||
|
||||
} |
||||
.sItemBox .image{ width: 100%; text-align: center; margin-top: 20rpx;} |
||||
.sItemBox .image image{width: 48rpx; height: auto; margin: 0rpx auto; } |
||||
.sItemBox .order_text{ |
||||
text-align: center; |
||||
font-size: 24rpx; |
||||
font-weight: 400; |
||||
line-height: 30rpx; |
||||
color: #FFFFFF; |
||||
} |
||||
.nowENV{ width: 200rpx; height: 60rpx; margin: 10rpx auto; text-align: center;} |
||||
.empty{width: 100%; height: 20rpx;} |
||||
|
||||
|
||||
.ad{ width: 100%; height: auto; text-align: center; margin:0rpx auto;} |
||||
.ad image{ width:100%; margin-top:20rpx;border-radius: 30rpx;} |
||||
.videoBox{ width: 750rpx; height: 900rpx;} |
||||
.videoBox video{ width: 100%; height: 100%;} |
||||
|
||||
</style> |
@ -0,0 +1,220 @@ |
||||
|
||||
|
||||
|
||||
<template> |
||||
<view class="u-page"> |
||||
|
||||
<view class="empty-select"> |
||||
<u-cell |
||||
:titleStyle="{fontWeight: 500}" |
||||
@click="changeLang" |
||||
:title="$t('pages.user.setting.index.setLang')" |
||||
:value="nowLang" |
||||
isLink |
||||
> |
||||
<image |
||||
slot="icon" |
||||
class="u-cell-icon" |
||||
src="@/static/img/common/lang.png" |
||||
mode="widthFix" |
||||
></image> |
||||
</u-cell> |
||||
</view> |
||||
|
||||
<view class="btn_box"> |
||||
<u-button type="primary" :plain="true" :hairline="true" :text="$t('locale.logout')" size="normal" @click="logout"></u-button> |
||||
</view> |
||||
|
||||
|
||||
<u-action-sheet |
||||
:show="changeLangShow" |
||||
@close="changeLangShow = false" |
||||
:actions="langList" |
||||
@select="confirmLang" |
||||
cancelText="取消" |
||||
> |
||||
</u-action-sheet> |
||||
|
||||
|
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
data() { |
||||
|
||||
return { |
||||
systemLocale: '', |
||||
applicationLocale: '', |
||||
nowLang:'',//当前语言 |
||||
// list: [{ |
||||
// imgName: 'lang', |
||||
// title: this.$t('pages.user.setting.index.setLang'), |
||||
// iconUrl: '/static/img/common/lang.png', |
||||
// }, |
||||
// ], |
||||
changeLangShow:false, |
||||
langList: [ |
||||
{ |
||||
name: this.$t('locale.zh-hans'), |
||||
val:'zh-Hans', |
||||
}, |
||||
{ |
||||
name: this.$t('locale.zh-hant'), |
||||
val:'zh-Hant', |
||||
}, |
||||
{ |
||||
name: this.$t('locale.en'), |
||||
val:'en' |
||||
}], |
||||
langListVal:['zh-Hans','zh-Hant','en'], |
||||
} |
||||
}, |
||||
onLoad() { |
||||
var _this =this; |
||||
_this.loadData(); |
||||
}, |
||||
methods: { |
||||
loadData(){ |
||||
var _this =this; |
||||
let systemInfo = uni.getSystemInfoSync(); |
||||
_this.systemLocale = systemInfo.language; |
||||
var lang = uni.getLocale(); |
||||
console.log('lang:'+lang); |
||||
_this.applicationLocale =lang; |
||||
_this.returnLang(lang); |
||||
_this.isAndroid = systemInfo.platform.toLowerCase() === 'android'; |
||||
uni.onLocaleChange((e) => { |
||||
_this.applicationLocale = e.locale; |
||||
_this.returnLang(e.locale); |
||||
}) |
||||
}, |
||||
returnLang(code) |
||||
{ |
||||
var _this =this; |
||||
var list = _this.langListVal; |
||||
list.forEach((item, $index, arr) => { |
||||
//console.log(`${item} => ${$index} => ${arr}`); |
||||
if(item == code) |
||||
{ |
||||
console.log(_this.langList[$index].name); |
||||
_this.nowLang = _this.langList[$index].name; |
||||
return; |
||||
} |
||||
}) |
||||
}, |
||||
//改变当前语言 |
||||
changeLang() { |
||||
//this[`show${index}`] = true |
||||
this[`changeLangShow`] = true; |
||||
}, |
||||
confirmLang(e){ |
||||
console.log(e); |
||||
var _this =this; |
||||
_this.onLocaleChange(e.val); |
||||
}, |
||||
|
||||
onLocaleChange(code) { |
||||
var _this =this; |
||||
_this[`changeLangShow`] = false; |
||||
uni.showModal({ |
||||
content: _this.$t('locale.language-change-confirm'), |
||||
success: (res) => { |
||||
if (res.confirm) { |
||||
uni.setLocale(code); |
||||
_this.$i18n.locale = code; |
||||
_this.returnLang(code); |
||||
uni.switchTab({ |
||||
url:'pages/index/index' |
||||
}) |
||||
} |
||||
} |
||||
}) |
||||
|
||||
|
||||
}, |
||||
|
||||
//退出登录 |
||||
logout(){ |
||||
var _this =this; |
||||
var post = {}; |
||||
_this.$api.post('user/logout',post,function(rs){ |
||||
console.log(rs); |
||||
uni.removeStorageSync('userInfo'); |
||||
uni.removeStorageSync('token'); |
||||
uni.$u.route({ |
||||
type:'switchTab', |
||||
url:'pages/index/index', |
||||
params: {} |
||||
}); |
||||
},function(err){ |
||||
console.log(err); |
||||
}); |
||||
uni.$u.toast(_this.$t('locale.logout')); |
||||
|
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
.u-block{ |
||||
padding: 14px; |
||||
&__section{ |
||||
margin-bottom:10px; |
||||
} |
||||
&__title { |
||||
margin-top:10px; |
||||
font-size: 15px; |
||||
color: $u-content-color; |
||||
margin-bottom:10px; |
||||
} |
||||
&__flex{ |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
} |
||||
} |
||||
|
||||
// 使用了cell组件的icon图片样式 |
||||
.u-cell-icon { |
||||
width: 36rpx; |
||||
height: 36rpx; |
||||
margin-right: 8rpx; |
||||
} |
||||
|
||||
.u-page { |
||||
padding: 15px 15px 40px 15px; |
||||
} |
||||
|
||||
.u-demo-block { |
||||
flex: 1; |
||||
margin-bottom: 23px; |
||||
|
||||
&__content { |
||||
@include flex(column); |
||||
} |
||||
|
||||
&__title { |
||||
font-size: 14px; |
||||
color: rgb(143, 156, 162); |
||||
margin-bottom: 8px; |
||||
@include flex; |
||||
} |
||||
} |
||||
|
||||
|
||||
.u-page { |
||||
padding: 40rpx 0px; |
||||
|
||||
&__top-box { |
||||
padding-left: 40rpx; |
||||
} |
||||
} |
||||
|
||||
.empty-select { |
||||
margin-top: 0px; |
||||
} |
||||
|
||||
.btn_box{ width: 80%; margin: 120rpx auto 20rpx auto;} |
||||
</style> |
@ -0,0 +1,77 @@ |
||||
<template> |
||||
<view> |
||||
<web-view ref="webview" :src="xurl" :update-title="true" :webview-styles="webviewStyles" @message="getMessage"></web-view> |
||||
</view> |
||||
</template> |
||||
|
||||
|
||||
<script> |
||||
var wv;//计划创建的webview |
||||
export default { |
||||
data() { |
||||
return { |
||||
xurl: '', |
||||
old_url:'', |
||||
webviewStyles: { |
||||
progress: { |
||||
color: '#FF3333' |
||||
} |
||||
}, |
||||
}; |
||||
}, |
||||
onReady() { |
||||
|
||||
}, |
||||
onLoad(option) { |
||||
//数据 |
||||
var _this =this; |
||||
var url = option.url; |
||||
var title = option.title; |
||||
if(_this.$com.isNull(url)) |
||||
{ |
||||
uni.navigateBack({ |
||||
delta:1, |
||||
}) |
||||
} |
||||
_this.old_url = url; |
||||
_this.xurl = decodeURIComponent(url); |
||||
if(!_this.$com.isNull(title)) |
||||
{ |
||||
var pageTitle = title; |
||||
} |
||||
else |
||||
{ |
||||
var pages = getCurrentPages(); |
||||
var page = pages[pages.length - 1]; |
||||
// #ifndef APP-PLUS |
||||
var pageTitle = page.$page.meta.navigationBar.titleText; |
||||
// #endif |
||||
|
||||
// #ifdef APP-PLUS |
||||
var webView = currentPage.$getAppWebview(); |
||||
var pageTitle = webView.getStyle().name; |
||||
// #endif |
||||
|
||||
if(_this.$com.isNull(pageTitle)) |
||||
{ |
||||
pageTitle = 'webview'; |
||||
} |
||||
} |
||||
uni.setNavigationBarTitle({ |
||||
title:pageTitle, |
||||
}) |
||||
|
||||
}, |
||||
methods: { |
||||
getMessage(event) { |
||||
// #ifdef APP-PLUS |
||||
console.log('event', event); |
||||
// #endif |
||||
}, |
||||
}, |
||||
}; |
||||
</script> |
||||
|
||||
<style> |
||||
|
||||
</style> |
@ -0,0 +1,53 @@ |
||||
<template> |
||||
<view class="container"> |
||||
<view class="videoBox"> |
||||
<video id="myVideo" :style="{width: '100%', height: '100%'}" |
||||
:src="video_src" @error="videoErrorCallback" :autoplay="false" controls></video> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
data() { |
||||
return { |
||||
old_url:'', |
||||
video_src:'', |
||||
} |
||||
}, |
||||
onLoad(option) { |
||||
//数据 |
||||
var _this =this; |
||||
var url = option.url; |
||||
var title = option.title; |
||||
if(_this.$com.isNull(url)) |
||||
{ |
||||
uni.navigateBack({ |
||||
delta:1, |
||||
}) |
||||
} |
||||
_this.old_url = url; |
||||
_this.video_src = decodeURIComponent(url); |
||||
if(!_this.$com.isNull(title)) |
||||
{ |
||||
var pageTitle = title; |
||||
uni.setNavigationBarTitle({ |
||||
title:pageTitle, |
||||
}) |
||||
} |
||||
|
||||
}, |
||||
methods: { |
||||
videoErrorCallback(e) |
||||
{ |
||||
console.log(e); |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
page{ height:100%; width: 100%;} |
||||
.container{width: 100%; height: 100%;} |
||||
.videoBox{ width: 100%; height: 100%;} |
||||
</style> |
After Width: | Height: | Size: 865 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 866 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 2.8 KiB |
After Width: | Height: | Size: 144 KiB |
After Width: | Height: | Size: 3.3 KiB |
After Width: | Height: | Size: 93 KiB |
After Width: | Height: | Size: 3.4 KiB |
After Width: | Height: | Size: 31 KiB |
After Width: | Height: | Size: 211 B |
After Width: | Height: | Size: 36 KiB |
After Width: | Height: | Size: 4.1 KiB |
After Width: | Height: | Size: 4.0 KiB |
After Width: | Height: | Size: 5.7 KiB |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 3.5 KiB |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 6.0 KiB |
After Width: | Height: | Size: 35 KiB |
After Width: | Height: | Size: 1.5 KiB |
After Width: | Height: | Size: 805 B |
After Width: | Height: | Size: 2.5 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 24 KiB |
After Width: | Height: | Size: 3.4 KiB |
After Width: | Height: | Size: 4.2 KiB |
After Width: | Height: | Size: 4.0 KiB |
After Width: | Height: | Size: 108 KiB |
After Width: | Height: | Size: 16 KiB |
@ -0,0 +1,78 @@ |
||||
/** |
||||
* 这里是uni-app内置的常用样式变量 |
||||
* |
||||
* uni-app 官方扩展插件及插件市场(https://ext.dcloud.net.cn)上很多三方插件均使用了这些样式变量 |
||||
* 如果你是插件开发者,建议你使用scss预处理,并在插件代码中直接使用这些变量(无需 import 这个文件),方便用户通过搭积木的方式开发整体风格一致的App |
||||
* |
||||
*/ |
||||
@import '@/uni_modules/uview-ui/theme.scss'; |
||||
/** |
||||
* 如果你是App开发者(插件使用者),你可以通过修改这些变量来定制自己的插件主题,实现自定义主题功能 |
||||
* |
||||
* 如果你的项目同样使用了scss预处理,你也可以直接在你的 scss 代码中使用如下变量,同时无需 import 这个文件 |
||||
*/ |
||||
|
||||
/* 颜色变量 */ |
||||
|
||||
/* 行为相关颜色 */ |
||||
$uni-color-primary: #007aff; |
||||
$uni-color-success: #4cd964; |
||||
$uni-color-warning: #f0ad4e; |
||||
$uni-color-error: #dd524d; |
||||
|
||||
/* 文字基本颜色 */ |
||||
$uni-text-color:#333;//基本色 |
||||
$uni-text-color-inverse:#fff;//反色 |
||||
$uni-text-color-grey:#999;//辅助灰色,如加载更多的提示信息 |
||||
$uni-text-color-placeholder: #808080; |
||||
$uni-text-color-disable:#c0c0c0; |
||||
|
||||
/* 背景颜色 */ |
||||
$uni-bg-color:#ffffff; |
||||
$uni-bg-color-grey:#f8f8f8; |
||||
$uni-bg-color-hover:#f1f1f1;//点击状态颜色 |
||||
$uni-bg-color-mask:rgba(0, 0, 0, 0.4);//遮罩颜色 |
||||
|
||||
/* 边框颜色 */ |
||||
$uni-border-color:#c8c7cc; |
||||
|
||||
/* 尺寸变量 */ |
||||
|
||||
/* 文字尺寸 */ |
||||
$uni-font-size-sm:12px; |
||||
$uni-font-size-base:14px; |
||||
$uni-font-size-lg:16; |
||||
|
||||
/* 图片尺寸 */ |
||||
$uni-img-size-sm:20px; |
||||
$uni-img-size-base:26px; |
||||
$uni-img-size-lg:40px; |
||||
|
||||
/* Border Radius */ |
||||
$uni-border-radius-sm: 2px; |
||||
$uni-border-radius-base: 3px; |
||||
$uni-border-radius-lg: 6px; |
||||
$uni-border-radius-circle: 50%; |
||||
|
||||
/* 水平间距 */ |
||||
$uni-spacing-row-sm: 5px; |
||||
$uni-spacing-row-base: 10px; |
||||
$uni-spacing-row-lg: 15px; |
||||
|
||||
/* 垂直间距 */ |
||||
$uni-spacing-col-sm: 4px; |
||||
$uni-spacing-col-base: 8px; |
||||
$uni-spacing-col-lg: 12px; |
||||
|
||||
/* 透明度 */ |
||||
$uni-opacity-disabled: 0.3; // 组件禁用态的透明度 |
||||
|
||||
/* 文章场景相关 */ |
||||
$uni-color-title: #2C405A; // 文章标题颜色 |
||||
$uni-font-size-title:20px; |
||||
$uni-color-subtitle: #555555; // 二级标题颜色 |
||||
$uni-font-size-subtitle:26px; |
||||
$uni-color-paragraph: #3F536E; // 文章段落颜色 |
||||
$uni-font-size-paragraph:15px; |
||||
|
||||
page{background-color: #F5F6FB;} |
@ -0,0 +1,21 @@ |
||||
MIT License |
||||
|
||||
Copyright (c) 2020 www.uviewui.com |
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy |
||||
of this software and associated documentation files (the "Software"), to deal |
||||
in the Software without restriction, including without limitation the rights |
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||
copies of the Software, and to permit persons to whom the Software is |
||||
furnished to do so, subject to the following conditions: |
||||
|
||||
The above copyright notice and this permission notice shall be included in all |
||||
copies or substantial portions of the Software. |
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
||||
SOFTWARE. |
@ -0,0 +1,66 @@ |
||||
<p align="center"> |
||||
<img alt="logo" src="https://uviewui.com/common/logo.png" width="120" height="120" style="margin-bottom: 10px;"> |
||||
</p> |
||||
<h3 align="center" style="margin: 30px 0 30px;font-weight: bold;font-size:40px;">uView 2.0</h3> |
||||
<h3 align="center">多平台快速开发的UI框架</h3> |
||||
|
||||
[](https://github.com/umicro/uView2.0) |
||||
[](https://github.com/umicro/uView2.0) |
||||
[](https://github.com/umicro/uView2.0/issues) |
||||
[](https://uviewui.com) |
||||
[](https://gitee.com/umicro/uView2.0/releases) |
||||
[](https://en.wikipedia.org/wiki/MIT_License) |
||||
|
||||
## 说明 |
||||
|
||||
uView UI,是[uni-app](https://uniapp.dcloud.io/)全面兼容nvue的uni-app生态框架,全面的组件和便捷的工具会让您信手拈来,如鱼得水 |
||||
|
||||
## [官方文档:https://uviewui.com](https://uviewui.com) |
||||
|
||||
|
||||
## 预览 |
||||
|
||||
您可以通过**微信**扫码,查看最佳的演示效果。 |
||||
<br> |
||||
<br> |
||||
<img src="https://uviewui.com/common/weixin_mini_qrcode.png" width="220" height="220" > |
||||
|
||||
|
||||
## 链接 |
||||
|
||||
- [官方文档](https://www.uviewui.com/) |
||||
- [更新日志](https://www.uviewui.com/components/changelog.html) |
||||
- [升级指南](https://www.uviewui.com/components/changeGuide.html) |
||||
- [关于我们](https://www.uviewui.com/cooperation/about.html) |
||||
|
||||
## 交流反馈 |
||||
|
||||
欢迎加入我们的QQ群交流反馈:[点此跳转](https://www.uviewui.com/components/addQQGroup.html) |
||||
|
||||
## 关于PR |
||||
|
||||
> 我们非常乐意接受各位的优质PR,但在此之前我希望您了解uView2.0是一个需要兼容多个平台的(小程序、h5、ios app、android app)包括nvue页面、vue页面。 |
||||
> 所以希望在您修复bug并提交之前尽可能的去这些平台测试一下兼容性。最好能携带测试截图以方便审核。非常感谢! |
||||
|
||||
## 安装 |
||||
|
||||
#### **uni-app插件市场链接** —— [https://ext.dcloud.net.cn/plugin?id=1593](https://ext.dcloud.net.cn/plugin?id=1593) |
||||
|
||||
请通过[官网安装文档](https://www.uviewui.com/components/install.html)了解更详细的内容 |
||||
|
||||
## 快速上手 |
||||
|
||||
请通过[快速上手](https://uviewui.com/components/quickstart.html)了解更详细的内容 |
||||
|
||||
## 使用方法 |
||||
配置easycom规则后,自动按需引入,无需`import`组件,直接引用即可。 |
||||
|
||||
```html |
||||
<template> |
||||
<u-button text="按钮"></u-button> |
||||
</template> |
||||
``` |
||||
|
||||
## 版权信息 |
||||
uView遵循[MIT](https://en.wikipedia.org/wiki/MIT_License)开源协议,意味着您无需支付任何费用,也无需授权,即可将uView应用到您的产品中。 |
||||
|
@ -0,0 +1,344 @@ |
||||
## 2.0.33(2022-06-17) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复`loadmore`组件`lineColor`类型错误问题 |
||||
2. 修复`u-parse`组件`imgtap`、`linktap`不生效问题 |
||||
## 2.0.32(2022-06-16) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
1. `u-loadmore`新增自定义颜色、虚/实线 |
||||
2. 修复`u-swiper-action`组件部分平台不能上下滑动的问题 |
||||
3. 修复`u-list`回弹问题 |
||||
4. 修复`notice-bar`组件动画在低端安卓机可能会抖动的问题 |
||||
5. `u-loading-page`添加控制图标大小的属性`iconSize` |
||||
6. 修复`u-tooltip`组件`color`参数不生效的问题 |
||||
7. 修复`u--input`组件使用`blur`事件输出为`undefined`的bug |
||||
8. `u-code-input`组件新增键盘弹起时,是否自动上推页面参数`adjustPosition` |
||||
9. 修复`image`组件`load`事件无回调对象问题 |
||||
10. 修复`button`组件`loadingSize`设置无效问题 |
||||
10. 其他修复 |
||||
## 2.0.31(2022-04-19) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复`upload`在`vue`页面上传成功后没有成功标志的问题 |
||||
2. 解决演示项目中微信小程序模拟上传图片一直出于上传中问题 |
||||
3. 修复`u-code-input`组件在`nvue`页面编译到`app`平台上光标异常问题(`app`去除此功能) |
||||
4. 修复`actionSheet`组件标题关闭按钮点击事件名称错误的问题 |
||||
5. 其他修复 |
||||
## 2.0.30(2022-04-04) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. `u-rate`增加`readonly`属性 |
||||
2. `tabs`滑块支持设置背景图片 |
||||
3. 修复`u-subsection` `mode`为`subsection`时,滑块样式不正确的问题 |
||||
4. `u-code-input`添加光标效果动画 |
||||
5. 修复`popup`的`open`事件不触发 |
||||
6. 修复`u-flex-column`无效的问题 |
||||
7. 修复`u-datetime-picker`索引在特定场合异常问题 |
||||
8. 修复`u-datetime-picker`最小时间字符串模板错误问题 |
||||
9. `u-swiper`添加`m3u8`验证 |
||||
10. `u-swiper`修改判断image和video逻辑 |
||||
11. 修复`swiper`无法使用本地图片问题,增加`type`参数 |
||||
12. 修复`u-row-notice`格式错误问题 |
||||
13. 修复`u-switch`组件当`unit`为`rpx`时,`nodeStyle`消失的问题 |
||||
14. 修复`datetime-picker`组件`showToolbar`与`visibleItemCount`属性无效的问题 |
||||
15. 修复`upload`组件条件编译位置判断错误,导致`previewImage`属性设置为`false`时,整个组件都会被隐藏的问题 |
||||
16. 修复`u-checkbox-group`设置`shape`属性无效的问题 |
||||
17. 修复`u-upload`的`capture`传入字符串的时候不生效的问题 |
||||
18. 修复`u-action-sheet`组件,关闭事件逻辑错误的问题 |
||||
19. 修复`u-list`触顶事件的触发错误的问题 |
||||
20. 修复`u-text`只有手机号可拨打的问题 |
||||
21. 修复`u-textarea`不能换行的问题 |
||||
22. 其他修复 |
||||
## 2.0.29(2022-03-13) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复`u--text`组件设置`decoration`属性未生效的问题 |
||||
2. 修复`u-datetime-picker`使用`formatter`后返回值不正确 |
||||
3. 修复`u-datetime-picker` `intercept` 可能为undefined |
||||
4. 修复已设置单位 uni..config.unit = 'rpx'时,线型指示器 `transform` 的位置翻倍,导致指示器超出宽度 |
||||
5. 修复mixin中bem方法生成的类名在支付宝和字节小程序中失效 |
||||
6. 修复默认值传值为空的时候,打开`u-datetime-picker`报错,不能选中第一列时间的bug |
||||
7. 修复`u-datetime-picker`使用`formatter`后返回值不正确 |
||||
8. 修复`u-image`组件`loading`无效果的问题 |
||||
9. 修复`config.unit`属性设为`rpx`时,导航栏占用高度不足导致塌陷的问题 |
||||
10. 修复`u-datetime-picker`组件`itemHeight`无效问题 |
||||
11. 其他修复 |
||||
## 2.0.28(2022-02-22) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. search组件新增searchIconSize属性 |
||||
2. 兼容Safari/Webkit中传入时间格式如2022-02-17 12:00:56 |
||||
3. 修复text value.js 判断日期出format错误问题 |
||||
4. priceFormat格式化金额出现精度错误 |
||||
5. priceFormat在部分情况下出现精度损失问题 |
||||
6. 优化表单rules提示 |
||||
7. 修复avatar组件src为空时,展示状态不对 |
||||
8. 其他修复 |
||||
## 2.0.27(2022-01-28) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1.样式修复 |
||||
## 2.0.26(2022-01-28) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1.样式修复 |
||||
## 2.0.25(2022-01-27) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复text组件mode=price时,可能会导致精度错误的问题 |
||||
2. 添加$u.setConfig()方法,可设置uView内置的config, props, zIndex, color属性,详见:[修改uView内置配置方案](https://uviewui.com/components/setting.html#%E9%BB%98%E8%AE%A4%E5%8D%95%E4%BD%8D%E9%85%8D%E7%BD%AE) |
||||
3. 优化form组件在errorType=toast时,如果输入错误页面会有抖动的问题 |
||||
4. 修复$u.addUnit()对配置默认单位可能无效的问题 |
||||
## 2.0.24(2022-01-25) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复swiper在current指定非0时缩放有误 |
||||
2. 修复u-icon添加stop属性的时候报错 |
||||
3. 优化遗留的通过正则判断rpx单位的问题 |
||||
4. 优化Layout布局 vue使用gutter时,会超出固定区域 |
||||
5. 优化search组件高度单位问题(rpx -> px) |
||||
6. 修复u-image slot 加载和错误的图片失去了高度 |
||||
7. 修复u-index-list中footer插槽与header插槽存在性判断错误 |
||||
8. 修复部分机型下u-popup关闭时会闪烁 |
||||
9. 修复u-image在nvue-app下失去宽高 |
||||
10. 修复u-popup运行报错 |
||||
11. 修复u-tooltip报错 |
||||
12. 修复box-sizing在app下的警告 |
||||
13. 修复u-navbar在小程序中报运行时错误 |
||||
14. 其他修复 |
||||
## 2.0.23(2022-01-24) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复image组件在hx3.3.9的nvue下可能会显示异常的问题 |
||||
2. 修复col组件gutter参数带rpx单位处理不正确的问题 |
||||
3. 修复text组件单行时无法显示省略号的问题 |
||||
4. navbar添加titleStyle参数 |
||||
5. 升级到hx3.3.9可消除nvue下控制台样式警告的问题 |
||||
## 2.0.22(2022-01-19) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. $u.page()方法优化,避免在特殊场景可能报错的问题 |
||||
2. picker组件添加immediateChange参数 |
||||
3. 新增$u.pages()方法 |
||||
## 2.0.21(2022-01-19) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 优化:form组件在用户设置rules的时候提示用户model必传 |
||||
2. 优化遗留的通过正则判断rpx单位的问题 |
||||
3. 修复微信小程序环境中tabbar组件开启safeAreaInsetBottom属性后,placeholder高度填充不正确 |
||||
4. 修复swiper在current指定非0时缩放有误 |
||||
5. 修复u-icon添加stop属性的时候报错 |
||||
6. 修复upload组件在accept=all的时候没有作用 |
||||
7. 修复在text组件mode为phone时call属性无效的问题 |
||||
8. 处理u-form clearValidate方法 |
||||
9. 其他修复 |
||||
## 2.0.20(2022-01-14) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复calendar默认会选择一个日期,如果直接点确定的话,无法取到值的问题 |
||||
2. 修复Slider缺少disabled props 还有注释 |
||||
3. 修复u-notice-bar点击事件无法拿到index索引值的问题 |
||||
4. 修复u-collapse-item在vue文件下,app端自定义插槽不生效的问题 |
||||
5. 优化头像为空时显示默认头像 |
||||
6. 修复图片地址赋值后判断加载状态为完成问题 |
||||
7. 修复日历滚动到默认日期月份区域 |
||||
8. search组件暴露点击左边icon事件 |
||||
9. 修复u-form clearValidate方法不生效 |
||||
10. upload h5端增加返回文件参数(文件的name参数) |
||||
11. 处理upload选择文件后url为blob类型无法预览的问题 |
||||
12. u-code-input 修复输入框没有往左移出一半屏幕 |
||||
13. 修复Upload上传 disabled为true时,控制台报hoverClass类型错误 |
||||
14. 临时处理ios app下grid点击坍塌问题 |
||||
15. 其他修复 |
||||
## 2.0.19(2021-12-29) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 优化微信小程序包体积可在微信中预览,请升级HbuilderX3.3.4,同时在“运行->运行到小程序模拟器”中勾选“运行时是否压缩代码” |
||||
2. 优化微信小程序setData性能,处理某些方法如$u.route()无法在模板中使用的问题 |
||||
3. navbar添加autoBack参数 |
||||
4. 允许avatar组件的事件冒泡 |
||||
5. 修复cell组件报错问题 |
||||
6. 其他修复 |
||||
## 2.0.18(2021-12-28) |
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复app端编译报错问题 |
||||
2. 重新处理微信小程序端setData过大的性能问题 |
||||
3. 修复边框问题 |
||||
4. 修复最大最小月份不大于0则没有数据出现的问题 |
||||
5. 修复SwipeAction微信小程序端无法上下滑动问题 |
||||
6. 修复input的placeholder在小程序端默认显示为true问题 |
||||
7. 修复divider组件click事件无效问题 |
||||
8. 修复u-code-input maxlength 属性值为 String 类型时显示异常 |
||||
9. 修复当 grid只有 1到2时 在小程序端algin设置无效的问题 |
||||
10. 处理form-item的label为top时,取消错误提示的左边距 |
||||
11. 其他修复 |
||||
## 2.0.17(2021-12-26) |
||||
## uView正在参与开源中国的“年度最佳项目”评选,之前投过票的现在也可以投票,恳请同学们投一票,[点此帮助uView](https://www.oschina.net/project/top_cn_2021/?id=583) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 解决HBuilderX3.3.3.20211225版本导致的样式问题 |
||||
2. calendar日历添加monthNum参数 |
||||
3. navbar添加center slot |
||||
## 2.0.16(2021-12-25) |
||||
## uView正在参与开源中国的“年度最佳项目”评选,之前投过票的现在也可以投票,恳请同学们投一票,[点此帮助uView](https://www.oschina.net/project/top_cn_2021/?id=583) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 解决微信小程序setData性能问题 |
||||
2. 修复count-down组件change事件不触发问题 |
||||
## 2.0.15(2021-12-21) |
||||
## uView正在参与开源中国的“年度最佳项目”评选,之前投过票的现在也可以投票,恳请同学们投一票,[点此帮助uView](https://www.oschina.net/project/top_cn_2021/?id=583) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复Cell单元格titleWidth无效 |
||||
2. 修复cheakbox组件ischecked不更新 |
||||
3. 修复keyboard是否显示"."按键默认值问题 |
||||
4. 修复number-keyboard是否显示键盘的"."符号问题 |
||||
5. 修复Input输入框 readonly无效 |
||||
6. 修复u-avatar 导致打包app、H5时候报错问题 |
||||
7. 修复Upload上传deletable无效 |
||||
8. 修复upload当设置maxSize时无效的问题 |
||||
9. 修复tabs lineWidth传入带单位的字符串的时候偏移量计算错误问题 |
||||
10. 修复rate组件在有padding的view内,显示的星星位置和可触摸区域不匹配,无法正常选中星星 |
||||
## 2.0.13(2021-12-14) |
||||
## [点击加群交流反馈:364463526](https://jq.qq.com/?_chanwv=1027&k=mCxS3TGY) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复配置默认单位为rpx可能会导致自定义导航栏高度异常的问题 |
||||
## 2.0.12(2021-12-14) |
||||
## [点击加群交流反馈:364463526](https://jq.qq.com/?_chanwv=1027&k=mCxS3TGY) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复tabs组件在vue环境下划线消失的问题 |
||||
2. 修复upload组件在安卓小程序无法选择视频的问题 |
||||
3. 添加uni.$u.config.unit配置,用于配置参数默认单位,详见:[默认单位配置](https://www.uviewui.com/components/setting.html#%E9%BB%98%E8%AE%A4%E5%8D%95%E4%BD%8D%E9%85%8D%E7%BD%AE) |
||||
4. 修复textarea组件在没绑定v-model时,字符统计不生效问题 |
||||
5. 修复nvue下控制是否出现滚动条失效问题 |
||||
## 2.0.11(2021-12-13) |
||||
## [点击加群交流反馈:364463526](https://jq.qq.com/?_chanwv=1027&k=mCxS3TGY) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. text组件align参数无效的问题 |
||||
2. subsection组件添加keyName参数 |
||||
3. upload组件无法判断[Object file]类型的问题 |
||||
4. 处理notify层级过低问题 |
||||
5. codeInput组件添加disabledDot参数 |
||||
6. 处理actionSheet组件round参数无效的问题 |
||||
7. calendar组件添加round参数用于控制圆角值 |
||||
8. 处理swipeAction组件在vue环境下默认被打开的问题 |
||||
9. button组件的throttleTime节流参数无效的问题 |
||||
10. 解决u-notify手动关闭方法close()无效的问题 |
||||
11. input组件readonly不生效问题 |
||||
12. tag组件type参数为info不生效问题 |
||||
## 2.0.10(2021-12-08) |
||||
## [点击加群交流反馈:364463526](https://jq.qq.com/?_chanwv=1027&k=mCxS3TGY) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复button sendMessagePath属性不生效 |
||||
2. 修复DatetimePicker选择器title无效 |
||||
3. 修复u-toast设置loading=true不生效 |
||||
4. 修复u-text金额模式传0报错 |
||||
5. 修复u-toast组件的icon属性配置不生效 |
||||
6. button的icon在特殊场景下的颜色优化 |
||||
7. IndexList优化,增加# |
||||
## 2.0.9(2021-12-01) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 优化swiper的height支持100%值(仅vue有效),修复嵌入视频时click事件无法触发的问题 |
||||
2. 优化tabs组件对list值为空的判断,或者动态变化list时重新计算相关尺寸的问题 |
||||
3. 优化datetime-picker组件逻辑,让其后续打开的默认值为上一次的选中值,需要通过v-model绑定值才有效 |
||||
4. 修复upload内嵌在其他组件中,选择图片可能不会换行的问题 |
||||
## 2.0.8(2021-12-01) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复toast的position参数无效问题 |
||||
2. 处理input在ios nvue上无法获得焦点的问题 |
||||
3. avatar-group组件添加extraValue参数,让剩余展示数量可手动控制 |
||||
4. tabs组件添加keyName参数用于配置从对象中读取的键名 |
||||
5. 处理text组件名字脱敏默认配置无效的问题 |
||||
6. 处理picker组件item文本太长换行问题 |
||||
## 2.0.7(2021-11-30) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 修复radio和checkbox动态改变v-model无效的问题。 |
||||
2. 优化form规则validator在微信小程序用法 |
||||
3. 修复backtop组件mode参数在微信小程序无效的问题 |
||||
4. 处理Album的previewFullImage属性无效的问题 |
||||
5. 处理u-datetime-picker组件mode='time'在选择改变时间时,控制台报错的问题 |
||||
## 2.0.6(2021-11-27) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. 处理tag组件在vue下边框无效的问题。 |
||||
2. 处理popup组件圆角参数可能无效的问题。 |
||||
3. 处理tabs组件lineColor参数可能无效的问题。 |
||||
4. propgress组件在值很小时,显示异常的问题。 |
||||
## 2.0.5(2021-11-25) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. calendar在vue下显示异常问题。 |
||||
2. form组件labelPosition和errorType参数无效的问题 |
||||
3. input组件inputAlign无效的问题 |
||||
4. 其他一些修复 |
||||
## 2.0.4(2021-11-23) |
||||
## [点击加群交流反馈:232041042](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
0. input组件缺失@confirm事件,以及subfix和prefix无效问题 |
||||
1. component.scss文件样式在vue下干扰全局布局问题 |
||||
2. 修复subsection在vue环境下表现异常的问题 |
||||
3. tag组件的bgColor等参数无效的问题 |
||||
4. upload组件不换行的问题 |
||||
5. 其他的一些修复处理 |
||||
## 2.0.3(2021-11-16) |
||||
## [点击加群交流反馈:1129077272](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. uView2.0已实现全面兼容nvue |
||||
2. uView2.0对1.x进行了架构重构,细节和性能都有极大提升 |
||||
3. 目前uView2.0为公测阶段,相关细节可能会有变动 |
||||
4. 我们写了一份与1.x的对比指南,详见[对比1.x](https://www.uviewui.com/components/diff1.x.html) |
||||
5. 处理modal的confirm回调事件拼写错误问题 |
||||
6. 处理input组件@input事件参数错误问题 |
||||
7. 其他一些修复 |
||||
## 2.0.2(2021-11-16) |
||||
## [点击加群交流反馈:1129077272](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. uView2.0已实现全面兼容nvue |
||||
2. uView2.0对1.x进行了架构重构,细节和性能都有极大提升 |
||||
3. 目前uView2.0为公测阶段,相关细节可能会有变动 |
||||
4. 我们写了一份与1.x的对比指南,详见[对比1.x](https://www.uviewui.com/components/diff1.x.html) |
||||
5. 修复input组件formatter参数缺失问题 |
||||
6. 优化loading-icon组件的scss写法问题,防止不兼容新版本scss |
||||
## 2.0.0(2020-11-15) |
||||
## [点击加群交流反馈:1129077272](https://jq.qq.com/?_wv=1027&k=KnbeceDU) |
||||
|
||||
# uView2.0重磅发布,利剑出鞘,一统江湖 |
||||
|
||||
1. uView2.0已实现全面兼容nvue |
||||
2. uView2.0对1.x进行了架构重构,细节和性能都有极大提升 |
||||
3. 目前uView2.0为公测阶段,相关细节可能会有变动 |
||||
4. 我们写了一份与1.x的对比指南,详见[对比1.x](https://www.uviewui.com/components/diff1.x.html) |
||||
5. 修复input组件formatter参数缺失问题 |
||||
|
||||
|
@ -0,0 +1,78 @@ |
||||
<template> |
||||
<uvForm |
||||
ref="uForm" |
||||
:model="model" |
||||
:rules="rules" |
||||
:errorType="errorType" |
||||
:borderBottom="borderBottom" |
||||
:labelPosition="labelPosition" |
||||
:labelWidth="labelWidth" |
||||
:labelAlign="labelAlign" |
||||
:labelStyle="labelStyle" |
||||
:customStyle="customStyle" |
||||
> |
||||
<slot /> |
||||
</uvForm> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 此组件存在的理由是,在nvue下,u-form被uni-app官方占用了,u-form在nvue中相当于form组件 |
||||
* 所以在nvue下,取名为u--form,内部其实还是u-form.vue,只不过做一层中转 |
||||
*/ |
||||
import uvForm from '../u-form/u-form.vue'; |
||||
import props from '../u-form/props.js' |
||||
export default { |
||||
// #ifdef MP-WEIXIN |
||||
name: 'u-form', |
||||
// #endif |
||||
// #ifndef MP-WEIXIN |
||||
name: 'u--form', |
||||
// #endif |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
components: { |
||||
uvForm |
||||
}, |
||||
created() { |
||||
this.children = [] |
||||
}, |
||||
methods: { |
||||
// 手动设置校验的规则,如果规则中有函数的话,微信小程序中会过滤掉,所以只能手动调用设置规则 |
||||
setRules(rules) { |
||||
this.$refs.uForm.setRules(rules) |
||||
}, |
||||
validate() { |
||||
/** |
||||
* 在微信小程序中,通过this.$parent拿到的父组件是u--form,而不是其内嵌的u-form |
||||
* 导致在u-form组件中,拿不到对应的children数组,从而校验无效,所以这里每次调用u-form组件中的 |
||||
* 对应方法的时候,在小程序中都先将u--form的children赋值给u-form中的children |
||||
*/ |
||||
// #ifdef MP-WEIXIN |
||||
this.setMpData() |
||||
// #endif |
||||
return this.$refs.uForm.validate() |
||||
}, |
||||
validateField(value, callback) { |
||||
// #ifdef MP-WEIXIN |
||||
this.setMpData() |
||||
// #endif |
||||
return this.$refs.uForm.validateField(value, callback) |
||||
}, |
||||
resetFields() { |
||||
// #ifdef MP-WEIXIN |
||||
this.setMpData() |
||||
// #endif |
||||
return this.$refs.uForm.resetFields() |
||||
}, |
||||
clearValidate(props) { |
||||
// #ifdef MP-WEIXIN |
||||
this.setMpData() |
||||
// #endif |
||||
return this.$refs.uForm.clearValidate(props) |
||||
}, |
||||
setMpData() { |
||||
this.$refs.uForm.children = this.children |
||||
} |
||||
}, |
||||
} |
||||
</script> |
@ -0,0 +1,47 @@ |
||||
<template> |
||||
<uvImage |
||||
:src="src" |
||||
:mode="mode" |
||||
:width="width" |
||||
:height="height" |
||||
:shape="shape" |
||||
:radius="radius" |
||||
:lazyLoad="lazyLoad" |
||||
:showMenuByLongpress="showMenuByLongpress" |
||||
:loadingIcon="loadingIcon" |
||||
:errorIcon="errorIcon" |
||||
:showLoading="showLoading" |
||||
:showError="showError" |
||||
:fade="fade" |
||||
:webp="webp" |
||||
:duration="duration" |
||||
:bgColor="bgColor" |
||||
:customStyle="customStyle" |
||||
@click="$emit('click')" |
||||
@error="$emit('error')" |
||||
@load="$emit('load')" |
||||
> |
||||
<template v-slot:loading> |
||||
<slot name="loading"></slot> |
||||
</template> |
||||
<template v-slot:error> |
||||
<slot name="error"></slot> |
||||
</template> |
||||
</uvImage> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 此组件存在的理由是,在nvue下,u-image被uni-app官方占用了,u-image在nvue中相当于image组件 |
||||
* 所以在nvue下,取名为u--image,内部其实还是u-iamge.vue,只不过做一层中转 |
||||
*/ |
||||
import uvImage from '../u-image/u-image.vue'; |
||||
import props from '../u-image/props.js'; |
||||
export default { |
||||
name: 'u--image', |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
components: { |
||||
uvImage |
||||
}, |
||||
} |
||||
</script> |
@ -0,0 +1,72 @@ |
||||
<template> |
||||
<uvInput |
||||
:value="value" |
||||
:type="type" |
||||
:fixed="fixed" |
||||
:disabled="disabled" |
||||
:disabledColor="disabledColor" |
||||
:clearable="clearable" |
||||
:password="password" |
||||
:maxlength="maxlength" |
||||
:placeholder="placeholder" |
||||
:placeholderClass="placeholderClass" |
||||
:placeholderStyle="placeholderStyle" |
||||
:showWordLimit="showWordLimit" |
||||
:confirmType="confirmType" |
||||
:confirmHold="confirmHold" |
||||
:holdKeyboard="holdKeyboard" |
||||
:focus="focus" |
||||
:autoBlur="autoBlur" |
||||
:disableDefaultPadding="disableDefaultPadding" |
||||
:cursor="cursor" |
||||
:cursorSpacing="cursorSpacing" |
||||
:selectionStart="selectionStart" |
||||
:selectionEnd="selectionEnd" |
||||
:adjustPosition="adjustPosition" |
||||
:inputAlign="inputAlign" |
||||
:fontSize="fontSize" |
||||
:color="color" |
||||
:prefixIcon="prefixIcon" |
||||
:suffixIcon="suffixIcon" |
||||
:suffixIconStyle="suffixIconStyle" |
||||
:prefixIconStyle="prefixIconStyle" |
||||
:border="border" |
||||
:readonly="readonly" |
||||
:shape="shape" |
||||
:customStyle="customStyle" |
||||
:formatter="formatter" |
||||
@focus="$emit('focus')" |
||||
@blur="e => $emit('blur', e)" |
||||
@keyboardheightchange="$emit('keyboardheightchange')" |
||||
@change="e => $emit('change', e)" |
||||
@input="e => $emit('input', e)" |
||||
@confirm="e => $emit('confirm', e)" |
||||
@clear="$emit('clear')" |
||||
@click="$emit('click')" |
||||
> |
||||
<!-- #ifdef MP --> |
||||
<slot name="prefix"></slot> |
||||
<slot name="suffix"></slot> |
||||
<!-- #endif --> |
||||
<!-- #ifndef MP --> |
||||
<slot name="prefix" slot="prefix"></slot> |
||||
<slot name="suffix" slot="suffix"></slot> |
||||
<!-- #endif --> |
||||
</uvInput> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 此组件存在的理由是,在nvue下,u-input被uni-app官方占用了,u-input在nvue中相当于input组件 |
||||
* 所以在nvue下,取名为u--input,内部其实还是u-input.vue,只不过做一层中转 |
||||
*/ |
||||
import uvInput from '../u-input/u-input.vue'; |
||||
import props from '../u-input/props.js' |
||||
export default { |
||||
name: 'u--input', |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
components: { |
||||
uvInput |
||||
}, |
||||
} |
||||
</script> |
@ -0,0 +1,44 @@ |
||||
<template> |
||||
<uvText |
||||
:type="type" |
||||
:show="show" |
||||
:text="text" |
||||
:prefixIcon="prefixIcon" |
||||
:suffixIcon="suffixIcon" |
||||
:mode="mode" |
||||
:href="href" |
||||
:format="format" |
||||
:call="call" |
||||
:openType="openType" |
||||
:bold="bold" |
||||
:block="block" |
||||
:lines="lines" |
||||
:color="color" |
||||
:decoration="decoration" |
||||
:size="size" |
||||
:iconStyle="iconStyle" |
||||
:margin="margin" |
||||
:lineHeight="lineHeight" |
||||
:align="align" |
||||
:wordWrap="wordWrap" |
||||
:customStyle="customStyle" |
||||
@click="$emit('click')" |
||||
></uvText> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 此组件存在的理由是,在nvue下,u-text被uni-app官方占用了,u-text在nvue中相当于input组件 |
||||
* 所以在nvue下,取名为u--input,内部其实还是u-text.vue,只不过做一层中转 |
||||
* 不使用v-bind="$attrs",而是分开独立写传参,是因为微信小程序不支持此写法 |
||||
*/ |
||||
import uvText from "../u-text/u-text.vue"; |
||||
import props from "../u-text/props.js"; |
||||
export default { |
||||
name: "u--text", |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
components: { |
||||
uvText, |
||||
}, |
||||
}; |
||||
</script> |
@ -0,0 +1,47 @@ |
||||
<template> |
||||
<uvTextarea |
||||
:value="value" |
||||
:placeholder="placeholder" |
||||
:height="height" |
||||
:confirmType="confirmType" |
||||
:disabled="disabled" |
||||
:count="count" |
||||
:focus="focus" |
||||
:autoHeight="autoHeight" |
||||
:fixed="fixed" |
||||
:cursorSpacing="cursorSpacing" |
||||
:cursor="cursor" |
||||
:showConfirmBar="showConfirmBar" |
||||
:selectionStart="selectionStart" |
||||
:selectionEnd="selectionEnd" |
||||
:adjustPosition="adjustPosition" |
||||
:disableDefaultPadding="disableDefaultPadding" |
||||
:holdKeyboard="holdKeyboard" |
||||
:maxlength="maxlength" |
||||
:border="border" |
||||
:customStyle="customStyle" |
||||
:formatter="formatter" |
||||
@focus="e => $emit('focus')" |
||||
@blur="e => $emit('blur')" |
||||
@linechange="e => $emit('linechange', e)" |
||||
@confirm="e => $emit('confirm')" |
||||
@input="e => $emit('input', e)" |
||||
@keyboardheightchange="e => $emit('keyboardheightchange')" |
||||
></uvTextarea> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 此组件存在的理由是,在nvue下,u--textarea被uni-app官方占用了,u-textarea在nvue中相当于textarea组件 |
||||
* 所以在nvue下,取名为u--textarea,内部其实还是u-textarea.vue,只不过做一层中转 |
||||
*/ |
||||
import uvTextarea from '../u-textarea/u-textarea.vue'; |
||||
import props from '../u-textarea/props.js' |
||||
export default { |
||||
name: 'u--textarea', |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
components: { |
||||
uvTextarea |
||||
}, |
||||
} |
||||
</script> |
@ -0,0 +1,54 @@ |
||||
export default { |
||||
props: { |
||||
// 操作菜单是否展示 (默认false)
|
||||
show: { |
||||
type: Boolean, |
||||
default: uni.$u.props.actionSheet.show |
||||
}, |
||||
// 标题
|
||||
title: { |
||||
type: String, |
||||
default: uni.$u.props.actionSheet.title |
||||
}, |
||||
// 选项上方的描述信息
|
||||
description: { |
||||
type: String, |
||||
default: uni.$u.props.actionSheet.description |
||||
}, |
||||
// 数据
|
||||
actions: { |
||||
type: Array, |
||||
default: uni.$u.props.actionSheet.actions |
||||
}, |
||||
// 取消按钮的文字,不为空时显示按钮
|
||||
cancelText: { |
||||
type: String, |
||||
default: uni.$u.props.actionSheet.cancelText |
||||
}, |
||||
// 点击某个菜单项时是否关闭弹窗
|
||||
closeOnClickAction: { |
||||
type: Boolean, |
||||
default: uni.$u.props.actionSheet.closeOnClickAction |
||||
}, |
||||
// 处理底部安全区(默认true)
|
||||
safeAreaInsetBottom: { |
||||
type: Boolean, |
||||
default: uni.$u.props.actionSheet.safeAreaInsetBottom |
||||
}, |
||||
// 小程序的打开方式
|
||||
openType: { |
||||
type: String, |
||||
default: uni.$u.props.actionSheet.openType |
||||
}, |
||||
// 点击遮罩是否允许关闭 (默认true)
|
||||
closeOnClickOverlay: { |
||||
type: Boolean, |
||||
default: uni.$u.props.actionSheet.closeOnClickOverlay |
||||
}, |
||||
// 圆角值
|
||||
round: { |
||||
type: [Boolean, String, Number], |
||||
default: uni.$u.props.actionSheet.round |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,278 @@ |
||||
|
||||
<template> |
||||
<u-popup |
||||
:show="show" |
||||
mode="bottom" |
||||
@close="closeHandler" |
||||
:safeAreaInsetBottom="safeAreaInsetBottom" |
||||
:round="round" |
||||
> |
||||
<view class="u-action-sheet"> |
||||
<view |
||||
class="u-action-sheet__header" |
||||
v-if="title" |
||||
> |
||||
<text class="u-action-sheet__header__title u-line-1">{{title}}</text> |
||||
<view |
||||
class="u-action-sheet__header__icon-wrap" |
||||
@tap.stop="cancel" |
||||
> |
||||
<u-icon |
||||
name="close" |
||||
size="17" |
||||
color="#c8c9cc" |
||||
bold |
||||
></u-icon> |
||||
</view> |
||||
</view> |
||||
<text |
||||
class="u-action-sheet__description" |
||||
:style="[{ |
||||
marginTop: `${title && description ? 0 : '18px'}` |
||||
}]" |
||||
v-if="description" |
||||
>{{description}}</text> |
||||
<slot> |
||||
<u-line v-if="description"></u-line> |
||||
<view class="u-action-sheet__item-wrap"> |
||||
<template v-for="(item, index) in actions"> |
||||
<!-- #ifdef MP --> |
||||
<button |
||||
:key="index" |
||||
class="u-reset-button" |
||||
:openType="item.openType" |
||||
@getuserinfo="onGetUserInfo" |
||||
@contact="onContact" |
||||
@getphonenumber="onGetPhoneNumber" |
||||
@error="onError" |
||||
@launchapp="onLaunchApp" |
||||
@opensetting="onOpenSetting" |
||||
:lang="lang" |
||||
:session-from="sessionFrom" |
||||
:send-message-title="sendMessageTitle" |
||||
:send-message-path="sendMessagePath" |
||||
:send-message-img="sendMessageImg" |
||||
:show-message-card="showMessageCard" |
||||
:app-parameter="appParameter" |
||||
@tap="selectHandler(index)" |
||||
:hover-class="!item.disabled && !item.loading ? 'u-action-sheet--hover' : ''" |
||||
> |
||||
<!-- #endif --> |
||||
<view |
||||
class="u-action-sheet__item-wrap__item" |
||||
@tap.stop="selectHandler(index)" |
||||
:hover-class="!item.disabled && !item.loading ? 'u-action-sheet--hover' : ''" |
||||
:hover-stay-time="150" |
||||
> |
||||
<template v-if="!item.loading"> |
||||
<text |
||||
class="u-action-sheet__item-wrap__item__name" |
||||
:style="[itemStyle(index)]" |
||||
>{{ item.name }}</text> |
||||
<text |
||||
v-if="item.subname" |
||||
class="u-action-sheet__item-wrap__item__subname" |
||||
>{{ item.subname }}</text> |
||||
</template> |
||||
<u-loading-icon |
||||
v-else |
||||
custom-class="van-action-sheet__loading" |
||||
size="18" |
||||
mode="circle" |
||||
/> |
||||
</view> |
||||
<!-- #ifdef MP --> |
||||
</button> |
||||
<!-- #endif --> |
||||
<u-line v-if="index !== actions.length - 1"></u-line> |
||||
</template> |
||||
</view> |
||||
</slot> |
||||
<u-gap |
||||
bgColor="#eaeaec" |
||||
height="6" |
||||
v-if="cancelText" |
||||
></u-gap> |
||||
<view hover-class="u-action-sheet--hover"> |
||||
<text |
||||
@touchmove.stop.prevent |
||||
:hover-stay-time="150" |
||||
v-if="cancelText" |
||||
class="u-action-sheet__cancel-text" |
||||
@tap="cancel" |
||||
>{{cancelText}}</text> |
||||
</view> |
||||
</view> |
||||
</u-popup> |
||||
</template> |
||||
|
||||
<script> |
||||
import openType from '../../libs/mixin/openType' |
||||
import button from '../../libs/mixin/button' |
||||
import props from './props.js'; |
||||
/** |
||||
* ActionSheet 操作菜单 |
||||
* @description 本组件用于从底部弹出一个操作菜单,供用户选择并返回结果。本组件功能类似于uni的uni.showActionSheetAPI,配置更加灵活,所有平台都表现一致。 |
||||
* @tutorial https://www.uviewui.com/components/actionSheet.html |
||||
* |
||||
* @property {Boolean} show 操作菜单是否展示 (默认 false ) |
||||
* @property {String} title 操作菜单标题 |
||||
* @property {String} description 选项上方的描述信息 |
||||
* @property {Array<Object>} actions 按钮的文字数组,见官方文档示例 |
||||
* @property {String} cancelText 取消按钮的提示文字,不为空时显示按钮 |
||||
* @property {Boolean} closeOnClickAction 点击某个菜单项时是否关闭弹窗 (默认 true ) |
||||
* @property {Boolean} safeAreaInsetBottom 处理底部安全区 (默认 true ) |
||||
* @property {String} openType 小程序的打开方式 (contact | launchApp | getUserInfo | openSetting |getPhoneNumber |error ) |
||||
* @property {Boolean} closeOnClickOverlay 点击遮罩是否允许关闭 (默认 true ) |
||||
* @property {Number|String} round 圆角值,默认无圆角 (默认 0 ) |
||||
* @property {String} lang 指定返回用户信息的语言,zh_CN 简体中文,zh_TW 繁体中文,en 英文 |
||||
* @property {String} sessionFrom 会话来源,openType="contact"时有效 |
||||
* @property {String} sendMessageTitle 会话内消息卡片标题,openType="contact"时有效 |
||||
* @property {String} sendMessagePath 会话内消息卡片点击跳转小程序路径,openType="contact"时有效 |
||||
* @property {String} sendMessageImg 会话内消息卡片图片,openType="contact"时有效 |
||||
* @property {Boolean} showMessageCard 是否显示会话内消息卡片,设置此参数为 true,用户进入客服会话会在右下角显示"可能要发送的小程序"提示,用户点击后可以快速发送小程序消息,openType="contact"时有效 (默认 false ) |
||||
* @property {String} appParameter 打开 APP 时,向 APP 传递的参数,openType=launchApp 时有效 |
||||
* |
||||
* @event {Function} select 点击ActionSheet列表项时触发 |
||||
* @event {Function} close 点击取消按钮时触发 |
||||
* @event {Function} getuserinfo 用户点击该按钮时,会返回获取到的用户信息,回调的 detail 数据与 wx.getUserInfo 返回的一致,openType="getUserInfo"时有效 |
||||
* @event {Function} contact 客服消息回调,openType="contact"时有效 |
||||
* @event {Function} getphonenumber 获取用户手机号回调,openType="getPhoneNumber"时有效 |
||||
* @event {Function} error 当使用开放能力时,发生错误的回调,openType="error"时有效 |
||||
* @event {Function} launchapp 打开 APP 成功的回调,openType="launchApp"时有效 |
||||
* @event {Function} opensetting 在打开授权设置页后回调,openType="openSetting"时有效 |
||||
* @example <u-action-sheet :actions="list" :title="title" :show="show"></u-action-sheet> |
||||
*/ |
||||
export default { |
||||
name: "u-action-sheet", |
||||
// 一些props参数和methods方法,通过mixin混入,因为其他文件也会用到 |
||||
mixins: [openType, button, uni.$u.mixin, props], |
||||
data() { |
||||
return { |
||||
|
||||
} |
||||
}, |
||||
computed: { |
||||
// 操作项目的样式 |
||||
itemStyle() { |
||||
return (index) => { |
||||
let style = {}; |
||||
if (this.actions[index].color) style.color = this.actions[index].color |
||||
if (this.actions[index].fontSize) style.fontSize = uni.$u.addUnit(this.actions[index].fontSize) |
||||
// 选项被禁用的样式 |
||||
if (this.actions[index].disabled) style.color = '#c0c4cc' |
||||
return style; |
||||
} |
||||
}, |
||||
}, |
||||
methods: { |
||||
closeHandler() { |
||||
// 允许点击遮罩关闭时,才发出close事件 |
||||
if(this.closeOnClickOverlay) { |
||||
this.$emit('close') |
||||
} |
||||
}, |
||||
// 点击取消按钮 |
||||
cancel() { |
||||
this.$emit('close') |
||||
}, |
||||
selectHandler(index) { |
||||
const item = this.actions[index] |
||||
if (item && !item.disabled && !item.loading) { |
||||
this.$emit('select', item) |
||||
if (this.closeOnClickAction) { |
||||
this.$emit('close') |
||||
} |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
$u-action-sheet-reset-button-width:100% !default; |
||||
$u-action-sheet-title-font-size: 16px !default; |
||||
$u-action-sheet-title-padding: 12px 30px !default; |
||||
$u-action-sheet-title-color: $u-main-color !default; |
||||
$u-action-sheet-header-icon-wrap-right:15px !default; |
||||
$u-action-sheet-header-icon-wrap-top:15px !default; |
||||
$u-action-sheet-description-font-size:13px !default; |
||||
$u-action-sheet-description-color:14px !default; |
||||
$u-action-sheet-description-margin: 18px 15px !default; |
||||
$u-action-sheet-item-wrap-item-padding:15px !default; |
||||
$u-action-sheet-item-wrap-name-font-size:16px !default; |
||||
$u-action-sheet-item-wrap-subname-font-size:13px !default; |
||||
$u-action-sheet-item-wrap-subname-color: #c0c4cc !default; |
||||
$u-action-sheet-item-wrap-subname-margin-top:10px !default; |
||||
$u-action-sheet-cancel-text-font-size:16px !default; |
||||
$u-action-sheet-cancel-text-color:$u-content-color !default; |
||||
$u-action-sheet-cancel-text-font-size:15px !default; |
||||
$u-action-sheet-cancel-text-hover-background-color:rgb(242, 243, 245) !default; |
||||
|
||||
.u-reset-button { |
||||
width: $u-action-sheet-reset-button-width; |
||||
} |
||||
|
||||
.u-action-sheet { |
||||
text-align: center; |
||||
&__header { |
||||
position: relative; |
||||
padding: $u-action-sheet-title-padding; |
||||
&__title { |
||||
font-size: $u-action-sheet-title-font-size; |
||||
color: $u-action-sheet-title-color; |
||||
font-weight: bold; |
||||
text-align: center; |
||||
} |
||||
|
||||
&__icon-wrap { |
||||
position: absolute; |
||||
right: $u-action-sheet-header-icon-wrap-right; |
||||
top: $u-action-sheet-header-icon-wrap-top; |
||||
} |
||||
} |
||||
|
||||
&__description { |
||||
font-size: $u-action-sheet-description-font-size; |
||||
color: $u-tips-color; |
||||
margin: $u-action-sheet-description-margin; |
||||
text-align: center; |
||||
} |
||||
|
||||
&__item-wrap { |
||||
|
||||
&__item { |
||||
padding: $u-action-sheet-item-wrap-item-padding; |
||||
@include flex; |
||||
align-items: center; |
||||
justify-content: center; |
||||
flex-direction: column; |
||||
|
||||
&__name { |
||||
font-size: $u-action-sheet-item-wrap-name-font-size; |
||||
color: $u-main-color; |
||||
text-align: center; |
||||
} |
||||
|
||||
&__subname { |
||||
font-size: $u-action-sheet-item-wrap-subname-font-size; |
||||
color: $u-action-sheet-item-wrap-subname-color; |
||||
margin-top: $u-action-sheet-item-wrap-subname-margin-top; |
||||
text-align: center; |
||||
} |
||||
} |
||||
} |
||||
|
||||
&__cancel-text { |
||||
font-size: $u-action-sheet-cancel-text-font-size; |
||||
color: $u-action-sheet-cancel-text-color; |
||||
text-align: center; |
||||
padding: $u-action-sheet-cancel-text-font-size; |
||||
} |
||||
|
||||
&--hover { |
||||
background-color: $u-action-sheet-cancel-text-hover-background-color; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,59 @@ |
||||
export default { |
||||
props: { |
||||
// 图片地址,Array<String>|Array<Object>形式
|
||||
urls: { |
||||
type: Array, |
||||
default: uni.$u.props.album.urls |
||||
}, |
||||
// 指定从数组的对象元素中读取哪个属性作为图片地址
|
||||
keyName: { |
||||
type: String, |
||||
default: uni.$u.props.album.keyName |
||||
}, |
||||
// 单图时,图片长边的长度
|
||||
singleSize: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.album.singleSize |
||||
}, |
||||
// 多图时,图片边长
|
||||
multipleSize: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.album.multipleSize |
||||
}, |
||||
// 多图时,图片水平和垂直之间的间隔
|
||||
space: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.album.space |
||||
}, |
||||
// 单图时,图片缩放裁剪的模式
|
||||
singleMode: { |
||||
type: String, |
||||
default: uni.$u.props.album.singleMode |
||||
}, |
||||
// 多图时,图片缩放裁剪的模式
|
||||
multipleMode: { |
||||
type: String, |
||||
default: uni.$u.props.album.multipleMode |
||||
}, |
||||
// 最多展示的图片数量,超出时最后一个位置将会显示剩余图片数量
|
||||
maxCount: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.album.maxCount |
||||
}, |
||||
// 是否可以预览图片
|
||||
previewFullImage: { |
||||
type: Boolean, |
||||
default: uni.$u.props.album.previewFullImage |
||||
}, |
||||
// 每行展示图片数量,如设置,singleSize和multipleSize将会无效
|
||||
rowCount: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.album.rowCount |
||||
}, |
||||
// 超出maxCount时是否显示查看更多的提示
|
||||
showMore: { |
||||
type: Boolean, |
||||
default: uni.$u.props.album.showMore |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,259 @@ |
||||
<template> |
||||
<view class="u-album"> |
||||
<view |
||||
class="u-album__row" |
||||
ref="u-album__row" |
||||
v-for="(arr, index) in showUrls" |
||||
:forComputedUse="albumWidth" |
||||
:key="index" |
||||
> |
||||
<view |
||||
class="u-album__row__wrapper" |
||||
v-for="(item, index1) in arr" |
||||
:key="index1" |
||||
:style="[imageStyle(index + 1, index1 + 1)]" |
||||
@tap="previewFullImage ? onPreviewTap(getSrc(item)) : ''" |
||||
> |
||||
<image |
||||
:src="getSrc(item)" |
||||
:mode=" |
||||
urls.length === 1 |
||||
? imageHeight > 0 |
||||
? singleMode |
||||
: 'widthFix' |
||||
: multipleMode |
||||
" |
||||
:style="[ |
||||
{ |
||||
width: imageWidth, |
||||
height: imageHeight |
||||
} |
||||
]" |
||||
></image> |
||||
<view |
||||
v-if=" |
||||
showMore && |
||||
urls.length > rowCount * showUrls.length && |
||||
index === showUrls.length - 1 && |
||||
index1 === showUrls[showUrls.length - 1].length - 1 |
||||
" |
||||
class="u-album__row__wrapper__text" |
||||
> |
||||
<u--text |
||||
:text="`+${urls.length - maxCount}`" |
||||
color="#fff" |
||||
:size="multipleSize * 0.3" |
||||
align="center" |
||||
customStyle="justify-content: center" |
||||
></u--text> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js' |
||||
// #ifdef APP-NVUE |
||||
// 由于weex为阿里的KPI业绩考核的产物,所以不支持百分比单位,这里需要通过dom查询组件的宽度 |
||||
const dom = uni.requireNativePlugin('dom') |
||||
// #endif |
||||
|
||||
/** |
||||
* Album 相册 |
||||
* @description 本组件提供一个类似相册的功能,让开发者开发起来更加得心应手。减少重复的模板代码 |
||||
* @tutorial https://www.uviewui.com/components/album.html |
||||
* |
||||
* @property {Array} urls 图片地址列表 Array<String>|Array<Object>形式 |
||||
* @property {String} keyName 指定从数组的对象元素中读取哪个属性作为图片地址 |
||||
* @property {String | Number} singleSize 单图时,图片长边的长度 (默认 180 ) |
||||
* @property {String | Number} multipleSize 多图时,图片边长 (默认 70 ) |
||||
* @property {String | Number} space 多图时,图片水平和垂直之间的间隔 (默认 6 ) |
||||
* @property {String} singleMode 单图时,图片缩放裁剪的模式 (默认 'scaleToFill' ) |
||||
* @property {String} multipleMode 多图时,图片缩放裁剪的模式 (默认 'aspectFill' ) |
||||
* @property {String | Number} maxCount 取消按钮的提示文字 (默认 9 ) |
||||
* @property {Boolean} previewFullImage 是否可以预览图片 (默认 true ) |
||||
* @property {String | Number} rowCount 每行展示图片数量,如设置,singleSize和multipleSize将会无效 (默认 3 ) |
||||
* @property {Boolean} showMore 超出maxCount时是否显示查看更多的提示 (默认 true ) |
||||
* |
||||
* @event {Function} albumWidth 某些特殊的情况下,需要让文字与相册的宽度相等,这里事件的形式对外发送 (回调参数 width ) |
||||
* @example <u-album :urls="urls2" @albumWidth="width => albumWidth = width" multipleSize="68" ></u-album> |
||||
*/ |
||||
export default { |
||||
name: 'u-album', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
data() { |
||||
return { |
||||
// 单图的宽度 |
||||
singleWidth: 0, |
||||
// 单图的高度 |
||||
singleHeight: 0, |
||||
// 单图时,如果无法获取图片的尺寸信息,让图片宽度默认为容器的一定百分比 |
||||
singlePercent: 0.6 |
||||
} |
||||
}, |
||||
watch: { |
||||
urls: { |
||||
immediate: true, |
||||
handler(newVal) { |
||||
if (newVal.length === 1) { |
||||
this.getImageRect() |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
imageStyle() { |
||||
return (index1, index2) => { |
||||
const { space, rowCount, multipleSize, urls } = this, |
||||
{ addUnit, addStyle } = uni.$u, |
||||
rowLen = this.showUrls.length, |
||||
allLen = this.urls.length |
||||
const style = { |
||||
marginRight: addUnit(space), |
||||
marginBottom: addUnit(space) |
||||
} |
||||
// 如果为最后一行,则每个图片都无需下边框 |
||||
if (index1 === rowLen) style.marginBottom = 0 |
||||
// 每行的最右边一张和总长度的最后一张无需右边框 |
||||
if ( |
||||
index2 === rowCount || |
||||
(index1 === rowLen && |
||||
index2 === this.showUrls[index1 - 1].length) |
||||
) |
||||
style.marginRight = 0 |
||||
return style |
||||
} |
||||
}, |
||||
// 将数组划分为二维数组 |
||||
showUrls() { |
||||
const arr = [] |
||||
this.urls.map((item, index) => { |
||||
// 限制最大展示数量 |
||||
if (index + 1 <= this.maxCount) { |
||||
// 计算该元素为第几个素组内 |
||||
const itemIndex = Math.floor(index / this.rowCount) |
||||
// 判断对应的索引是否存在 |
||||
if (!arr[itemIndex]) { |
||||
arr[itemIndex] = [] |
||||
} |
||||
arr[itemIndex].push(item) |
||||
} |
||||
}) |
||||
return arr |
||||
}, |
||||
imageWidth() { |
||||
return uni.$u.addUnit( |
||||
this.urls.length === 1 ? this.singleWidth : this.multipleSize |
||||
) |
||||
}, |
||||
imageHeight() { |
||||
return uni.$u.addUnit( |
||||
this.urls.length === 1 ? this.singleHeight : this.multipleSize |
||||
) |
||||
}, |
||||
// 此变量无实际用途,仅仅是为了利用computed特性,让其在urls长度等变化时,重新计算图片的宽度 |
||||
// 因为用户在某些特殊的情况下,需要让文字与相册的宽度相等,所以这里事件的形式对外发送 |
||||
albumWidth() { |
||||
let width = 0 |
||||
if (this.urls.length === 1) { |
||||
width = this.singleWidth |
||||
} else { |
||||
width = |
||||
this.showUrls[0].length * this.multipleSize + |
||||
this.space * (this.showUrls[0].length - 1) |
||||
} |
||||
this.$emit('albumWidth', width) |
||||
return width |
||||
} |
||||
}, |
||||
methods: { |
||||
// 预览图片 |
||||
onPreviewTap(url) { |
||||
const urls = this.urls.map((item) => { |
||||
return this.getSrc(item) |
||||
}) |
||||
uni.previewImage({ |
||||
current: url, |
||||
urls |
||||
}) |
||||
}, |
||||
// 获取图片的路径 |
||||
getSrc(item) { |
||||
return uni.$u.test.object(item) |
||||
? (this.keyName && item[this.keyName]) || item.src |
||||
: item |
||||
}, |
||||
// 单图时,获取图片的尺寸 |
||||
// 在小程序中,需要将网络图片的的域名添加到小程序的download域名才可能获取尺寸 |
||||
// 在没有添加的情况下,让单图宽度默认为盒子的一定宽度(singlePercent) |
||||
getImageRect() { |
||||
const src = this.getSrc(this.urls[0]) |
||||
uni.getImageInfo({ |
||||
src, |
||||
success: (res) => { |
||||
// 判断图片横向还是竖向展示方式 |
||||
const isHorizotal = res.width >= res.height |
||||
this.singleWidth = isHorizotal |
||||
? this.singleSize |
||||
: (res.width / res.height) * this.singleSize |
||||
this.singleHeight = !isHorizotal |
||||
? this.singleSize |
||||
: (res.height / res.width) * this.singleWidth |
||||
}, |
||||
fail: () => { |
||||
this.getComponentWidth() |
||||
} |
||||
}) |
||||
}, |
||||
// 获取组件的宽度 |
||||
async getComponentWidth() { |
||||
// 延时一定时间,以获取dom尺寸 |
||||
await uni.$u.sleep(30) |
||||
// #ifndef APP-NVUE |
||||
this.$uGetRect('.u-album__row').then((size) => { |
||||
this.singleWidth = size.width * this.singlePercent |
||||
}) |
||||
// #endif |
||||
|
||||
// #ifdef APP-NVUE |
||||
// 这里ref="u-album__row"所在的标签为通过for循环出来,导致this.$refs['u-album__row']是一个数组 |
||||
const ref = this.$refs['u-album__row'][0] |
||||
ref && |
||||
dom.getComponentRect(ref, (res) => { |
||||
this.singleWidth = res.size.width * this.singlePercent |
||||
}) |
||||
// #endif |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import '../../libs/css/components.scss'; |
||||
|
||||
.u-album { |
||||
@include flex(column); |
||||
|
||||
&__row { |
||||
@include flex(row); |
||||
flex-wrap: wrap; |
||||
|
||||
&__wrapper { |
||||
position: relative; |
||||
|
||||
&__text { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
background-color: rgba(0, 0, 0, 0.3); |
||||
@include flex(row); |
||||
justify-content: center; |
||||
align-items: center; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,44 @@ |
||||
export default { |
||||
props: { |
||||
// 显示文字
|
||||
title: { |
||||
type: String, |
||||
default: uni.$u.props.alert.title |
||||
}, |
||||
// 主题,success/warning/info/error
|
||||
type: { |
||||
type: String, |
||||
default: uni.$u.props.alert.type |
||||
}, |
||||
// 辅助性文字
|
||||
description: { |
||||
type: String, |
||||
default: uni.$u.props.alert.description |
||||
}, |
||||
// 是否可关闭
|
||||
closable: { |
||||
type: Boolean, |
||||
default: uni.$u.props.alert.closable |
||||
}, |
||||
// 是否显示图标
|
||||
showIcon: { |
||||
type: Boolean, |
||||
default: uni.$u.props.alert.showIcon |
||||
}, |
||||
// 浅或深色调,light-浅色,dark-深色
|
||||
effect: { |
||||
type: String, |
||||
default: uni.$u.props.alert.effect |
||||
}, |
||||
// 文字是否居中
|
||||
center: { |
||||
type: Boolean, |
||||
default: uni.$u.props.alert.center |
||||
}, |
||||
// 字体大小
|
||||
fontSize: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.alert.fontSize |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,243 @@ |
||||
<template> |
||||
<u-transition |
||||
mode="fade" |
||||
:show="show" |
||||
> |
||||
<view |
||||
class="u-alert" |
||||
:class="[`u-alert--${type}--${effect}`]" |
||||
@tap.stop="clickHandler" |
||||
:style="[$u.addStyle(customStyle)]" |
||||
> |
||||
<view |
||||
class="u-alert__icon" |
||||
v-if="showIcon" |
||||
> |
||||
<u-icon |
||||
:name="iconName" |
||||
size="18" |
||||
:color="iconColor" |
||||
></u-icon> |
||||
</view> |
||||
<view |
||||
class="u-alert__content" |
||||
:style="[{ |
||||
paddingRight: closable ? '20px' : 0 |
||||
}]" |
||||
> |
||||
<text |
||||
class="u-alert__content__title" |
||||
v-if="title" |
||||
:style="[{ |
||||
fontSize: $u.addUnit(fontSize), |
||||
textAlign: center ? 'center' : 'left' |
||||
}]" |
||||
:class="[effect === 'dark' ? 'u-alert__text--dark' : `u-alert__text--${type}--light`]" |
||||
>{{ title }}</text> |
||||
<text |
||||
class="u-alert__content__desc" |
||||
v-if="description" |
||||
:style="[{ |
||||
fontSize: $u.addUnit(fontSize), |
||||
textAlign: center ? 'center' : 'left' |
||||
}]" |
||||
:class="[effect === 'dark' ? 'u-alert__text--dark' : `u-alert__text--${type}--light`]" |
||||
>{{ description }}</text> |
||||
</view> |
||||
<view |
||||
class="u-alert__close" |
||||
v-if="closable" |
||||
@tap.stop="closeHandler" |
||||
> |
||||
<u-icon |
||||
name="close" |
||||
:color="iconColor" |
||||
size="15" |
||||
></u-icon> |
||||
</view> |
||||
</view> |
||||
</u-transition> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js'; |
||||
/** |
||||
* Alert 警告提示 |
||||
* @description 警告提示,展现需要关注的信息。 |
||||
* @tutorial https://www.uviewui.com/components/alertTips.html |
||||
* |
||||
* @property {String} title 显示的文字 |
||||
* @property {String} type 使用预设的颜色 (默认 'warning' ) |
||||
* @property {String} description 辅助性文字,颜色比title浅一点,字号也小一点,可选 |
||||
* @property {Boolean} closable 关闭按钮(默认为叉号icon图标) (默认 false ) |
||||
* @property {Boolean} showIcon 是否显示左边的辅助图标 ( 默认 false ) |
||||
* @property {String} effect 多图时,图片缩放裁剪的模式 (默认 'light' ) |
||||
* @property {Boolean} center 文字是否居中 (默认 false ) |
||||
* @property {String | Number} fontSize 字体大小 (默认 14 ) |
||||
* @property {Object} customStyle 定义需要用到的外部样式 |
||||
* @event {Function} click 点击组件时触发 |
||||
* @example <u-alert :title="title" type = "warning" :closable="closable" :description = "description"></u-alert> |
||||
*/ |
||||
export default { |
||||
name: 'u-alert', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
data() { |
||||
return { |
||||
show: true |
||||
} |
||||
}, |
||||
computed: { |
||||
iconColor() { |
||||
return this.effect === 'light' ? this.type : '#fff' |
||||
}, |
||||
// 不同主题对应不同的图标 |
||||
iconName() { |
||||
switch (this.type) { |
||||
case 'success': |
||||
return 'checkmark-circle-fill'; |
||||
break; |
||||
case 'error': |
||||
return 'close-circle-fill'; |
||||
break; |
||||
case 'warning': |
||||
return 'error-circle-fill'; |
||||
break; |
||||
case 'info': |
||||
return 'info-circle-fill'; |
||||
break; |
||||
case 'primary': |
||||
return 'more-circle-fill'; |
||||
break; |
||||
default: |
||||
return 'error-circle-fill'; |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
// 点击内容 |
||||
clickHandler() { |
||||
this.$emit('click') |
||||
}, |
||||
// 点击关闭按钮 |
||||
closeHandler() { |
||||
this.show = false |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
.u-alert { |
||||
position: relative; |
||||
background-color: $u-primary; |
||||
padding: 8px 10px; |
||||
@include flex(row); |
||||
align-items: center; |
||||
border-top-left-radius: 4px; |
||||
border-top-right-radius: 4px; |
||||
border-bottom-left-radius: 4px; |
||||
border-bottom-right-radius: 4px; |
||||
|
||||
&--primary--dark { |
||||
background-color: $u-primary; |
||||
} |
||||
|
||||
&--primary--light { |
||||
background-color: #ecf5ff; |
||||
} |
||||
|
||||
&--error--dark { |
||||
background-color: $u-error; |
||||
} |
||||
|
||||
&--error--light { |
||||
background-color: #FEF0F0; |
||||
} |
||||
|
||||
&--success--dark { |
||||
background-color: $u-success; |
||||
} |
||||
|
||||
&--success--light { |
||||
background-color: #f5fff0; |
||||
} |
||||
|
||||
&--warning--dark { |
||||
background-color: $u-warning; |
||||
} |
||||
|
||||
&--warning--light { |
||||
background-color: #FDF6EC; |
||||
} |
||||
|
||||
&--info--dark { |
||||
background-color: $u-info; |
||||
} |
||||
|
||||
&--info--light { |
||||
background-color: #f4f4f5; |
||||
} |
||||
|
||||
&__icon { |
||||
margin-right: 5px; |
||||
} |
||||
|
||||
&__content { |
||||
@include flex(column); |
||||
flex: 1; |
||||
|
||||
&__title { |
||||
color: $u-main-color; |
||||
font-size: 14px; |
||||
font-weight: bold; |
||||
color: #fff; |
||||
margin-bottom: 2px; |
||||
} |
||||
|
||||
&__desc { |
||||
color: $u-main-color; |
||||
font-size: 14px; |
||||
flex-wrap: wrap; |
||||
color: #fff; |
||||
} |
||||
} |
||||
|
||||
&__title--dark, |
||||
&__desc--dark { |
||||
color: #FFFFFF; |
||||
} |
||||
|
||||
&__text--primary--light, |
||||
&__text--primary--light { |
||||
color: $u-primary; |
||||
} |
||||
|
||||
&__text--success--light, |
||||
&__text--success--light { |
||||
color: $u-success; |
||||
} |
||||
|
||||
&__text--warning--light, |
||||
&__text--warning--light { |
||||
color: $u-warning; |
||||
} |
||||
|
||||
&__text--error--light, |
||||
&__text--error--light { |
||||
color: $u-error; |
||||
} |
||||
|
||||
&__text--info--light, |
||||
&__text--info--light { |
||||
color: $u-info; |
||||
} |
||||
|
||||
&__close { |
||||
position: absolute; |
||||
top: 11px; |
||||
right: 10px; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,52 @@ |
||||
export default { |
||||
props: { |
||||
// 头像图片组
|
||||
urls: { |
||||
type: Array, |
||||
default: uni.$u.props.avatarGroup.urls |
||||
}, |
||||
// 最多展示的头像数量
|
||||
maxCount: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.avatarGroup.maxCount |
||||
}, |
||||
// 头像形状
|
||||
shape: { |
||||
type: String, |
||||
default: uni.$u.props.avatarGroup.shape |
||||
}, |
||||
// 图片裁剪模式
|
||||
mode: { |
||||
type: String, |
||||
default: uni.$u.props.avatarGroup.mode |
||||
}, |
||||
// 超出maxCount时是否显示查看更多的提示
|
||||
showMore: { |
||||
type: Boolean, |
||||
default: uni.$u.props.avatarGroup.showMore |
||||
}, |
||||
// 头像大小
|
||||
size: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.avatarGroup.size |
||||
}, |
||||
// 指定从数组的对象元素中读取哪个属性作为图片地址
|
||||
keyName: { |
||||
type: String, |
||||
default: uni.$u.props.avatarGroup.keyName |
||||
}, |
||||
// 头像之间的遮挡比例
|
||||
gap: { |
||||
type: [String, Number], |
||||
validator(value) { |
||||
return value >= 0 && value <= 1 |
||||
}, |
||||
default: uni.$u.props.avatarGroup.gap |
||||
}, |
||||
// 需额外显示的值
|
||||
extraValue: { |
||||
type: [Number, String], |
||||
default: uni.$u.props.avatarGroup.extraValue |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,103 @@ |
||||
<template> |
||||
<view class="u-avatar-group"> |
||||
<view |
||||
class="u-avatar-group__item" |
||||
v-for="(item, index) in showUrl" |
||||
:key="index" |
||||
:style="{ |
||||
marginLeft: index === 0 ? 0 : $u.addUnit(-size * gap) |
||||
}" |
||||
> |
||||
<u-avatar |
||||
:size="size" |
||||
:shape="shape" |
||||
:mode="mode" |
||||
:src="$u.test.object(item) ? keyName && item[keyName] || item.url : item" |
||||
></u-avatar> |
||||
<view |
||||
class="u-avatar-group__item__show-more" |
||||
v-if="showMore && index === showUrl.length - 1 && (urls.length > maxCount || extraValue > 0)" |
||||
@tap="clickHandler" |
||||
> |
||||
<u--text |
||||
color="#ffffff" |
||||
:size="size * 0.4" |
||||
:text="`+${extraValue || urls.length - showUrl.length}`" |
||||
align="center" |
||||
customStyle="justify-content: center" |
||||
></u--text> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js'; |
||||
/** |
||||
* AvatarGroup 头像组 |
||||
* @description 本组件一般用于展示头像的地方,如个人中心,或者评论列表页的用户头像展示等场所。 |
||||
* @tutorial https://www.uviewui.com/components/avatar.html |
||||
* |
||||
* @property {Array} urls 头像图片组 (默认 [] ) |
||||
* @property {String | Number} maxCount 最多展示的头像数量 ( 默认 5 ) |
||||
* @property {String} shape 头像形状( 'circle' (默认) | 'square' ) |
||||
* @property {String} mode 图片裁剪模式(默认 'scaleToFill' ) |
||||
* @property {Boolean} showMore 超出maxCount时是否显示查看更多的提示 (默认 true ) |
||||
* @property {String | Number} size 头像大小 (默认 40 ) |
||||
* @property {String} keyName 指定从数组的对象元素中读取哪个属性作为图片地址 |
||||
* @property {String | Number} gap 头像之间的遮挡比例(0.4代表遮挡40%) (默认 0.5 ) |
||||
* @property {String | Number} extraValue 需额外显示的值 |
||||
* @event {Function} showMore 头像组更多点击 |
||||
* @example <u-avatar-group:urls="urls" size="35" gap="0.4" ></u-avatar-group:urls=> |
||||
*/ |
||||
export default { |
||||
name: 'u-avatar-group', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
data() { |
||||
return { |
||||
|
||||
} |
||||
}, |
||||
computed: { |
||||
showUrl() { |
||||
return this.urls.slice(0, this.maxCount) |
||||
} |
||||
}, |
||||
methods: { |
||||
clickHandler() { |
||||
this.$emit('showMore') |
||||
} |
||||
}, |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
.u-avatar-group { |
||||
@include flex; |
||||
|
||||
&__item { |
||||
margin-left: -10px; |
||||
position: relative; |
||||
|
||||
&--no-indent { |
||||
// 如果你想质疑作者不会使用:first-child,说明你太年轻,因为nvue不支持 |
||||
margin-left: 0; |
||||
} |
||||
|
||||
&__show-more { |
||||
position: absolute; |
||||
top: 0; |
||||
bottom: 0; |
||||
left: 0; |
||||
right: 0; |
||||
background-color: rgba(0, 0, 0, 0.3); |
||||
@include flex; |
||||
align-items: center; |
||||
justify-content: center; |
||||
border-radius: 100px; |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,78 @@ |
||||
export default { |
||||
props: { |
||||
// 头像图片路径(不能为相对路径)
|
||||
src: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.src |
||||
}, |
||||
// 头像形状,circle-圆形,square-方形
|
||||
shape: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.shape |
||||
}, |
||||
// 头像尺寸
|
||||
size: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.avatar.size |
||||
}, |
||||
// 裁剪模式
|
||||
mode: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.mode |
||||
}, |
||||
// 显示的文字
|
||||
text: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.text |
||||
}, |
||||
// 背景色
|
||||
bgColor: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.bgColor |
||||
}, |
||||
// 文字颜色
|
||||
color: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.color |
||||
}, |
||||
// 文字大小
|
||||
fontSize: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.avatar.fontSize |
||||
}, |
||||
// 显示的图标
|
||||
icon: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.icon |
||||
}, |
||||
// 显示小程序头像,只对百度,微信,QQ小程序有效
|
||||
mpAvatar: { |
||||
type: Boolean, |
||||
default: uni.$u.props.avatar.mpAvatar |
||||
}, |
||||
// 是否使用随机背景色
|
||||
randomBgColor: { |
||||
type: Boolean, |
||||
default: uni.$u.props.avatar.randomBgColor |
||||
}, |
||||
// 加载失败的默认头像(组件有内置默认图片)
|
||||
defaultUrl: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.defaultUrl |
||||
}, |
||||
// 如果配置了randomBgColor为true,且配置了此值,则从默认的背景色数组中取出对应索引的颜色值,取值0-19之间
|
||||
colorIndex: { |
||||
type: [String, Number], |
||||
// 校验参数规则,索引在0-19之间
|
||||
validator(n) { |
||||
return uni.$u.test.range(n, [0, 19]) || n === '' |
||||
}, |
||||
default: uni.$u.props.avatar.colorIndex |
||||
}, |
||||
// 组件标识符
|
||||
name: { |
||||
type: String, |
||||
default: uni.$u.props.avatar.name |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,172 @@ |
||||
<template> |
||||
<view |
||||
class="u-avatar" |
||||
:class="[`u-avatar--${shape}`]" |
||||
:style="[{ |
||||
backgroundColor: (text || icon) ? (randomBgColor ? colors[colorIndex !== '' ? colorIndex : $u.random(0, 19)] : bgColor) : 'transparent', |
||||
width: $u.addUnit(size), |
||||
height: $u.addUnit(size), |
||||
}, $u.addStyle(customStyle)]" |
||||
@tap="clickHandler" |
||||
> |
||||
<slot> |
||||
<!-- #ifdef MP-WEIXIN || MP-QQ || MP-BAIDU --> |
||||
<open-data |
||||
v-if="mpAvatar && allowMp" |
||||
type="userAvatarUrl" |
||||
:style="[{ |
||||
width: $u.addUnit(size), |
||||
height: $u.addUnit(size) |
||||
}]" |
||||
/> |
||||
<!-- #endif --> |
||||
<!-- #ifndef MP-WEIXIN && MP-QQ && MP-BAIDU --> |
||||
<template v-if="mpAvatar && allowMp"></template> |
||||
<!-- #endif --> |
||||
<u-icon |
||||
v-else-if="icon" |
||||
:name="icon" |
||||
:size="fontSize" |
||||
:color="color" |
||||
></u-icon> |
||||
<u--text |
||||
v-else-if="text" |
||||
:text="text" |
||||
:size="fontSize" |
||||
:color="color" |
||||
align="center" |
||||
customStyle="justify-content: center" |
||||
></u--text> |
||||
<image |
||||
class="u-avatar__image" |
||||
v-else |
||||
:class="[`u-avatar__image--${shape}`]" |
||||
:src="avatarUrl || defaultUrl" |
||||
:mode="mode" |
||||
@error="errorHandler" |
||||
:style="[{ |
||||
width: $u.addUnit(size), |
||||
height: $u.addUnit(size) |
||||
}]" |
||||
></image> |
||||
</slot> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js'; |
||||
const base64Avatar = |
||||
""; |
||||
/** |
||||
* Avatar 头像 |
||||
* @description 本组件一般用于展示头像的地方,如个人中心,或者评论列表页的用户头像展示等场所。 |
||||
* @tutorial https://www.uviewui.com/components/avatar.html |
||||
* |
||||
* @property {String} src 头像路径,如加载失败,将会显示默认头像(不能为相对路径) |
||||
* @property {String} shape 头像形状 ( circle (默认) | square) |
||||
* @property {String | Number} size 头像尺寸,可以为指定字符串(large, default, mini),或者数值 (默认 40 ) |
||||
* @property {String} mode 头像图片的裁剪类型,与uni的image组件的mode参数一致,如效果达不到需求,可尝试传widthFix值 (默认 'scaleToFill' ) |
||||
* @property {String} text 用文字替代图片,级别优先于src |
||||
* @property {String} bgColor 背景颜色,一般显示文字时用 (默认 '#c0c4cc' ) |
||||
* @property {String} color 文字颜色 (默认 '#ffffff' ) |
||||
* @property {String | Number} fontSize 文字大小 (默认 18 ) |
||||
* @property {String} icon 显示的图标 |
||||
* @property {Boolean} mpAvatar 显示小程序头像,只对百度,微信,QQ小程序有效 (默认 false ) |
||||
* @property {Boolean} randomBgColor 是否使用随机背景色 (默认 false ) |
||||
* @property {String} defaultUrl 加载失败的默认头像(组件有内置默认图片) |
||||
* @property {String | Number} colorIndex 如果配置了randomBgColor为true,且配置了此值,则从默认的背景色数组中取出对应索引的颜色值,取值0-19之间 |
||||
* @property {String} name 组件标识符 (默认 'level' ) |
||||
* @property {Object} customStyle 定义需要用到的外部样式 |
||||
* |
||||
* @event {Function} click 点击组件时触发 index: 用户传递的标识符 |
||||
* @example <u-avatar :src="src" mode="square"></u-avatar> |
||||
*/ |
||||
export default { |
||||
name: 'u-avatar', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
data() { |
||||
return { |
||||
// 如果配置randomBgColor参数为true,在图标或者文字的模式下,会随机从中取出一个颜色值当做背景色 |
||||
colors: ['#ffb34b', '#f2bba9', '#f7a196', '#f18080', '#88a867', '#bfbf39', '#89c152', '#94d554', '#f19ec2', |
||||
'#afaae4', '#e1b0df', '#c38cc1', '#72dcdc', '#9acdcb', '#77b1cc', '#448aca', '#86cefa', '#98d1ee', |
||||
'#73d1f1', |
||||
'#80a7dc' |
||||
], |
||||
avatarUrl: this.src, |
||||
allowMp: false |
||||
} |
||||
}, |
||||
watch: { |
||||
// 监听头像src的变化,赋值给内部的avatarUrl变量,因为图片加载失败时,需要修改图片的src为默认值 |
||||
// 而组件内部不能直接修改props的值,所以需要一个中间变量 |
||||
src: { |
||||
immediate: true, |
||||
handler(newVal) { |
||||
this.avatarUrl = newVal |
||||
// 如果没有传src,则主动触发error事件,用于显示默认的头像,否则src为''空字符等的时候,会无内容展示 |
||||
if(!newVal) { |
||||
this.errorHandler() |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
imageStyle() { |
||||
const style = {} |
||||
return style |
||||
} |
||||
}, |
||||
created() { |
||||
this.init() |
||||
}, |
||||
methods: { |
||||
init() { |
||||
// 目前只有这几个小程序平台具有open-data标签 |
||||
// 其他平台可以通过uni.getUserInfo类似接口获取信息,但是需要弹窗授权(首次),不合符组件逻辑 |
||||
// 故目前自动获取小程序头像只支持这几个平台 |
||||
// #ifdef MP-WEIXIN || MP-QQ || MP-BAIDU |
||||
this.allowMp = true |
||||
// #endif |
||||
}, |
||||
// 判断传入的name属性,是否图片路径,只要带有"/"均认为是图片形式 |
||||
isImg() { |
||||
return this.src.indexOf('/') !== -1 |
||||
}, |
||||
// 图片加载时失败时触发 |
||||
errorHandler() { |
||||
this.avatarUrl = this.defaultUrl || base64Avatar |
||||
}, |
||||
clickHandler() { |
||||
this.$emit('click', this.name) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
.u-avatar { |
||||
@include flex; |
||||
align-items: center; |
||||
justify-content: center; |
||||
|
||||
&--circle { |
||||
border-radius: 100px; |
||||
} |
||||
|
||||
&--square { |
||||
border-radius: 4px; |
||||
} |
||||
|
||||
&__image { |
||||
&--circle { |
||||
border-radius: 100px; |
||||
} |
||||
|
||||
&--square { |
||||
border-radius: 4px; |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,54 @@ |
||||
export default { |
||||
props: { |
||||
// 返回顶部的形状,circle-圆形,square-方形
|
||||
mode: { |
||||
type: String, |
||||
default: uni.$u.props.backtop.mode |
||||
}, |
||||
// 自定义图标
|
||||
icon: { |
||||
type: String, |
||||
default: uni.$u.props.backtop.icon |
||||
}, |
||||
// 提示文字
|
||||
text: { |
||||
type: String, |
||||
default: uni.$u.props.backtop.text |
||||
}, |
||||
// 返回顶部滚动时间
|
||||
duration: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.duration |
||||
}, |
||||
// 滚动距离
|
||||
scrollTop: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.scrollTop |
||||
}, |
||||
// 距离顶部多少距离显示,单位px
|
||||
top: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.top |
||||
}, |
||||
// 返回顶部按钮到底部的距离,单位px
|
||||
bottom: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.bottom |
||||
}, |
||||
// 返回顶部按钮到右边的距离,单位px
|
||||
right: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.right |
||||
}, |
||||
// 层级
|
||||
zIndex: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.backtop.zIndex |
||||
}, |
||||
// 图标的样式,对象形式
|
||||
iconStyle: { |
||||
type: Object, |
||||
default: uni.$u.props.backtop.iconStyle |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,129 @@ |
||||
<template> |
||||
<u-transition |
||||
mode="fade" |
||||
:customStyle="backTopStyle" |
||||
:show="show" |
||||
> |
||||
<view |
||||
class="u-back-top" |
||||
:style="[contentStyle]" |
||||
v-if="!$slots.default && !$slots.$default" |
||||
@click="backToTop" |
||||
> |
||||
<u-icon |
||||
:name="icon" |
||||
:custom-style="iconStyle" |
||||
></u-icon> |
||||
<text |
||||
v-if="text" |
||||
class="u-back-top__text" |
||||
>{{text}}</text> |
||||
</view> |
||||
<slot v-else /> |
||||
</u-transition> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js'; |
||||
// #ifdef APP-NVUE |
||||
const dom = weex.requireModule('dom') |
||||
// #endif |
||||
/** |
||||
* backTop 返回顶部 |
||||
* @description 本组件一个用于长页面,滑动一定距离后,出现返回顶部按钮,方便快速返回顶部的场景。 |
||||
* @tutorial https://uviewui.com/components/backTop.html |
||||
* |
||||
* @property {String} mode 返回顶部的形状,circle-圆形,square-方形 (默认 'circle' ) |
||||
* @property {String} icon 自定义图标 (默认 'arrow-upward' ) 见官方文档示例 |
||||
* @property {String} text 提示文字 |
||||
* @property {String | Number} duration 返回顶部滚动时间 (默认 100) |
||||
* @property {String | Number} scrollTop 滚动距离 (默认 0 ) |
||||
* @property {String | Number} top 距离顶部多少距离显示,单位px (默认 400 ) |
||||
* @property {String | Number} bottom 返回顶部按钮到底部的距离,单位px (默认 100 ) |
||||
* @property {String | Number} right 返回顶部按钮到右边的距离,单位px (默认 20 ) |
||||
* @property {String | Number} zIndex 层级 (默认 9 ) |
||||
* @property {Object<Object>} iconStyle 图标的样式,对象形式 (默认 {color: '#909399',fontSize: '19px'}) |
||||
* @property {Object} customStyle 定义需要用到的外部样式 |
||||
* |
||||
* @example <u-back-top :scrollTop="scrollTop"></u-back-top> |
||||
*/ |
||||
export default { |
||||
name: 'u-back-top', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin,props], |
||||
computed: { |
||||
backTopStyle() { |
||||
// 动画组件样式 |
||||
const style = { |
||||
bottom: uni.$u.addUnit(this.bottom), |
||||
right: uni.$u.addUnit(this.right), |
||||
width: '40px', |
||||
height: '40px', |
||||
position: 'fixed', |
||||
zIndex: 10, |
||||
} |
||||
return style |
||||
}, |
||||
show() { |
||||
return uni.$u.getPx(this.scrollTop) > uni.$u.getPx(this.top) |
||||
}, |
||||
contentStyle() { |
||||
const style = {} |
||||
let radius = 0 |
||||
// 是否圆形 |
||||
if(this.mode === 'circle') { |
||||
radius = '100px' |
||||
} else { |
||||
radius = '4px' |
||||
} |
||||
// 为了兼容安卓nvue,只能这么分开写 |
||||
style.borderTopLeftRadius = radius |
||||
style.borderTopRightRadius = radius |
||||
style.borderBottomLeftRadius = radius |
||||
style.borderBottomRightRadius = radius |
||||
return uni.$u.deepMerge(style, uni.$u.addStyle(this.customStyle)) |
||||
} |
||||
}, |
||||
methods: { |
||||
backToTop() { |
||||
// #ifdef APP-NVUE |
||||
if (!this.$parent.$refs['u-back-top']) { |
||||
uni.$u.error(`nvue页面需要给页面最外层元素设置"ref='u-back-top'`) |
||||
} |
||||
dom.scrollToElement(this.$parent.$refs['u-back-top'], { |
||||
offset: 0 |
||||
}) |
||||
// #endif |
||||
|
||||
// #ifndef APP-NVUE |
||||
uni.pageScrollTo({ |
||||
scrollTop: 0, |
||||
duration: this.duration |
||||
}); |
||||
// #endif |
||||
this.$emit('click') |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import '../../libs/css/components.scss'; |
||||
$u-back-top-flex:1 !default; |
||||
$u-back-top-height:100% !default; |
||||
$u-back-top-background-color:#E1E1E1 !default; |
||||
$u-back-top-tips-font-size:12px !default; |
||||
.u-back-top { |
||||
@include flex; |
||||
flex-direction: column; |
||||
align-items: center; |
||||
flex:$u-back-top-flex; |
||||
height: $u-back-top-height; |
||||
justify-content: center; |
||||
background-color: $u-back-top-background-color; |
||||
|
||||
&__tips { |
||||
font-size:$u-back-top-tips-font-size; |
||||
transform: scale(0.8); |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,72 @@ |
||||
export default { |
||||
props: { |
||||
// 是否显示圆点
|
||||
isDot: { |
||||
type: Boolean, |
||||
default: uni.$u.props.badge.isDot |
||||
}, |
||||
// 显示的内容
|
||||
value: { |
||||
type: [Number, String], |
||||
default: uni.$u.props.badge.value |
||||
}, |
||||
// 是否显示
|
||||
show: { |
||||
type: Boolean, |
||||
default: uni.$u.props.badge.show |
||||
}, |
||||
// 最大值,超过最大值会显示 '{max}+'
|
||||
max: { |
||||
type: [Number, String], |
||||
default: uni.$u.props.badge.max |
||||
}, |
||||
// 主题类型,error|warning|success|primary
|
||||
type: { |
||||
type: String, |
||||
default: uni.$u.props.badge.type |
||||
}, |
||||
// 当数值为 0 时,是否展示 Badge
|
||||
showZero: { |
||||
type: Boolean, |
||||
default: uni.$u.props.badge.showZero |
||||
}, |
||||
// 背景颜色,优先级比type高,如设置,type参数会失效
|
||||
bgColor: { |
||||
type: [String, null], |
||||
default: uni.$u.props.badge.bgColor |
||||
}, |
||||
// 字体颜色
|
||||
color: { |
||||
type: [String, null], |
||||
default: uni.$u.props.badge.color |
||||
}, |
||||
// 徽标形状,circle-四角均为圆角,horn-左下角为直角
|
||||
shape: { |
||||
type: String, |
||||
default: uni.$u.props.badge.shape |
||||
}, |
||||
// 设置数字的显示方式,overflow|ellipsis|limit
|
||||
// overflow会根据max字段判断,超出显示`${max}+`
|
||||
// ellipsis会根据max判断,超出显示`${max}...`
|
||||
// limit会依据1000作为判断条件,超出1000,显示`${value/1000}K`,比如2.2k、3.34w,最多保留2位小数
|
||||
numberType: { |
||||
type: String, |
||||
default: uni.$u.props.badge.numberType |
||||
}, |
||||
// 设置badge的位置偏移,格式为 [x, y],也即设置的为top和right的值,absolute为true时有效
|
||||
offset: { |
||||
type: Array, |
||||
default: uni.$u.props.badge.offset |
||||
}, |
||||
// 是否反转背景和字体颜色
|
||||
inverted: { |
||||
type: Boolean, |
||||
default: uni.$u.props.badge.inverted |
||||
}, |
||||
// 是否绝对定位
|
||||
absolute: { |
||||
type: Boolean, |
||||
default: uni.$u.props.badge.absolute |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,171 @@ |
||||
<template> |
||||
<text |
||||
v-if="show && ((Number(value) === 0 ? showZero : true) || isDot)" |
||||
:class="[isDot ? 'u-badge--dot' : 'u-badge--not-dot', inverted && 'u-badge--inverted', shape === 'horn' && 'u-badge--horn', `u-badge--${type}${inverted ? '--inverted' : ''}`]" |
||||
:style="[$u.addStyle(customStyle), badgeStyle]" |
||||
class="u-badge" |
||||
>{{ isDot ? '' :showValue }}</text> |
||||
</template> |
||||
|
||||
<script> |
||||
import props from './props.js'; |
||||
/** |
||||
* badge 徽标数 |
||||
* @description 该组件一般用于图标右上角显示未读的消息数量,提示用户点击,有圆点和圆包含文字两种形式。 |
||||
* @tutorial https://uviewui.com/components/badge.html |
||||
* |
||||
* @property {Boolean} isDot 是否显示圆点 (默认 false ) |
||||
* @property {String | Number} value 显示的内容 |
||||
* @property {Boolean} show 是否显示 (默认 true ) |
||||
* @property {String | Number} max 最大值,超过最大值会显示 '{max}+' (默认999) |
||||
* @property {String} type 主题类型,error|warning|success|primary (默认 'error' ) |
||||
* @property {Boolean} showZero 当数值为 0 时,是否展示 Badge (默认 false ) |
||||
* @property {String} bgColor 背景颜色,优先级比type高,如设置,type参数会失效 |
||||
* @property {String} color 字体颜色 (默认 '#ffffff' ) |
||||
* @property {String} shape 徽标形状,circle-四角均为圆角,horn-左下角为直角 (默认 'circle' ) |
||||
* @property {String} numberType 设置数字的显示方式,overflow|ellipsis|limit (默认 'overflow' ) |
||||
* @property {Array}} offset 设置badge的位置偏移,格式为 [x, y],也即设置的为top和right的值,absolute为true时有效 |
||||
* @property {Boolean} inverted 是否反转背景和字体颜色(默认 false ) |
||||
* @property {Boolean} absolute 是否绝对定位(默认 false ) |
||||
* @property {Object} customStyle 定义需要用到的外部样式 |
||||
* @example <u-badge :type="type" :count="count"></u-badge> |
||||
*/ |
||||
export default { |
||||
name: 'u-badge', |
||||
mixins: [uni.$u.mpMixin, props, uni.$u.mixin], |
||||
computed: { |
||||
// 是否将badge中心与父组件右上角重合 |
||||
boxStyle() { |
||||
let style = {}; |
||||
return style; |
||||
}, |
||||
// 整个组件的样式 |
||||
badgeStyle() { |
||||
const style = {} |
||||
if(this.color) { |
||||
style.color = this.color |
||||
} |
||||
if (this.bgColor && !this.inverted) { |
||||
style.backgroundColor = this.bgColor |
||||
} |
||||
if (this.absolute) { |
||||
style.position = 'absolute' |
||||
// 如果有设置offset参数 |
||||
if(this.offset.length) { |
||||
// top和right分为为offset的第一个和第二个值,如果没有第二个值,则right等于top |
||||
const top = this.offset[0] |
||||
const right = this.offset[1] || top |
||||
style.top = uni.$u.addUnit(top) |
||||
style.right = uni.$u.addUnit(right) |
||||
} |
||||
} |
||||
return style |
||||
}, |
||||
showValue() { |
||||
switch (this.numberType) { |
||||
case "overflow": |
||||
return Number(this.value) > Number(this.max) ? this.max + "+" : this.value |
||||
break; |
||||
case "ellipsis": |
||||
return Number(this.value) > Number(this.max) ? "..." : this.value |
||||
break; |
||||
case "limit": |
||||
return Number(this.value) > 999 ? Number(this.value) >= 9999 ? |
||||
Math.floor(this.value / 1e4 * 100) / 100 + "w" : Math.floor(this.value / |
||||
1e3 * 100) / 100 + "k" : this.value |
||||
break; |
||||
default: |
||||
return Number(this.value) |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
$u-badge-primary: $u-primary !default; |
||||
$u-badge-error: $u-error !default; |
||||
$u-badge-success: $u-success !default; |
||||
$u-badge-info: $u-info !default; |
||||
$u-badge-warning: $u-warning !default; |
||||
$u-badge-dot-radius: 100px !default; |
||||
$u-badge-dot-size: 8px !default; |
||||
$u-badge-dot-right: 4px !default; |
||||
$u-badge-dot-top: 0 !default; |
||||
$u-badge-text-font-size: 11px !default; |
||||
$u-badge-text-right: 10px !default; |
||||
$u-badge-text-padding: 2px 5px !default; |
||||
$u-badge-text-align: center !default; |
||||
$u-badge-text-color: #FFFFFF !default; |
||||
|
||||
.u-badge { |
||||
border-top-right-radius: $u-badge-dot-radius; |
||||
border-top-left-radius: $u-badge-dot-radius; |
||||
border-bottom-left-radius: $u-badge-dot-radius; |
||||
border-bottom-right-radius: $u-badge-dot-radius; |
||||
@include flex; |
||||
line-height: $u-badge-text-font-size; |
||||
text-align: $u-badge-text-align; |
||||
font-size: $u-badge-text-font-size; |
||||
color: $u-badge-text-color; |
||||
|
||||
&--dot { |
||||
height: $u-badge-dot-size; |
||||
width: $u-badge-dot-size; |
||||
} |
||||
|
||||
&--inverted { |
||||
font-size: 13px; |
||||
} |
||||
|
||||
&--not-dot { |
||||
padding: $u-badge-text-padding; |
||||
} |
||||
|
||||
&--horn { |
||||
border-bottom-left-radius: 0; |
||||
} |
||||
|
||||
&--primary { |
||||
background-color: $u-badge-primary; |
||||
} |
||||
|
||||
&--primary--inverted { |
||||
color: $u-badge-primary; |
||||
} |
||||
|
||||
&--error { |
||||
background-color: $u-badge-error; |
||||
} |
||||
|
||||
&--error--inverted { |
||||
color: $u-badge-error; |
||||
} |
||||
|
||||
&--success { |
||||
background-color: $u-badge-success; |
||||
} |
||||
|
||||
&--success--inverted { |
||||
color: $u-badge-success; |
||||
} |
||||
|
||||
&--info { |
||||
background-color: $u-badge-info; |
||||
} |
||||
|
||||
&--info--inverted { |
||||
color: $u-badge-info; |
||||
} |
||||
|
||||
&--warning { |
||||
background-color: $u-badge-warning; |
||||
} |
||||
|
||||
&--warning--inverted { |
||||
color: $u-badge-warning; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,46 @@ |
||||
$u-button-active-opacity:0.75 !default; |
||||
$u-button-loading-text-margin-left:4px !default; |
||||
$u-button-text-color: #FFFFFF !default; |
||||
$u-button-text-plain-error-color:$u-error !default; |
||||
$u-button-text-plain-warning-color:$u-warning !default; |
||||
$u-button-text-plain-success-color:$u-success !default; |
||||
$u-button-text-plain-info-color:$u-info !default; |
||||
$u-button-text-plain-primary-color:$u-primary !default; |
||||
.u-button { |
||||
&--active { |
||||
opacity: $u-button-active-opacity; |
||||
} |
||||
|
||||
&--active--plain { |
||||
background-color: rgb(217, 217, 217); |
||||
} |
||||
|
||||
&__loading-text { |
||||
margin-left:$u-button-loading-text-margin-left; |
||||
} |
||||
|
||||
&__text, |
||||
&__loading-text { |
||||
color:$u-button-text-color; |
||||
} |
||||
|
||||
&__text--plain--error { |
||||
color:$u-button-text-plain-error-color; |
||||
} |
||||
|
||||
&__text--plain--warning { |
||||
color:$u-button-text-plain-warning-color; |
||||
} |
||||
|
||||
&__text--plain--success{ |
||||
color:$u-button-text-plain-success-color; |
||||
} |
||||
|
||||
&__text--plain--info { |
||||
color:$u-button-text-plain-info-color; |
||||
} |
||||
|
||||
&__text--plain--primary { |
||||
color:$u-button-text-plain-primary-color; |
||||
} |
||||
} |
@ -0,0 +1,161 @@ |
||||
/* |
||||
* @Author : LQ |
||||
* @Description : |
||||
* @version : 1.0 |
||||
* @Date : 2021-08-16 10:04:04 |
||||
* @LastAuthor : LQ |
||||
* @lastTime : 2021-08-16 10:04:24 |
||||
* @FilePath : /u-view2.0/uview-ui/components/u-button/props.js |
||||
*/ |
||||
export default { |
||||
props: { |
||||
// 是否细边框
|
||||
hairline: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.hairline |
||||
}, |
||||
// 按钮的预置样式,info,primary,error,warning,success
|
||||
type: { |
||||
type: String, |
||||
default: uni.$u.props.button.type |
||||
}, |
||||
// 按钮尺寸,large,normal,small,mini
|
||||
size: { |
||||
type: String, |
||||
default: uni.$u.props.button.size |
||||
}, |
||||
// 按钮形状,circle(两边为半圆),square(带圆角)
|
||||
shape: { |
||||
type: String, |
||||
default: uni.$u.props.button.shape |
||||
}, |
||||
// 按钮是否镂空
|
||||
plain: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.plain |
||||
}, |
||||
// 是否禁止状态
|
||||
disabled: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.disabled |
||||
}, |
||||
// 是否加载中
|
||||
loading: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.loading |
||||
}, |
||||
// 加载中提示文字
|
||||
loadingText: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.loadingText |
||||
}, |
||||
// 加载状态图标类型
|
||||
loadingMode: { |
||||
type: String, |
||||
default: uni.$u.props.button.loadingMode |
||||
}, |
||||
// 加载图标大小
|
||||
loadingSize: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.loadingSize |
||||
}, |
||||
// 开放能力,具体请看uniapp稳定关于button组件部分说明
|
||||
// https://uniapp.dcloud.io/component/button
|
||||
openType: { |
||||
type: String, |
||||
default: uni.$u.props.button.openType |
||||
}, |
||||
// 用于 <form> 组件,点击分别会触发 <form> 组件的 submit/reset 事件
|
||||
// 取值为submit(提交表单),reset(重置表单)
|
||||
formType: { |
||||
type: String, |
||||
default: uni.$u.props.button.formType |
||||
}, |
||||
// 打开 APP 时,向 APP 传递的参数,open-type=launchApp时有效
|
||||
// 只微信小程序、QQ小程序有效
|
||||
appParameter: { |
||||
type: String, |
||||
default: uni.$u.props.button.appParameter |
||||
}, |
||||
// 指定是否阻止本节点的祖先节点出现点击态,微信小程序有效
|
||||
hoverStopPropagation: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.hoverStopPropagation |
||||
}, |
||||
// 指定返回用户信息的语言,zh_CN 简体中文,zh_TW 繁体中文,en 英文。只微信小程序有效
|
||||
lang: { |
||||
type: String, |
||||
default: uni.$u.props.button.lang |
||||
}, |
||||
// 会话来源,open-type="contact"时有效。只微信小程序有效
|
||||
sessionFrom: { |
||||
type: String, |
||||
default: uni.$u.props.button.sessionFrom |
||||
}, |
||||
// 会话内消息卡片标题,open-type="contact"时有效
|
||||
// 默认当前标题,只微信小程序有效
|
||||
sendMessageTitle: { |
||||
type: String, |
||||
default: uni.$u.props.button.sendMessageTitle |
||||
}, |
||||
// 会话内消息卡片点击跳转小程序路径,open-type="contact"时有效
|
||||
// 默认当前分享路径,只微信小程序有效
|
||||
sendMessagePath: { |
||||
type: String, |
||||
default: uni.$u.props.button.sendMessagePath |
||||
}, |
||||
// 会话内消息卡片图片,open-type="contact"时有效
|
||||
// 默认当前页面截图,只微信小程序有效
|
||||
sendMessageImg: { |
||||
type: String, |
||||
default: uni.$u.props.button.sendMessageImg |
||||
}, |
||||
// 是否显示会话内消息卡片,设置此参数为 true,用户进入客服会话会在右下角显示"可能要发送的小程序"提示,
|
||||
// 用户点击后可以快速发送小程序消息,open-type="contact"时有效
|
||||
showMessageCard: { |
||||
type: Boolean, |
||||
default: uni.$u.props.button.showMessageCard |
||||
}, |
||||
// 额外传参参数,用于小程序的data-xxx属性,通过target.dataset.name获取
|
||||
dataName: { |
||||
type: String, |
||||
default: uni.$u.props.button.dataName |
||||
}, |
||||
// 节流,一定时间内只能触发一次
|
||||
throttleTime: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.throttleTime |
||||
}, |
||||
// 按住后多久出现点击态,单位毫秒
|
||||
hoverStartTime: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.hoverStartTime |
||||
}, |
||||
// 手指松开后点击态保留时间,单位毫秒
|
||||
hoverStayTime: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.hoverStayTime |
||||
}, |
||||
// 按钮文字,之所以通过props传入,是因为slot传入的话
|
||||
// nvue中无法控制文字的样式
|
||||
text: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.button.text |
||||
}, |
||||
// 按钮图标
|
||||
icon: { |
||||
type: String, |
||||
default: uni.$u.props.button.icon |
||||
}, |
||||
// 按钮图标
|
||||
iconColor: { |
||||
type: String, |
||||
default: uni.$u.props.button.icon |
||||
}, |
||||
// 按钮颜色,支持传入linear-gradient渐变色
|
||||
color: { |
||||
type: String, |
||||
default: uni.$u.props.button.color |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,490 @@ |
||||
<template> |
||||
<!-- #ifndef APP-NVUE --> |
||||
<button |
||||
:hover-start-time="Number(hoverStartTime)" |
||||
:hover-stay-time="Number(hoverStayTime)" |
||||
:form-type="formType" |
||||
:open-type="openType" |
||||
:app-parameter="appParameter" |
||||
:hover-stop-propagation="hoverStopPropagation" |
||||
:send-message-title="sendMessageTitle" |
||||
:send-message-path="sendMessagePath" |
||||
:lang="lang" |
||||
:data-name="dataName" |
||||
:session-from="sessionFrom" |
||||
:send-message-img="sendMessageImg" |
||||
:show-message-card="showMessageCard" |
||||
@getphonenumber="getphonenumber" |
||||
@getuserinfo="getuserinfo" |
||||
@error="error" |
||||
@opensetting="opensetting" |
||||
@launchapp="launchapp" |
||||
:hover-class="!disabled && !loading ? 'u-button--active' : ''" |
||||
class="u-button u-reset-button" |
||||
:style="[baseColor, $u.addStyle(customStyle)]" |
||||
@tap="clickHandler" |
||||
:class="bemClass" |
||||
> |
||||
<template v-if="loading"> |
||||
<u-loading-icon |
||||
:mode="loadingMode" |
||||
:size="loadingSize * 1.15" |
||||
:color="loadingColor" |
||||
></u-loading-icon> |
||||
<text |
||||
class="u-button__loading-text" |
||||
:style="[{ fontSize: textSize + 'px' }]" |
||||
>{{ loadingText || text }}</text |
||||
> |
||||
</template> |
||||
<template v-else> |
||||
<u-icon |
||||
v-if="icon" |
||||
:name="icon" |
||||
:color="iconColorCom" |
||||
:size="textSize * 1.35" |
||||
:customStyle="{ marginRight: '2px' }" |
||||
></u-icon> |
||||
<slot> |
||||
<text |
||||
class="u-button__text" |
||||
:style="[{ fontSize: textSize + 'px' }]" |
||||
>{{ text }}</text |
||||
> |
||||
</slot> |
||||
</template> |
||||
</button> |
||||
<!-- #endif --> |
||||
|
||||
<!-- #ifdef APP-NVUE --> |
||||
<view |
||||
:hover-start-time="Number(hoverStartTime)" |
||||
:hover-stay-time="Number(hoverStayTime)" |
||||
class="u-button" |
||||
:hover-class=" |
||||
!disabled && !loading && !color && (plain || type === 'info') |
||||
? 'u-button--active--plain' |
||||
: !disabled && !loading && !plain |
||||
? 'u-button--active' |
||||
: '' |
||||
" |
||||
@tap="clickHandler" |
||||
:class="bemClass" |
||||
:style="[baseColor, $u.addStyle(customStyle)]" |
||||
> |
||||
<template v-if="loading"> |
||||
<u-loading-icon |
||||
:mode="loadingMode" |
||||
:size="loadingSize * 1.15" |
||||
:color="loadingColor" |
||||
></u-loading-icon> |
||||
<text |
||||
class="u-button__loading-text" |
||||
:style="[nvueTextStyle]" |
||||
:class="[plain && `u-button__text--plain--${type}`]" |
||||
>{{ loadingText || text }}</text |
||||
> |
||||
</template> |
||||
<template v-else> |
||||
<u-icon |
||||
v-if="icon" |
||||
:name="icon" |
||||
:color="iconColorCom" |
||||
:size="textSize * 1.35" |
||||
></u-icon> |
||||
<text |
||||
class="u-button__text" |
||||
:style="[ |
||||
{ |
||||
marginLeft: icon ? '2px' : 0, |
||||
}, |
||||
nvueTextStyle, |
||||
]" |
||||
:class="[plain && `u-button__text--plain--${type}`]" |
||||
>{{ text }}</text |
||||
> |
||||
</template> |
||||
</view> |
||||
<!-- #endif --> |
||||
</template> |
||||
|
||||
<script> |
||||
import button from "../../libs/mixin/button.js"; |
||||
import openType from "../../libs/mixin/openType.js"; |
||||
import props from "./props.js"; |
||||
/** |
||||
* button 按钮 |
||||
* @description Button 按钮 |
||||
* @tutorial https://www.uviewui.com/components/button.html |
||||
* |
||||
* @property {Boolean} hairline 是否显示按钮的细边框 (默认 true ) |
||||
* @property {String} type 按钮的预置样式,info,primary,error,warning,success (默认 'info' ) |
||||
* @property {String} size 按钮尺寸,large,normal,mini (默认 normal) |
||||
* @property {String} shape 按钮形状,circle(两边为半圆),square(带圆角) (默认 'square' ) |
||||
* @property {Boolean} plain 按钮是否镂空,背景色透明 (默认 false) |
||||
* @property {Boolean} disabled 是否禁用 (默认 false) |
||||
* @property {Boolean} loading 按钮名称前是否带 loading 图标(App-nvue 平台,在 ios 上为雪花,Android上为圆圈) (默认 false) |
||||
* @property {String | Number} loadingText 加载中提示文字 |
||||
* @property {String} loadingMode 加载状态图标类型 (默认 'spinner' ) |
||||
* @property {String | Number} loadingSize 加载图标大小 (默认 15 ) |
||||
* @property {String} openType 开放能力,具体请看uniapp稳定关于button组件部分说明 |
||||
* @property {String} formType 用于 <form> 组件,点击分别会触发 <form> 组件的 submit/reset 事件 |
||||
* @property {String} appParameter 打开 APP 时,向 APP 传递的参数,open-type=launchApp时有效 (注:只微信小程序、QQ小程序有效) |
||||
* @property {Boolean} hoverStopPropagation 指定是否阻止本节点的祖先节点出现点击态,微信小程序有效(默认 true ) |
||||
* @property {String} lang 指定返回用户信息的语言,zh_CN 简体中文,zh_TW 繁体中文,en 英文(默认 en ) |
||||
* @property {String} sessionFrom 会话来源,openType="contact"时有效 |
||||
* @property {String} sendMessageTitle 会话内消息卡片标题,openType="contact"时有效 |
||||
* @property {String} sendMessagePath 会话内消息卡片点击跳转小程序路径,openType="contact"时有效 |
||||
* @property {String} sendMessageImg 会话内消息卡片图片,openType="contact"时有效 |
||||
* @property {Boolean} showMessageCard 是否显示会话内消息卡片,设置此参数为 true,用户进入客服会话会在右下角显示"可能要发送的小程序"提示,用户点击后可以快速发送小程序消息,openType="contact"时有效(默认false) |
||||
* @property {String} dataName 额外传参参数,用于小程序的data-xxx属性,通过target.dataset.name获取 |
||||
* @property {String | Number} throttleTime 节流,一定时间内只能触发一次 (默认 0 ) |
||||
* @property {String | Number} hoverStartTime 按住后多久出现点击态,单位毫秒 (默认 0 ) |
||||
* @property {String | Number} hoverStayTime 手指松开后点击态保留时间,单位毫秒 (默认 200 ) |
||||
* @property {String | Number} text 按钮文字,之所以通过props传入,是因为slot传入的话(注:nvue中无法控制文字的样式) |
||||
* @property {String} icon 按钮图标 |
||||
* @property {String} iconColor 按钮图标颜色 |
||||
* @property {String} color 按钮颜色,支持传入linear-gradient渐变色 |
||||
* @property {Object} customStyle 定义需要用到的外部样式 |
||||
* |
||||
* @event {Function} click 非禁止并且非加载中,才能点击 |
||||
* @event {Function} getphonenumber open-type="getPhoneNumber"时有效 |
||||
* @event {Function} getuserinfo 用户点击该按钮时,会返回获取到的用户信息,从返回参数的detail中获取到的值同uni.getUserInfo |
||||
* @event {Function} error 当使用开放能力时,发生错误的回调 |
||||
* @event {Function} opensetting 在打开授权设置页并关闭后回调 |
||||
* @event {Function} launchapp 打开 APP 成功的回调 |
||||
* @example <u-button>月落</u-button> |
||||
*/ |
||||
export default { |
||||
name: "u-button", |
||||
// #ifdef MP |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, button, openType, props], |
||||
// #endif |
||||
// #ifndef MP |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
// #endif |
||||
data() { |
||||
return {}; |
||||
}, |
||||
computed: { |
||||
// 生成bem风格的类名 |
||||
bemClass() { |
||||
// this.bem为一个computed变量,在mixin中 |
||||
if (!this.color) { |
||||
return this.bem( |
||||
"button", |
||||
["type", "shape", "size"], |
||||
["disabled", "plain", "hairline"] |
||||
); |
||||
} else { |
||||
// 由于nvue的原因,在有color参数时,不需要传入type,否则会生成type相关的类型,影响最终的样式 |
||||
return this.bem( |
||||
"button", |
||||
["shape", "size"], |
||||
["disabled", "plain", "hairline"] |
||||
); |
||||
} |
||||
}, |
||||
loadingColor() { |
||||
if (this.plain) { |
||||
// 如果有设置color值,则用color值,否则使用type主题颜色 |
||||
return this.color |
||||
? this.color |
||||
: uni.$u.config.color[`u-${this.type}`]; |
||||
} |
||||
if (this.type === "info") { |
||||
return "#c9c9c9"; |
||||
} |
||||
return "rgb(200, 200, 200)"; |
||||
}, |
||||
iconColorCom() { |
||||
// 如果是镂空状态,设置了color就用color值,否则使用主题颜色, |
||||
// u-icon的color能接受一个主题颜色的值 |
||||
if (this.iconColor) return this.iconColor; |
||||
if (this.plain) { |
||||
return this.color ? this.color : this.type; |
||||
} else { |
||||
return this.type === "info" ? "#000000" : "#ffffff"; |
||||
} |
||||
}, |
||||
baseColor() { |
||||
let style = {}; |
||||
if (this.color) { |
||||
// 针对自定义了color颜色的情况,镂空状态下,就是用自定义的颜色 |
||||
style.color = this.plain ? this.color : "white"; |
||||
if (!this.plain) { |
||||
// 非镂空,背景色使用自定义的颜色 |
||||
style["background-color"] = this.color; |
||||
} |
||||
if (this.color.indexOf("gradient") !== -1) { |
||||
// 如果自定义的颜色为渐变色,不显示边框,以及通过backgroundImage设置渐变色 |
||||
// weex文档说明可以写borderWidth的形式,为什么这里需要分开写? |
||||
// 因为weex是阿里巴巴为了部门业绩考核而做的你懂的东西,所以需要这么写才有效 |
||||
style.borderTopWidth = 0; |
||||
style.borderRightWidth = 0; |
||||
style.borderBottomWidth = 0; |
||||
style.borderLeftWidth = 0; |
||||
if (!this.plain) { |
||||
style.backgroundImage = this.color; |
||||
} |
||||
} else { |
||||
// 非渐变色,则设置边框相关的属性 |
||||
style.borderColor = this.color; |
||||
style.borderWidth = "1px"; |
||||
style.borderStyle = "solid"; |
||||
} |
||||
} |
||||
return style; |
||||
}, |
||||
// nvue版本按钮的字体不会继承父组件的颜色,需要对每一个text组件进行单独的设置 |
||||
nvueTextStyle() { |
||||
let style = {}; |
||||
// 针对自定义了color颜色的情况,镂空状态下,就是用自定义的颜色 |
||||
if (this.type === "info") { |
||||
style.color = "#323233"; |
||||
} |
||||
if (this.color) { |
||||
style.color = this.plain ? this.color : "white"; |
||||
} |
||||
style.fontSize = this.textSize + "px"; |
||||
return style; |
||||
}, |
||||
// 字体大小 |
||||
textSize() { |
||||
let fontSize = 14, |
||||
{ size } = this; |
||||
if (size === "large") fontSize = 16; |
||||
if (size === "normal") fontSize = 14; |
||||
if (size === "small") fontSize = 12; |
||||
if (size === "mini") fontSize = 10; |
||||
return fontSize; |
||||
}, |
||||
}, |
||||
methods: { |
||||
clickHandler() { |
||||
// 非禁止并且非加载中,才能点击 |
||||
if (!this.disabled && !this.loading) { |
||||
// 进行节流控制,每this.throttle毫秒内,只在开始处执行 |
||||
uni.$u.throttle(() => { |
||||
this.$emit("click"); |
||||
}, this.throttleTime); |
||||
} |
||||
}, |
||||
// 下面为对接uniapp官方按钮开放能力事件回调的对接 |
||||
getphonenumber(res) { |
||||
this.$emit("getphonenumber", res); |
||||
}, |
||||
getuserinfo(res) { |
||||
this.$emit("getuserinfo", res); |
||||
}, |
||||
error(res) { |
||||
this.$emit("error", res); |
||||
}, |
||||
opensetting(res) { |
||||
this.$emit("opensetting", res); |
||||
}, |
||||
launchapp(res) { |
||||
this.$emit("launchapp", res); |
||||
}, |
||||
}, |
||||
}; |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
/* #ifndef APP-NVUE */ |
||||
@import "./vue.scss"; |
||||
/* #endif */ |
||||
|
||||
/* #ifdef APP-NVUE */ |
||||
@import "./nvue.scss"; |
||||
/* #endif */ |
||||
|
||||
$u-button-u-button-height: 40px !default; |
||||
$u-button-text-font-size: 15px !default; |
||||
$u-button-loading-text-font-size: 15px !default; |
||||
$u-button-loading-text-margin-left: 4px !default; |
||||
$u-button-large-width: 100% !default; |
||||
$u-button-large-height: 50px !default; |
||||
$u-button-normal-padding: 0 12px !default; |
||||
$u-button-large-padding: 0 15px !default; |
||||
$u-button-normal-font-size: 14px !default; |
||||
$u-button-small-min-width: 60px !default; |
||||
$u-button-small-height: 30px !default; |
||||
$u-button-small-padding: 0px 8px !default; |
||||
$u-button-mini-padding: 0px 8px !default; |
||||
$u-button-small-font-size: 12px !default; |
||||
$u-button-mini-height: 22px !default; |
||||
$u-button-mini-font-size: 10px !default; |
||||
$u-button-mini-min-width: 50px !default; |
||||
$u-button-disabled-opacity: 0.5 !default; |
||||
$u-button-info-color: #323233 !default; |
||||
$u-button-info-background-color: #fff !default; |
||||
$u-button-info-border-color: #ebedf0 !default; |
||||
$u-button-info-border-width: 1px !default; |
||||
$u-button-info-border-style: solid !default; |
||||
$u-button-success-color: #fff !default; |
||||
$u-button-success-background-color: $u-success !default; |
||||
$u-button-success-border-color: $u-button-success-background-color !default; |
||||
$u-button-success-border-width: 1px !default; |
||||
$u-button-success-border-style: solid !default; |
||||
$u-button-primary-color: #fff !default; |
||||
$u-button-primary-background-color: $u-primary !default; |
||||
$u-button-primary-border-color: $u-button-primary-background-color !default; |
||||
$u-button-primary-border-width: 1px !default; |
||||
$u-button-primary-border-style: solid !default; |
||||
$u-button-error-color: #fff !default; |
||||
$u-button-error-background-color: $u-error !default; |
||||
$u-button-error-border-color: $u-button-error-background-color !default; |
||||
$u-button-error-border-width: 1px !default; |
||||
$u-button-error-border-style: solid !default; |
||||
$u-button-warning-color: #fff !default; |
||||
$u-button-warning-background-color: $u-warning !default; |
||||
$u-button-warning-border-color: $u-button-warning-background-color !default; |
||||
$u-button-warning-border-width: 1px !default; |
||||
$u-button-warning-border-style: solid !default; |
||||
$u-button-block-width: 100% !default; |
||||
$u-button-circle-border-top-right-radius: 100px !default; |
||||
$u-button-circle-border-top-left-radius: 100px !default; |
||||
$u-button-circle-border-bottom-left-radius: 100px !default; |
||||
$u-button-circle-border-bottom-right-radius: 100px !default; |
||||
$u-button-square-border-top-right-radius: 3px !default; |
||||
$u-button-square-border-top-left-radius: 3px !default; |
||||
$u-button-square-border-bottom-left-radius: 3px !default; |
||||
$u-button-square-border-bottom-right-radius: 3px !default; |
||||
$u-button-icon-min-width: 1em !default; |
||||
$u-button-plain-background-color: #fff !default; |
||||
$u-button-hairline-border-width: 0.5px !default; |
||||
|
||||
.u-button { |
||||
height: $u-button-u-button-height; |
||||
position: relative; |
||||
align-items: center; |
||||
justify-content: center; |
||||
@include flex; |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
|
||||
&__text { |
||||
font-size: $u-button-text-font-size; |
||||
} |
||||
|
||||
&__loading-text { |
||||
font-size: $u-button-loading-text-font-size; |
||||
margin-left: $u-button-loading-text-margin-left; |
||||
} |
||||
|
||||
&--large { |
||||
/* #ifndef APP-NVUE */ |
||||
width: $u-button-large-width; |
||||
/* #endif */ |
||||
height: $u-button-large-height; |
||||
padding: $u-button-large-padding; |
||||
} |
||||
|
||||
&--normal { |
||||
padding: $u-button-normal-padding; |
||||
font-size: $u-button-normal-font-size; |
||||
} |
||||
|
||||
&--small { |
||||
/* #ifndef APP-NVUE */ |
||||
min-width: $u-button-small-min-width; |
||||
/* #endif */ |
||||
height: $u-button-small-height; |
||||
padding: $u-button-small-padding; |
||||
font-size: $u-button-small-font-size; |
||||
} |
||||
|
||||
&--mini { |
||||
height: $u-button-mini-height; |
||||
font-size: $u-button-mini-font-size; |
||||
/* #ifndef APP-NVUE */ |
||||
min-width: $u-button-mini-min-width; |
||||
/* #endif */ |
||||
padding: $u-button-mini-padding; |
||||
} |
||||
|
||||
&--disabled { |
||||
opacity: $u-button-disabled-opacity; |
||||
} |
||||
|
||||
&--info { |
||||
color: $u-button-info-color; |
||||
background-color: $u-button-info-background-color; |
||||
border-color: $u-button-info-border-color; |
||||
border-width: $u-button-info-border-width; |
||||
border-style: $u-button-info-border-style; |
||||
} |
||||
|
||||
&--success { |
||||
color: $u-button-success-color; |
||||
background-color: $u-button-success-background-color; |
||||
border-color: $u-button-success-border-color; |
||||
border-width: $u-button-success-border-width; |
||||
border-style: $u-button-success-border-style; |
||||
} |
||||
|
||||
&--primary { |
||||
color: $u-button-primary-color; |
||||
background-color: $u-button-primary-background-color; |
||||
border-color: $u-button-primary-border-color; |
||||
border-width: $u-button-primary-border-width; |
||||
border-style: $u-button-primary-border-style; |
||||
} |
||||
|
||||
&--error { |
||||
color: $u-button-error-color; |
||||
background-color: $u-button-error-background-color; |
||||
border-color: $u-button-error-border-color; |
||||
border-width: $u-button-error-border-width; |
||||
border-style: $u-button-error-border-style; |
||||
} |
||||
|
||||
&--warning { |
||||
color: $u-button-warning-color; |
||||
background-color: $u-button-warning-background-color; |
||||
border-color: $u-button-warning-border-color; |
||||
border-width: $u-button-warning-border-width; |
||||
border-style: $u-button-warning-border-style; |
||||
} |
||||
|
||||
&--block { |
||||
@include flex; |
||||
width: $u-button-block-width; |
||||
} |
||||
|
||||
&--circle { |
||||
border-top-right-radius: $u-button-circle-border-top-right-radius; |
||||
border-top-left-radius: $u-button-circle-border-top-left-radius; |
||||
border-bottom-left-radius: $u-button-circle-border-bottom-left-radius; |
||||
border-bottom-right-radius: $u-button-circle-border-bottom-right-radius; |
||||
} |
||||
|
||||
&--square { |
||||
border-bottom-left-radius: $u-button-square-border-top-right-radius; |
||||
border-bottom-right-radius: $u-button-square-border-top-left-radius; |
||||
border-top-left-radius: $u-button-square-border-bottom-left-radius; |
||||
border-top-right-radius: $u-button-square-border-bottom-right-radius; |
||||
} |
||||
|
||||
&__icon { |
||||
/* #ifndef APP-NVUE */ |
||||
min-width: $u-button-icon-min-width; |
||||
line-height: inherit !important; |
||||
vertical-align: top; |
||||
/* #endif */ |
||||
} |
||||
|
||||
&--plain { |
||||
background-color: $u-button-plain-background-color; |
||||
} |
||||
|
||||
&--hairline { |
||||
border-width: $u-button-hairline-border-width !important; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,80 @@ |
||||
// nvue下hover-class无效 |
||||
$u-button-before-top:50% !default; |
||||
$u-button-before-left:50% !default; |
||||
$u-button-before-width:100% !default; |
||||
$u-button-before-height:100% !default; |
||||
$u-button-before-transform:translate(-50%, -50%) !default; |
||||
$u-button-before-opacity:0 !default; |
||||
$u-button-before-background-color:#000 !default; |
||||
$u-button-before-border-color:#000 !default; |
||||
$u-button-active-before-opacity:.15 !default; |
||||
$u-button-icon-margin-left:4px !default; |
||||
$u-button-plain-u-button-info-color:$u-info; |
||||
$u-button-plain-u-button-success-color:$u-success; |
||||
$u-button-plain-u-button-error-color:$u-error; |
||||
$u-button-plain-u-button-warning-color:$u-error; |
||||
|
||||
.u-button { |
||||
width: 100%; |
||||
|
||||
&__text { |
||||
white-space: nowrap; |
||||
line-height: 1; |
||||
} |
||||
|
||||
&:before { |
||||
position: absolute; |
||||
top:$u-button-before-top; |
||||
left:$u-button-before-left; |
||||
width:$u-button-before-width; |
||||
height:$u-button-before-height; |
||||
border: inherit; |
||||
border-radius: inherit; |
||||
transform:$u-button-before-transform; |
||||
opacity:$u-button-before-opacity; |
||||
content: " "; |
||||
background-color:$u-button-before-background-color; |
||||
border-color:$u-button-before-border-color; |
||||
} |
||||
|
||||
&--active { |
||||
&:before { |
||||
opacity: .15 |
||||
} |
||||
} |
||||
|
||||
&__icon+&__text:not(:empty), |
||||
&__loading-text { |
||||
margin-left:$u-button-icon-margin-left; |
||||
} |
||||
|
||||
&--plain { |
||||
&.u-button--primary { |
||||
color: $u-primary; |
||||
} |
||||
} |
||||
|
||||
&--plain { |
||||
&.u-button--info { |
||||
color:$u-button-plain-u-button-info-color; |
||||
} |
||||
} |
||||
|
||||
&--plain { |
||||
&.u-button--success { |
||||
color:$u-button-plain-u-button-success-color; |
||||
} |
||||
} |
||||
|
||||
&--plain { |
||||
&.u-button--error { |
||||
color:$u-button-plain-u-button-error-color; |
||||
} |
||||
} |
||||
|
||||
&--plain { |
||||
&.u-button--warning { |
||||
color:$u-button-plain-u-button-warning-color; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,99 @@ |
||||
<template> |
||||
<view class="u-calendar-header u-border-bottom"> |
||||
<text |
||||
class="u-calendar-header__title" |
||||
v-if="showTitle" |
||||
>{{ title }}</text> |
||||
<text |
||||
class="u-calendar-header__subtitle" |
||||
v-if="showSubtitle" |
||||
>{{ subtitle }}</text> |
||||
<view class="u-calendar-header__weekdays"> |
||||
<text class="u-calendar-header__weekdays__weekday">一</text> |
||||
<text class="u-calendar-header__weekdays__weekday">二</text> |
||||
<text class="u-calendar-header__weekdays__weekday">三</text> |
||||
<text class="u-calendar-header__weekdays__weekday">四</text> |
||||
<text class="u-calendar-header__weekdays__weekday">五</text> |
||||
<text class="u-calendar-header__weekdays__weekday">六</text> |
||||
<text class="u-calendar-header__weekdays__weekday">日</text> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: 'u-calendar-header', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin], |
||||
props: { |
||||
// 标题 |
||||
title: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
// 副标题 |
||||
subtitle: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
// 是否显示标题 |
||||
showTitle: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 是否显示副标题 |
||||
showSubtitle: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
}, |
||||
data() { |
||||
return { |
||||
|
||||
} |
||||
}, |
||||
methods: { |
||||
name() { |
||||
|
||||
} |
||||
}, |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
.u-calendar-header { |
||||
padding-bottom: 4px; |
||||
|
||||
&__title { |
||||
font-size: 16px; |
||||
color: $u-main-color; |
||||
text-align: center; |
||||
height: 42px; |
||||
line-height: 42px; |
||||
font-weight: bold; |
||||
} |
||||
|
||||
&__subtitle { |
||||
font-size: 14px; |
||||
color: $u-main-color; |
||||
height: 40px; |
||||
text-align: center; |
||||
line-height: 40px; |
||||
font-weight: bold; |
||||
} |
||||
|
||||
&__weekdays { |
||||
@include flex; |
||||
justify-content: space-between; |
||||
|
||||
&__weekday { |
||||
font-size: 13px; |
||||
color: $u-main-color; |
||||
line-height: 30px; |
||||
flex: 1; |
||||
text-align: center; |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,579 @@ |
||||
<template> |
||||
<view class="u-calendar-month-wrapper" ref="u-calendar-month-wrapper"> |
||||
<view v-for="(item, index) in months" :key="index" :class="[`u-calendar-month-${index}`]" |
||||
:ref="`u-calendar-month-${index}`" :id="`month-${index}`"> |
||||
<text v-if="index !== 0" class="u-calendar-month__title">{{ item.year }}年{{ item.month }}月</text> |
||||
<view class="u-calendar-month__days"> |
||||
<view v-if="showMark" class="u-calendar-month__days__month-mark-wrapper"> |
||||
<text class="u-calendar-month__days__month-mark-wrapper__text">{{ item.month }}</text> |
||||
</view> |
||||
<view class="u-calendar-month__days__day" v-for="(item1, index1) in item.date" :key="index1" |
||||
:style="[dayStyle(index, index1, item1)]" @tap="clickHandler(index, index1, item1)" |
||||
:class="[item1.selected && 'u-calendar-month__days__day__select--selected']"> |
||||
<view class="u-calendar-month__days__day__select" :style="[daySelectStyle(index, index1, item1)]"> |
||||
<text class="u-calendar-month__days__day__select__info" |
||||
:class="[item1.disabled && 'u-calendar-month__days__day__select__info--disabled']" |
||||
:style="[textStyle(item1)]">{{ item1.day }}</text> |
||||
<text v-if="getBottomInfo(index, index1, item1)" |
||||
class="u-calendar-month__days__day__select__buttom-info" |
||||
:class="[item1.disabled && 'u-calendar-month__days__day__select__buttom-info--disabled']" |
||||
:style="[textStyle(item1)]">{{ getBottomInfo(index, index1, item1) }}</text> |
||||
<text v-if="item1.dot" class="u-calendar-month__days__day__select__dot"></text> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
// #ifdef APP-NVUE |
||||
// 由于nvue不支持百分比单位,需要查询宽度来计算每个日期的宽度 |
||||
const dom = uni.requireNativePlugin('dom') |
||||
// #endif |
||||
import dayjs from '../../libs/util/dayjs.js'; |
||||
export default { |
||||
name: 'u-calendar-month', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin], |
||||
props: { |
||||
// 是否显示月份背景色 |
||||
showMark: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 主题色,对底部按钮和选中日期有效 |
||||
color: { |
||||
type: String, |
||||
default: '#3c9cff' |
||||
}, |
||||
// 月份数据 |
||||
months: { |
||||
type: Array, |
||||
default: () => [] |
||||
}, |
||||
// 日期选择类型 |
||||
mode: { |
||||
type: String, |
||||
default: 'single' |
||||
}, |
||||
// 日期行高 |
||||
rowHeight: { |
||||
type: [String, Number], |
||||
default: 58 |
||||
}, |
||||
// mode=multiple时,最多可选多少个日期 |
||||
maxCount: { |
||||
type: [String, Number], |
||||
default: Infinity |
||||
}, |
||||
// mode=range时,第一个日期底部的提示文字 |
||||
startText: { |
||||
type: String, |
||||
default: '开始' |
||||
}, |
||||
// mode=range时,最后一个日期底部的提示文字 |
||||
endText: { |
||||
type: String, |
||||
default: '结束' |
||||
}, |
||||
// 默认选中的日期,mode为multiple或range是必须为数组格式 |
||||
defaultDate: { |
||||
type: [Array, String, Date], |
||||
default: null |
||||
}, |
||||
// 最小的可选日期 |
||||
minDate: { |
||||
type: [String, Number], |
||||
default: 0 |
||||
}, |
||||
// 最大可选日期 |
||||
maxDate: { |
||||
type: [String, Number], |
||||
default: 0 |
||||
}, |
||||
// 如果没有设置maxDate,则往后推多少个月 |
||||
maxMonth: { |
||||
type: [String, Number], |
||||
default: 2 |
||||
}, |
||||
// 是否为只读状态,只读状态下禁止选择日期 |
||||
readonly: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.readonly |
||||
}, |
||||
// 日期区间最多可选天数,默认无限制,mode = range时有效 |
||||
maxRange: { |
||||
type: [Number, String], |
||||
default: Infinity |
||||
}, |
||||
// 范围选择超过最多可选天数时的提示文案,mode = range时有效 |
||||
rangePrompt: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
// 范围选择超过最多可选天数时,是否展示提示文案,mode = range时有效 |
||||
showRangePrompt: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 是否允许日期范围的起止时间为同一天,mode = range时有效 |
||||
allowSameDay: { |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
// 每个日期的宽度 |
||||
width: 0, |
||||
// 当前选中的日期item |
||||
item: {}, |
||||
selected: [] |
||||
} |
||||
}, |
||||
watch: { |
||||
selectedChange: { |
||||
immediate: true, |
||||
handler(n) { |
||||
this.setDefaultDate() |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
// 多个条件的变化,会引起选中日期的变化,这里统一管理监听 |
||||
selectedChange() { |
||||
return [this.minDate, this.maxDate, this.defaultDate] |
||||
}, |
||||
dayStyle(index1, index2, item) { |
||||
return (index1, index2, item) => { |
||||
const style = {} |
||||
let week = item.week |
||||
// 不进行四舍五入的形式保留2位小数 |
||||
const dayWidth = Number(parseFloat(this.width / 7).toFixed(3).slice(0, -1)) |
||||
// 得出每个日期的宽度 |
||||
// #ifdef APP-NVUE |
||||
style.width = uni.$u.addUnit(dayWidth) |
||||
// #endif |
||||
style.height = uni.$u.addUnit(this.rowHeight) |
||||
if (index2 === 0) { |
||||
// 获取当前为星期几,如果为0,则为星期天,减一为每月第一天时,需要向左偏移的item个数 |
||||
week = (week === 0 ? 7 : week) - 1 |
||||
style.marginLeft = uni.$u.addUnit(week * dayWidth) |
||||
} |
||||
if (this.mode === 'range') { |
||||
// 之所以需要这么写,是因为DCloud公司的iOS客户端的开发者能力有限导致的bug |
||||
style.paddingLeft = 0 |
||||
style.paddingRight = 0 |
||||
style.paddingBottom = 0 |
||||
style.paddingTop = 0 |
||||
} |
||||
return style |
||||
} |
||||
}, |
||||
daySelectStyle() { |
||||
return (index1, index2, item) => { |
||||
let date = dayjs(item.date).format("YYYY-MM-DD"), |
||||
style = {} |
||||
// 判断date是否在selected数组中,因为月份可能会需要补0,所以使用dateSame判断,而不用数组的includes判断 |
||||
if (this.selected.some(item => this.dateSame(item, date))) { |
||||
style.backgroundColor = this.color |
||||
} |
||||
if (this.mode === 'single') { |
||||
if (date === this.selected[0]) { |
||||
// 因为需要对nvue的兼容,只能这么写,无法缩写,也无法通过类名控制等等 |
||||
style.borderTopLeftRadius = '3px' |
||||
style.borderBottomLeftRadius = '3px' |
||||
style.borderTopRightRadius = '3px' |
||||
style.borderBottomRightRadius = '3px' |
||||
} |
||||
} else if (this.mode === 'range') { |
||||
if (this.selected.length >= 2) { |
||||
const len = this.selected.length - 1 |
||||
// 第一个日期设置左上角和左下角的圆角 |
||||
if (this.dateSame(date, this.selected[0])) { |
||||
style.borderTopLeftRadius = '3px' |
||||
style.borderBottomLeftRadius = '3px' |
||||
} |
||||
// 最后一个日期设置右上角和右下角的圆角 |
||||
if (this.dateSame(date, this.selected[len])) { |
||||
style.borderTopRightRadius = '3px' |
||||
style.borderBottomRightRadius = '3px' |
||||
} |
||||
// 处于第一和最后一个之间的日期,背景色设置为浅色,通过将对应颜色进行等分,再取其尾部的颜色值 |
||||
if (dayjs(date).isAfter(dayjs(this.selected[0])) && dayjs(date).isBefore(dayjs(this |
||||
.selected[len]))) { |
||||
style.backgroundColor = uni.$u.colorGradient(this.color, '#ffffff', 100)[90] |
||||
// 增加一个透明度,让范围区间的背景色也能看到底部的mark水印字符 |
||||
style.opacity = 0.7 |
||||
} |
||||
} else if (this.selected.length === 1) { |
||||
// 之所以需要这么写,是因为DCloud公司的iOS客户端的开发者能力有限导致的bug |
||||
// 进行还原操作,否则在nvue的iOS,uni-app有bug,会导致诡异的表现 |
||||
style.borderTopLeftRadius = '3px' |
||||
style.borderBottomLeftRadius = '3px' |
||||
} |
||||
} else { |
||||
if (this.selected.some(item => this.dateSame(item, date))) { |
||||
style.borderTopLeftRadius = '3px' |
||||
style.borderBottomLeftRadius = '3px' |
||||
style.borderTopRightRadius = '3px' |
||||
style.borderBottomRightRadius = '3px' |
||||
} |
||||
} |
||||
return style |
||||
} |
||||
}, |
||||
// 某个日期是否被选中 |
||||
textStyle() { |
||||
return (item) => { |
||||
const date = dayjs(item.date).format("YYYY-MM-DD"), |
||||
style = {} |
||||
// 选中的日期,提示文字设置白色 |
||||
if (this.selected.some(item => this.dateSame(item, date))) { |
||||
style.color = '#ffffff' |
||||
} |
||||
if (this.mode === 'range') { |
||||
const len = this.selected.length - 1 |
||||
// 如果是范围选择模式,第一个和最后一个之间的日期,文字颜色设置为高亮的主题色 |
||||
if (dayjs(date).isAfter(dayjs(this.selected[0])) && dayjs(date).isBefore(dayjs(this |
||||
.selected[len]))) { |
||||
style.color = this.color |
||||
} |
||||
} |
||||
return style |
||||
} |
||||
}, |
||||
// 获取底部的提示文字 |
||||
getBottomInfo() { |
||||
return (index1, index2, item) => { |
||||
const date = dayjs(item.date).format("YYYY-MM-DD") |
||||
const bottomInfo = item.bottomInfo |
||||
// 当为日期范围模式时,且选择的日期个数大于0时 |
||||
if (this.mode === 'range' && this.selected.length > 0) { |
||||
if (this.selected.length === 1) { |
||||
// 选择了一个日期时,如果当前日期为数组中的第一个日期,则显示底部文字为“开始” |
||||
if (this.dateSame(date, this.selected[0])) return this.startText |
||||
else return bottomInfo |
||||
} else { |
||||
const len = this.selected.length - 1 |
||||
// 如果数组中的日期大于2个时,第一个和最后一个显示为开始和结束日期 |
||||
if (this.dateSame(date, this.selected[0]) && this.dateSame(date, this.selected[1]) && |
||||
len === 1) { |
||||
// 如果长度为2,且第一个等于第二个日期,则提示语放在同一个item中 |
||||
return `${this.startText}/${this.endText}` |
||||
} else if (this.dateSame(date, this.selected[0])) { |
||||
return this.startText |
||||
} else if (this.dateSame(date, this.selected[len])) { |
||||
return this.endText |
||||
} else { |
||||
return bottomInfo |
||||
} |
||||
} |
||||
} else { |
||||
return bottomInfo |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.init() |
||||
}, |
||||
methods: { |
||||
init() { |
||||
// 初始化默认选中 |
||||
this.$emit('monthSelected', this.selected) |
||||
this.$nextTick(() => { |
||||
// 这里需要另一个延时,因为获取宽度后,会进行月份数据渲染,只有渲染完成之后,才有真正的高度 |
||||
// 因为nvue下,$nextTick并不是100%可靠的 |
||||
uni.$u.sleep(10).then(() => { |
||||
this.getWrapperWidth() |
||||
this.getMonthRect() |
||||
}) |
||||
}) |
||||
}, |
||||
// 判断两个日期是否相等 |
||||
dateSame(date1, date2) { |
||||
return dayjs(date1).isSame(dayjs(date2)) |
||||
}, |
||||
// 获取月份数据区域的宽度,因为nvue不支持百分比,所以无法通过css设置每个日期item的宽度 |
||||
getWrapperWidth() { |
||||
// #ifdef APP-NVUE |
||||
dom.getComponentRect(this.$refs['u-calendar-month-wrapper'], res => { |
||||
this.width = res.size.width |
||||
}) |
||||
// #endif |
||||
// #ifndef APP-NVUE |
||||
this.$uGetRect('.u-calendar-month-wrapper').then(size => { |
||||
this.width = size.width |
||||
}) |
||||
// #endif |
||||
}, |
||||
getMonthRect() { |
||||
// 获取每个月份数据的尺寸,用于父组件在scroll-view滚动事件中,监听当前滚动到了第几个月份 |
||||
const promiseAllArr = this.months.map((item, index) => this.getMonthRectByPromise( |
||||
`u-calendar-month-${index}`)) |
||||
// 一次性返回 |
||||
Promise.all(promiseAllArr).then( |
||||
sizes => { |
||||
let height = 1 |
||||
const topArr = [] |
||||
for (let i = 0; i < this.months.length; i++) { |
||||
// 添加到months数组中,供scroll-view滚动事件中,判断当前滚动到哪个月份 |
||||
topArr[i] = height |
||||
height += sizes[i].height |
||||
} |
||||
// 由于微信下,无法通过this.months[i].top的形式(引用类型)去修改父组件的month的top值,所以使用事件形式对外发出 |
||||
this.$emit('updateMonthTop', topArr) |
||||
}) |
||||
}, |
||||
// 获取每个月份区域的尺寸 |
||||
getMonthRectByPromise(el) { |
||||
// #ifndef APP-NVUE |
||||
// $uGetRect为uView自带的节点查询简化方法,详见文档介绍:https://www.uviewui.com/js/getRect.html |
||||
// 组件内部一般用this.$uGetRect,对外的为uni.$u.getRect,二者功能一致,名称不同 |
||||
return new Promise(resolve => { |
||||
this.$uGetRect(`.${el}`).then(size => { |
||||
resolve(size) |
||||
}) |
||||
}) |
||||
// #endif |
||||
|
||||
// #ifdef APP-NVUE |
||||
// nvue下,使用dom模块查询元素高度 |
||||
// 返回一个promise,让调用此方法的主体能使用then回调 |
||||
return new Promise(resolve => { |
||||
dom.getComponentRect(this.$refs[el][0], res => { |
||||
resolve(res.size) |
||||
}) |
||||
}) |
||||
// #endif |
||||
}, |
||||
// 点击某一个日期 |
||||
clickHandler(index1, index2, item) { |
||||
if (this.readonly) { |
||||
return; |
||||
} |
||||
this.item = item |
||||
const date = dayjs(item.date).format("YYYY-MM-DD") |
||||
if (item.disabled) return |
||||
// 对上一次选择的日期数组进行深度克隆 |
||||
let selected = uni.$u.deepClone(this.selected) |
||||
if (this.mode === 'single') { |
||||
// 单选情况下,让数组中的元素为当前点击的日期 |
||||
selected = [date] |
||||
} else if (this.mode === 'multiple') { |
||||
if (selected.some(item => this.dateSame(item, date))) { |
||||
// 如果点击的日期已在数组中,则进行移除操作,也就是达到反选的效果 |
||||
const itemIndex = selected.findIndex(item => item === date) |
||||
selected.splice(itemIndex, 1) |
||||
} else { |
||||
// 如果点击的日期不在数组中,且已有的长度小于总可选长度时,则添加到数组中去 |
||||
if (selected.length < this.maxCount) selected.push(date) |
||||
} |
||||
} else { |
||||
// 选择区间形式 |
||||
if (selected.length === 0 || selected.length >= 2) { |
||||
// 如果原来就为0或者大于2的长度,则当前点击的日期,就是开始日期 |
||||
selected = [date] |
||||
} else if (selected.length === 1) { |
||||
// 如果已经选择了开始日期 |
||||
const existsDate = selected[0] |
||||
// 如果当前选择的日期小于上一次选择的日期,则当前的日期定为开始日期 |
||||
if (dayjs(date).isBefore(existsDate)) { |
||||
selected = [date] |
||||
} else if (dayjs(date).isAfter(existsDate)) { |
||||
// 当前日期减去最大可选的日期天数,如果大于起始时间,则进行提示 |
||||
if(dayjs(dayjs(date).subtract(this.maxRange, 'day')).isAfter(dayjs(selected[0])) && this.showRangePrompt) { |
||||
if(this.rangePrompt) { |
||||
uni.$u.toast(this.rangePrompt) |
||||
} else { |
||||
uni.$u.toast(`选择天数不能超过 ${this.maxRange} 天`) |
||||
} |
||||
return |
||||
} |
||||
// 如果当前日期大于已有日期,将当前的添加到数组尾部 |
||||
selected.push(date) |
||||
const startDate = selected[0] |
||||
const endDate = selected[1] |
||||
const arr = [] |
||||
let i = 0 |
||||
do { |
||||
// 将开始和结束日期之间的日期添加到数组中 |
||||
arr.push(dayjs(startDate).add(i, 'day').format("YYYY-MM-DD")) |
||||
i++ |
||||
// 累加的日期小于结束日期时,继续下一次的循环 |
||||
} while (dayjs(startDate).add(i, 'day').isBefore(dayjs(endDate))) |
||||
// 为了一次性修改数组,避免computed中多次触发,这里才用arr变量一次性赋值的方式,同时将最后一个日期添加近来 |
||||
arr.push(endDate) |
||||
selected = arr |
||||
} else { |
||||
// 选择区间时,只有一个日期的情况下,且不允许选择起止为同一天的话,不允许选择自己 |
||||
if (selected[0] === date && !this.allowSameDay) return |
||||
selected.push(date) |
||||
} |
||||
} |
||||
} |
||||
this.setSelected(selected) |
||||
}, |
||||
// 设置默认日期 |
||||
setDefaultDate() { |
||||
if (!this.defaultDate) { |
||||
// 如果没有设置默认日期,则将当天日期设置为默认选中的日期 |
||||
const selected = [dayjs().format("YYYY-MM-DD")] |
||||
return this.setSelected(selected, false) |
||||
} |
||||
let defaultDate = [] |
||||
const minDate = this.minDate || dayjs().format("YYYY-MM-DD") |
||||
const maxDate = this.maxDate || dayjs(minDate).add(this.maxMonth - 1, 'month').format("YYYY-MM-DD") |
||||
if (this.mode === 'single') { |
||||
// 单选模式,可以是字符串或数组,Date对象等 |
||||
if (!uni.$u.test.array(this.defaultDate)) { |
||||
defaultDate = [dayjs(this.defaultDate).format("YYYY-MM-DD")] |
||||
} else { |
||||
defaultDate = [this.defaultDate[0]] |
||||
} |
||||
} else { |
||||
// 如果为非数组,则不执行 |
||||
if (!uni.$u.test.array(this.defaultDate)) return |
||||
defaultDate = this.defaultDate |
||||
} |
||||
// 过滤用户传递的默认数组,取出只在可允许最大值与最小值之间的元素 |
||||
defaultDate = defaultDate.filter(item => { |
||||
return dayjs(item).isAfter(dayjs(minDate).subtract(1, 'day')) && dayjs(item).isBefore(dayjs( |
||||
maxDate).add(1, 'day')) |
||||
}) |
||||
this.setSelected(defaultDate, false) |
||||
}, |
||||
setSelected(selected, event = true) { |
||||
this.selected = selected |
||||
event && this.$emit('monthSelected', this.selected) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import "../../libs/css/components.scss"; |
||||
|
||||
.u-calendar-month-wrapper { |
||||
margin-top: 4px; |
||||
} |
||||
|
||||
.u-calendar-month { |
||||
|
||||
&__title { |
||||
font-size: 14px; |
||||
line-height: 42px; |
||||
height: 42px; |
||||
color: $u-main-color; |
||||
text-align: center; |
||||
font-weight: bold; |
||||
} |
||||
|
||||
&__days { |
||||
position: relative; |
||||
@include flex; |
||||
flex-wrap: wrap; |
||||
|
||||
&__month-mark-wrapper { |
||||
position: absolute; |
||||
top: 0; |
||||
bottom: 0; |
||||
left: 0; |
||||
right: 0; |
||||
@include flex; |
||||
justify-content: center; |
||||
align-items: center; |
||||
|
||||
&__text { |
||||
font-size: 155px; |
||||
color: rgba(231, 232, 234, 0.83); |
||||
} |
||||
} |
||||
|
||||
&__day { |
||||
@include flex; |
||||
padding: 2px; |
||||
/* #ifndef APP-NVUE */ |
||||
// vue下使用css进行宽度计算,因为某些安卓机会无法进行js获取父元素宽度进行计算得出,会有偏移 |
||||
width: calc(100% / 7); |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
|
||||
&__select { |
||||
flex: 1; |
||||
@include flex; |
||||
align-items: center; |
||||
justify-content: center; |
||||
position: relative; |
||||
|
||||
&__dot { |
||||
width: 7px; |
||||
height: 7px; |
||||
border-radius: 100px; |
||||
background-color: $u-error; |
||||
position: absolute; |
||||
top: 12px; |
||||
right: 7px; |
||||
} |
||||
|
||||
&__buttom-info { |
||||
color: $u-content-color; |
||||
text-align: center; |
||||
position: absolute; |
||||
bottom: 5px; |
||||
font-size: 10px; |
||||
text-align: center; |
||||
left: 0; |
||||
right: 0; |
||||
|
||||
&--selected { |
||||
color: #ffffff; |
||||
} |
||||
|
||||
&--disabled { |
||||
color: #cacbcd; |
||||
} |
||||
} |
||||
|
||||
&__info { |
||||
text-align: center; |
||||
font-size: 16px; |
||||
|
||||
&--selected { |
||||
color: #ffffff; |
||||
} |
||||
|
||||
&--disabled { |
||||
color: #cacbcd; |
||||
} |
||||
} |
||||
|
||||
&--selected { |
||||
background-color: $u-primary; |
||||
@include flex; |
||||
justify-content: center; |
||||
align-items: center; |
||||
flex: 1; |
||||
border-radius: 3px; |
||||
} |
||||
|
||||
&--range-selected { |
||||
opacity: 0.3; |
||||
border-radius: 0; |
||||
} |
||||
|
||||
&--range-start-selected { |
||||
border-top-right-radius: 0; |
||||
border-bottom-right-radius: 0; |
||||
} |
||||
|
||||
&--range-end-selected { |
||||
border-top-left-radius: 0; |
||||
border-bottom-left-radius: 0; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,144 @@ |
||||
export default { |
||||
props: { |
||||
// 日历顶部标题
|
||||
title: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.title |
||||
}, |
||||
// 是否显示标题
|
||||
showTitle: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showTitle |
||||
}, |
||||
// 是否显示副标题
|
||||
showSubtitle: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showSubtitle |
||||
}, |
||||
// 日期类型选择,single-选择单个日期,multiple-可以选择多个日期,range-选择日期范围
|
||||
mode: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.mode |
||||
}, |
||||
// mode=range时,第一个日期底部的提示文字
|
||||
startText: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.startText |
||||
}, |
||||
// mode=range时,最后一个日期底部的提示文字
|
||||
endText: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.endText |
||||
}, |
||||
// 自定义列表
|
||||
customList: { |
||||
type: Array, |
||||
default: uni.$u.props.calendar.customList |
||||
}, |
||||
// 主题色,对底部按钮和选中日期有效
|
||||
color: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.color |
||||
}, |
||||
// 最小的可选日期
|
||||
minDate: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.calendar.minDate |
||||
}, |
||||
// 最大可选日期
|
||||
maxDate: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.calendar.maxDate |
||||
}, |
||||
// 默认选中的日期,mode为multiple或range是必须为数组格式
|
||||
defaultDate: { |
||||
type: [Array, String, Date, null], |
||||
default: uni.$u.props.calendar.defaultDate |
||||
}, |
||||
// mode=multiple时,最多可选多少个日期
|
||||
maxCount: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.calendar.maxCount |
||||
}, |
||||
// 日期行高
|
||||
rowHeight: { |
||||
type: [String, Number], |
||||
default: uni.$u.props.calendar.rowHeight |
||||
}, |
||||
// 日期格式化函数
|
||||
formatter: { |
||||
type: [Function, null], |
||||
default: uni.$u.props.calendar.formatter |
||||
}, |
||||
// 是否显示农历
|
||||
showLunar: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showLunar |
||||
}, |
||||
// 是否显示月份背景色
|
||||
showMark: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showMark |
||||
}, |
||||
// 确定按钮的文字
|
||||
confirmText: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.confirmText |
||||
}, |
||||
// 确认按钮处于禁用状态时的文字
|
||||
confirmDisabledText: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.confirmDisabledText |
||||
}, |
||||
// 是否显示日历弹窗
|
||||
show: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.show |
||||
}, |
||||
// 是否允许点击遮罩关闭日历
|
||||
closeOnClickOverlay: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.closeOnClickOverlay |
||||
}, |
||||
// 是否为只读状态,只读状态下禁止选择日期
|
||||
readonly: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.readonly |
||||
}, |
||||
// 是否展示确认按钮
|
||||
showConfirm: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showConfirm |
||||
}, |
||||
// 日期区间最多可选天数,默认无限制,mode = range时有效
|
||||
maxRange: { |
||||
type: [Number, String], |
||||
default: uni.$u.props.calendar.maxRange |
||||
}, |
||||
// 范围选择超过最多可选天数时的提示文案,mode = range时有效
|
||||
rangePrompt: { |
||||
type: String, |
||||
default: uni.$u.props.calendar.rangePrompt |
||||
}, |
||||
// 范围选择超过最多可选天数时,是否展示提示文案,mode = range时有效
|
||||
showRangePrompt: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.showRangePrompt |
||||
}, |
||||
// 是否允许日期范围的起止时间为同一天,mode = range时有效
|
||||
allowSameDay: { |
||||
type: Boolean, |
||||
default: uni.$u.props.calendar.allowSameDay |
||||
}, |
||||
// 圆角值
|
||||
round: { |
||||
type: [Boolean, String, Number], |
||||
default: uni.$u.props.calendar.round |
||||
}, |
||||
// 最多展示月份数量
|
||||
monthNum: { |
||||
type: [Number, String], |
||||
default: 3 |
||||
}
|
||||
} |
||||
} |
@ -0,0 +1,383 @@ |
||||
<template> |
||||
<u-popup |
||||
:show="show" |
||||
mode="bottom" |
||||
closeable |
||||
@close="close" |
||||
:round="round" |
||||
:closeOnClickOverlay="closeOnClickOverlay" |
||||
> |
||||
<view class="u-calendar"> |
||||
<uHeader |
||||
:title="title" |
||||
:subtitle="subtitle" |
||||
:showSubtitle="showSubtitle" |
||||
:showTitle="showTitle" |
||||
></uHeader> |
||||
<scroll-view |
||||
:style="{ |
||||
height: $u.addUnit(listHeight) |
||||
}" |
||||
scroll-y |
||||
@scroll="onScroll" |
||||
:scroll-top="scrollTop" |
||||
:scrollIntoView="scrollIntoView" |
||||
> |
||||
<uMonth |
||||
:color="color" |
||||
:rowHeight="rowHeight" |
||||
:showMark="showMark" |
||||
:months="months" |
||||
:mode="mode" |
||||
:maxCount="maxCount" |
||||
:startText="startText" |
||||
:endText="endText" |
||||
:defaultDate="defaultDate" |
||||
:minDate="innerMinDate" |
||||
:maxDate="innerMaxDate" |
||||
:maxMonth="monthNum" |
||||
:readonly="readonly" |
||||
:maxRange="maxRange" |
||||
:rangePrompt="rangePrompt" |
||||
:showRangePrompt="showRangePrompt" |
||||
:allowSameDay="allowSameDay" |
||||
ref="month" |
||||
@monthSelected="monthSelected" |
||||
@updateMonthTop="updateMonthTop" |
||||
></uMonth> |
||||
</scroll-view> |
||||
<slot name="footer" v-if="showConfirm"> |
||||
<view class="u-calendar__confirm"> |
||||
<u-button |
||||
shape="circle" |
||||
:text=" |
||||
buttonDisabled ? confirmDisabledText : confirmText |
||||
" |
||||
:color="color" |
||||
@click="confirm" |
||||
:disabled="buttonDisabled" |
||||
></u-button> |
||||
</view> |
||||
</slot> |
||||
</view> |
||||
</u-popup> |
||||
</template> |
||||
|
||||
<script> |
||||
import uHeader from './header.vue' |
||||
import uMonth from './month.vue' |
||||
import props from './props.js' |
||||
import util from './util.js' |
||||
import dayjs from '../../libs/util/dayjs.js' |
||||
import Calendar from '../../libs/util/calendar.js' |
||||
/** |
||||
* Calendar 日历 |
||||
* @description 此组件用于单个选择日期,范围选择日期等,日历被包裹在底部弹起的容器中. |
||||
* @tutorial https://www.uviewui.com/components/calendar.html |
||||
* |
||||
* @property {String} title 标题内容 (默认 日期选择 ) |
||||
* @property {Boolean} showTitle 是否显示标题 (默认 true ) |
||||
* @property {Boolean} showSubtitle 是否显示副标题 (默认 true ) |
||||
* @property {String} mode 日期类型选择 single-选择单个日期,multiple-可以选择多个日期,range-选择日期范围 ( 默认 'single' ) |
||||
* @property {String} startText mode=range时,第一个日期底部的提示文字 (默认 '开始' ) |
||||
* @property {String} endText mode=range时,最后一个日期底部的提示文字 (默认 '结束' ) |
||||
* @property {Array} customList 自定义列表 |
||||
* @property {String} color 主题色,对底部按钮和选中日期有效 (默认 ‘#3c9cff' ) |
||||
* @property {String | Number} minDate 最小的可选日期 (默认 0 ) |
||||
* @property {String | Number} maxDate 最大可选日期 (默认 0 ) |
||||
* @property {Array | String| Date} defaultDate 默认选中的日期,mode为multiple或range是必须为数组格式 |
||||
* @property {String | Number} maxCount mode=multiple时,最多可选多少个日期 (默认 Number.MAX_SAFE_INTEGER ) |
||||
* @property {String | Number} rowHeight 日期行高 (默认 56 ) |
||||
* @property {Function} formatter 日期格式化函数 |
||||
* @property {Boolean} showLunar 是否显示农历 (默认 false ) |
||||
* @property {Boolean} showMark 是否显示月份背景色 (默认 true ) |
||||
* @property {String} confirmText 确定按钮的文字 (默认 '确定' ) |
||||
* @property {String} confirmDisabledText 确认按钮处于禁用状态时的文字 (默认 '确定' ) |
||||
* @property {Boolean} show 是否显示日历弹窗 (默认 false ) |
||||
* @property {Boolean} closeOnClickOverlay 是否允许点击遮罩关闭日历 (默认 false ) |
||||
* @property {Boolean} readonly 是否为只读状态,只读状态下禁止选择日期 (默认 false ) |
||||
* @property {String | Number} maxRange 日期区间最多可选天数,默认无限制,mode = range时有效 |
||||
* @property {String} rangePrompt 范围选择超过最多可选天数时的提示文案,mode = range时有效 |
||||
* @property {Boolean} showRangePrompt 范围选择超过最多可选天数时,是否展示提示文案,mode = range时有效 (默认 true ) |
||||
* @property {Boolean} allowSameDay 是否允许日期范围的起止时间为同一天,mode = range时有效 (默认 false ) |
||||
* @property {Number|String} round 圆角值,默认无圆角 (默认 0 ) |
||||
* @property {Number|String} monthNum 最多展示的月份数量 (默认 3 ) |
||||
* |
||||
* @event {Function()} confirm 点击确定按钮时触发 选择日期相关的返回参数 |
||||
* @event {Function()} close 日历关闭时触发 可定义页面关闭时的回调事件 |
||||
* @example <u-calendar :defaultDate="defaultDateMultiple" :show="show" mode="multiple" @confirm="confirm"> |
||||
</u-calendar> |
||||
* */ |
||||
export default { |
||||
name: 'u-calendar', |
||||
mixins: [uni.$u.mpMixin, uni.$u.mixin, props], |
||||
components: { |
||||
uHeader, |
||||
uMonth |
||||
}, |
||||
data() { |
||||
return { |
||||
// 需要显示的月份的数组 |
||||
months: [], |
||||
// 在月份滚动区域中,当前视图中月份的index索引 |
||||
monthIndex: 0, |
||||
// 月份滚动区域的高度 |
||||
listHeight: 0, |
||||
// month组件中选择的日期数组 |
||||
selected: [], |
||||
scrollIntoView: '', |
||||
scrollTop:0, |
||||
// 过滤处理方法 |
||||
innerFormatter: (value) => value |
||||
} |
||||
}, |
||||
watch: { |
||||
selectedChange: { |
||||
immediate: true, |
||||
handler(n) { |
||||
this.setMonth() |
||||
} |
||||
}, |
||||
// 打开弹窗时,设置月份数据 |
||||
show: { |
||||
immediate: true, |
||||
handler(n) { |
||||
this.setMonth() |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
// 由于maxDate和minDate可以为字符串(2021-10-10),或者数值(时间戳),但是dayjs如果接受字符串形式的时间戳会有问题,这里进行处理 |
||||
innerMaxDate() { |
||||
return uni.$u.test.number(this.maxDate) |
||||
? Number(this.maxDate) |
||||
: this.maxDate |
||||
}, |
||||
innerMinDate() { |
||||
return uni.$u.test.number(this.minDate) |
||||
? Number(this.minDate) |
||||
: this.minDate |
||||
}, |
||||
// 多个条件的变化,会引起选中日期的变化,这里统一管理监听 |
||||
selectedChange() { |
||||
return [this.innerMinDate, this.innerMaxDate, this.defaultDate] |
||||
}, |
||||
subtitle() { |
||||
// 初始化时,this.months为空数组,所以需要特别判断处理 |
||||
if (this.months.length) { |
||||
return `${this.months[this.monthIndex].year}年${ |
||||
this.months[this.monthIndex].month |
||||
}月` |
||||
} else { |
||||
return '' |
||||
} |
||||
}, |
||||
buttonDisabled() { |
||||
// 如果为range类型,且选择的日期个数不足1个时,让底部的按钮出于disabled状态 |
||||
if (this.mode === 'range') { |
||||
if (this.selected.length <= 1) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.start = Date.now() |
||||
this.init() |
||||
}, |
||||
methods: { |
||||
// 在微信小程序中,不支持将函数当做props参数,故只能通过ref形式调用 |
||||
setFormatter(e) { |
||||
this.innerFormatter = e |
||||
}, |
||||
// month组件内部选择日期后,通过事件通知给父组件 |
||||
monthSelected(e) { |
||||
this.selected = e |
||||
if (!this.showConfirm) { |
||||
// 在不需要确认按钮的情况下,如果为单选,或者范围多选且已选长度大于2,则直接进行返还 |
||||
if ( |
||||
this.mode === 'multiple' || |
||||
this.mode === 'single' || |
||||
(this.mode === 'range' && this.selected.length >= 2) |
||||
) { |
||||
this.$emit('confirm', this.selected) |
||||
} |
||||
} |
||||
}, |
||||
init() { |
||||
// 校验maxDate,不能小于当前时间 |
||||
if ( |
||||
this.innerMaxDate && |
||||
new Date(this.innerMaxDate).getTime() <= Date.now() |
||||
) { |
||||
return uni.$u.error('maxDate不能小于当前时间') |
||||
} |
||||
// 滚动区域的高度 |
||||
this.listHeight = this.rowHeight * 5 + 30 |
||||
this.setMonth() |
||||
}, |
||||
close() { |
||||
this.$emit('close') |
||||
}, |
||||
// 点击确定按钮 |
||||
confirm() { |
||||
if (!this.buttonDisabled) { |
||||
this.$emit('confirm', this.selected) |
||||
} |
||||
}, |
||||
// 获得两个日期之间的月份数 |
||||
getMonths(minDate, maxDate) { |
||||
const minYear = dayjs(minDate).year() |
||||
const minMonth = dayjs(minDate).month() + 1 |
||||
const maxYear = dayjs(maxDate).year() |
||||
const maxMonth = dayjs(maxDate).month() + 1 |
||||
return (maxYear - minYear) * 12 + (maxMonth - minMonth) + 1 |
||||
}, |
||||
// 设置月份数据 |
||||
setMonth() { |
||||
// 最小日期的毫秒数 |
||||
const minDate = this.innerMinDate || dayjs().valueOf() |
||||
// 如果没有指定最大日期,则往后推3个月 |
||||
const maxDate = |
||||
this.innerMaxDate || |
||||
dayjs(minDate) |
||||
.add(this.monthNum - 1, 'month') |
||||
.valueOf() |
||||
// 最大最小月份之间的共有多少个月份, |
||||
const months = uni.$u.range( |
||||
1, |
||||
this.monthNum, |
||||
this.getMonths(minDate, maxDate) |
||||
) |
||||
// 先清空数组 |
||||
this.months = [] |
||||
for (let i = 0; i < months; i++) { |
||||
this.months.push({ |
||||
date: new Array( |
||||
dayjs(minDate).add(i, 'month').daysInMonth() |
||||
) |
||||
.fill(1) |
||||
.map((item, index) => { |
||||
// 日期,取值1-31 |
||||
let day = index + 1 |
||||
// 星期,0-6,0为周日 |
||||
const week = dayjs(minDate) |
||||
.add(i, 'month') |
||||
.date(day) |
||||
.day() |
||||
const date = dayjs(minDate) |
||||
.add(i, 'month') |
||||
.date(day) |
||||
.format('YYYY-MM-DD') |
||||
let bottomInfo = '' |
||||
if (this.showLunar) { |
||||
// 将日期转为农历格式 |
||||
const lunar = Calendar.solar2lunar( |
||||
dayjs(date).year(), |
||||
dayjs(date).month() + 1, |
||||
dayjs(date).date() |
||||
) |
||||
bottomInfo = lunar.IDayCn |
||||
} |
||||
let config = { |
||||
day, |
||||
week, |
||||
// 小于最小允许的日期,或者大于最大的日期,则设置为disabled状态 |
||||
disabled: |
||||
dayjs(date).isBefore( |
||||
dayjs(minDate).format('YYYY-MM-DD') |
||||
) || |
||||
dayjs(date).isAfter( |
||||
dayjs(maxDate).format('YYYY-MM-DD') |
||||
), |
||||
// 返回一个日期对象,供外部的formatter获取当前日期的年月日等信息,进行加工处理 |
||||
date: new Date(date), |
||||
bottomInfo, |
||||
dot: false, |
||||
month: |
||||
dayjs(minDate).add(i, 'month').month() + 1 |
||||
} |
||||
const formatter = |
||||
this.formatter || this.innerFormatter |
||||
return formatter(config) |
||||
}), |
||||
// 当前所属的月份 |
||||
month: dayjs(minDate).add(i, 'month').month() + 1, |
||||
// 当前年份 |
||||
year: dayjs(minDate).add(i, 'month').year() |
||||
}) |
||||
} |
||||
|
||||
}, |
||||
// 滚动到默认设置的月份 |
||||
scrollIntoDefaultMonth(selected) { |
||||
// 查询默认日期在可选列表的下标 |
||||
const _index = this.months.findIndex(({ |
||||
year, |
||||
month |
||||
}) => { |
||||
month = uni.$u.padZero(month) |
||||
return `${year}-${month}` === selected |
||||
}) |
||||
if (_index !== -1) { |
||||
// #ifndef MP-WEIXIN |
||||
this.$nextTick(() => { |
||||
this.scrollIntoView = `month-${_index}` |
||||
}) |
||||
// #endif |
||||
// #ifdef MP-WEIXIN |
||||
this.scrollTop = this.months[_index].top || 0; |
||||
// #endif |
||||
} |
||||
}, |
||||
// scroll-view滚动监听 |
||||
onScroll(event) { |
||||
// 不允许小于0的滚动值,如果scroll-view到顶了,继续下拉,会出现负数值 |
||||
const scrollTop = Math.max(0, event.detail.scrollTop) |
||||
// 将当前滚动条数值,除以滚动区域的高度,可以得出当前滚动到了哪一个月份的索引 |
||||
for (let i = 0; i < this.months.length; i++) { |
||||
if (scrollTop >= (this.months[i].top || this.listHeight)) { |
||||
this.monthIndex = i |
||||
} |
||||
} |
||||
}, |
||||
// 更新月份的top值 |
||||
updateMonthTop(topArr = []) { |
||||
// 设置对应月份的top值,用于onScroll方法更新月份 |
||||
topArr.map((item, index) => { |
||||
this.months[index].top = item |
||||
}) |
||||
|
||||
// 获取默认日期的下标 |
||||
if (!this.defaultDate) { |
||||
// 如果没有设置默认日期,则将当天日期设置为默认选中的日期 |
||||
const selected = dayjs().format("YYYY-MM") |
||||
this.scrollIntoDefaultMonth(selected) |
||||
return |
||||
} |
||||
let selected = dayjs().format("YYYY-MM"); |
||||
// 单选模式,可以是字符串或数组,Date对象等 |
||||
if (!uni.$u.test.array(this.defaultDate)) { |
||||
selected = dayjs(this.defaultDate).format("YYYY-MM") |
||||
} else { |
||||
selected = dayjs(this.defaultDate[0]).format("YYYY-MM"); |
||||
} |
||||
this.scrollIntoDefaultMonth(selected) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
@import '../../libs/css/components.scss'; |
||||
|
||||
.u-calendar { |
||||
&__confirm { |
||||
padding: 7px 18px; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,85 @@ |
||||
export default { |
||||
methods: { |
||||
// 设置月份数据
|
||||
setMonth() { |
||||
// 月初是周几
|
||||
const day = dayjs(this.date).date(1).day() |
||||
const start = day == 0 ? 6 : day - 1 |
||||
|
||||
// 本月天数
|
||||
const days = dayjs(this.date).endOf('month').format('D') |
||||
|
||||
// 上个月天数
|
||||
const prevDays = dayjs(this.date).endOf('month').subtract(1, 'month').format('D') |
||||
|
||||
// 日期数据
|
||||
const arr = [] |
||||
// 清空表格
|
||||
this.month = [] |
||||
|
||||
// 添加上月数据
|
||||
arr.push( |
||||
...new Array(start).fill(1).map((e, i) => { |
||||
const day = prevDays - start + i + 1 |
||||
|
||||
return { |
||||
value: day, |
||||
disabled: true, |
||||
date: dayjs(this.date).subtract(1, 'month').date(day).format('YYYY-MM-DD') |
||||
} |
||||
}) |
||||
) |
||||
|
||||
// 添加本月数据
|
||||
arr.push( |
||||
...new Array(days - 0).fill(1).map((e, i) => { |
||||
const day = i + 1 |
||||
|
||||
return { |
||||
value: day, |
||||
date: dayjs(this.date).date(day).format('YYYY-MM-DD') |
||||
} |
||||
}) |
||||
) |
||||
|
||||
// 添加下个月
|
||||
arr.push( |
||||
...new Array(42 - days - start).fill(1).map((e, i) => { |
||||
const day = i + 1 |
||||
|
||||
return { |
||||
value: day, |
||||
disabled: true, |
||||
date: dayjs(this.date).add(1, 'month').date(day).format('YYYY-MM-DD') |
||||
} |
||||
}) |
||||
) |
||||
|
||||
// 分割数组
|
||||
for (let n = 0; n < arr.length; n += 7) { |
||||
this.month.push( |
||||
arr.slice(n, n + 7).map((e, i) => { |
||||
e.index = i + n |
||||
|
||||
// 自定义信息
|
||||
const custom = this.customList.find((c) => c.date == e.date) |
||||
|
||||
// 农历
|
||||
if (this.lunar) { |
||||
const { |
||||
IDayCn, |
||||
IMonthCn |
||||
} = this.getLunar(e.date) |
||||
e.lunar = IDayCn == '初一' ? IMonthCn : IDayCn |
||||
} |
||||
|
||||
return { |
||||
...e, |
||||
...custom |
||||
} |
||||
}) |
||||
) |
||||
} |
||||
} |
||||
} |
||||
} |