一键登录完成
This commit is contained in:
parent
d7c7f82ec3
commit
a0ee08bf85
19
api/user.js
19
api/user.js
@ -13,9 +13,9 @@ import Cache from '@/utils/cache';
|
||||
* 获取版本信息
|
||||
*/
|
||||
export function Appversion(data) {
|
||||
return request.get(`app/version`, data, {
|
||||
noAuth: true
|
||||
});
|
||||
return request.get(`app/version`, data, {
|
||||
noAuth: true
|
||||
});
|
||||
}
|
||||
/**
|
||||
* 获取用户信息
|
||||
@ -117,6 +117,17 @@ export function registerForget(data) {
|
||||
noAuth: true
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 一键登录
|
||||
*/
|
||||
export function phonelogin(data) {
|
||||
return request.post("auth/phonelogin", data, {
|
||||
noAuth: true
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取用户中心菜单
|
||||
*
|
||||
@ -257,7 +268,7 @@ export function extractCash(data) {
|
||||
* 申请提现
|
||||
*/
|
||||
export function getAdminApplyAPI(merId, data) {
|
||||
return request.get(`admin/${merId}/apply`, data);
|
||||
return request.get(`admin/${merId}/apply`, data);
|
||||
}
|
||||
/**
|
||||
* 提现银行/提现最低金额
|
||||
|
@ -39,7 +39,7 @@
|
||||
<view class="right-btn-box event_box">
|
||||
<view class="btn-item"
|
||||
v-if="evaluate != 2 && item.is_refund ==0 && pay_type!= 10 && status==0 && (evaluate != 10 && evaluate != 11) && orderData.refund_status || item.refund_num > 0"
|
||||
@click.stop="refund(item)">申请退款</view>
|
||||
@click.stop="showTips=true,refundItem=item">申请退款</view>
|
||||
<view class="btn-item err" v-if="item.is_refund ==1">退款中 x
|
||||
{{item.product_num - item.refund_num}}
|
||||
</view>
|
||||
@ -113,7 +113,7 @@
|
||||
</view>
|
||||
<view class="btn-item"
|
||||
v-if="evaluate != 2 && item.refund_switch==1 && pay_type!= 10 && status==0 && (item.is_refund == 0 && evaluate != 9 && orderData.refund_status || item.refund_num > 0)"
|
||||
@click.stop="refund(item)">申请退款</view>
|
||||
@click.stop="showTips=true,refundItem=item">申请退款</view>
|
||||
<view class='btn-item' v-if='item.is_reply==0 && evaluate==2 && item.refund_num > 0'
|
||||
@click.stop="evaluateTap(item.order_product_id,orderId)">去评价</view>
|
||||
<view class='btn-item on' v-else-if="item.is_reply==1 && evaluate==2">已评价</view>
|
||||
@ -121,6 +121,22 @@
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<u-overlay :show="showTips">
|
||||
<view class="tips">
|
||||
<view class="" style="font-size: 32rpx;">
|
||||
温馨提示
|
||||
</view>
|
||||
<view class="">
|
||||
下单后1小时内申请退款可极速退款
|
||||
</view>
|
||||
<u-button color="#61D137" text="我知道了" @tap="refund(refundItem)"></u-button>
|
||||
</view>
|
||||
</u-overlay>
|
||||
|
||||
|
||||
|
||||
|
||||
</view>
|
||||
</template>
|
||||
|
||||
@ -169,14 +185,14 @@
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
pay_type:{
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
status:{
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
pay_type: {
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
status: {
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
jump: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
@ -187,14 +203,16 @@
|
||||
return {};
|
||||
}
|
||||
},
|
||||
sale_type: {
|
||||
type: [Number, String],
|
||||
default: 1,
|
||||
},
|
||||
sale_type: {
|
||||
type: [Number, String],
|
||||
default: 1,
|
||||
},
|
||||
},
|
||||
computed: mapGetters(['viewColor']),
|
||||
data() {
|
||||
return {
|
||||
showTips: false, //退款提示弹窗
|
||||
refundItem: null,
|
||||
domain: HTTP_REQUEST_URL,
|
||||
totalNmu: '',
|
||||
isTimePay: false, //是否到支付时间
|
||||
@ -245,7 +263,8 @@
|
||||
let id = res.data.product_assist_set_id;
|
||||
uni.hideLoading();
|
||||
uni.navigateTo({
|
||||
url: '/pages/activity/assist_detail/index?id=' + id + '&sale_type=' + this.sale_type
|
||||
url: '/pages/activity/assist_detail/index?id=' + id +
|
||||
'&sale_type=' + this.sale_type
|
||||
});
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
@ -257,6 +276,7 @@
|
||||
},
|
||||
// 退款
|
||||
refund(item) {
|
||||
if (this.showTips) this.showTips = false;
|
||||
// #ifdef MP
|
||||
openOrderRefundSubscribe().then(() => {
|
||||
uni.hideLoading();
|
||||
@ -281,12 +301,14 @@
|
||||
if (this.evaluate == 0 || this.evaluate == 9 || this.orderData.is_virtual != 0) {
|
||||
uni.navigateTo({
|
||||
url: '/pages/users/refund/confirm?order_id=' + this.orderId + '&type=1' + '&ids=' + item
|
||||
.order_product_id + '&refund_type=1&order_type=' + this.orderData.order_type + '&sale_type=' + this.sale_type
|
||||
.order_product_id + '&refund_type=1&order_type=' + this.orderData.order_type +
|
||||
'&sale_type=' + this.sale_type
|
||||
})
|
||||
} else {
|
||||
uni.navigateTo({
|
||||
url: '/pages/users/refund/select?order_id=' + this.orderId + '&type=1&order_type=' + this
|
||||
.orderData.order_type + '&ids=' + item.order_product_id + '&sale_type=' + this.sale_type
|
||||
.orderData.order_type + '&ids=' + item.order_product_id + '&sale_type=' + this
|
||||
.sale_type
|
||||
})
|
||||
}
|
||||
// #endif
|
||||
@ -459,4 +481,21 @@
|
||||
background: #F0F0F0;
|
||||
}
|
||||
}
|
||||
|
||||
.tips {
|
||||
width: 590rpx;
|
||||
height: 354rpx;
|
||||
background-color: white;
|
||||
border-radius: 16rpx;
|
||||
position: absolute;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
transform: translate(-50%, -50%);
|
||||
box-sizing: border-box;
|
||||
padding: 50rpx 40rpx;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
}
|
||||
</style>
|
703
manifest.json
703
manifest.json
@ -1,352 +1,353 @@
|
||||
{
|
||||
"name": "惠农生活",
|
||||
"appid": "__UNI__3A527D1",
|
||||
"description": "",
|
||||
"versionName": "2.0.37",
|
||||
"versionCode": 2037,
|
||||
"transformPx": false,
|
||||
/* 5+App特有相关 */
|
||||
"app-plus": {
|
||||
"titleNView": true,
|
||||
"usingComponents": true,
|
||||
"nvueCompiler": "uni-app",
|
||||
"nvueStyleCompiler": "uni-app",
|
||||
"compilerVersion": 3,
|
||||
"compatible": {
|
||||
"ignoreVersion": true //true表示忽略版本检查提示框,HBuilderX1.9.0及以上版本支持
|
||||
},
|
||||
"splashscreen": {
|
||||
"alwaysShowBeforeRender": false,
|
||||
"waiting": false,
|
||||
"autoclose": true,
|
||||
"delay": 0
|
||||
},
|
||||
/* 模块配置 */
|
||||
"modules": {
|
||||
"VideoPlayer": {},
|
||||
"OAuth": {},
|
||||
"Payment": {},
|
||||
"Share": {},
|
||||
"iBeacon": {},
|
||||
"Maps": {},
|
||||
"Geolocation": {},
|
||||
"UniMP": {
|
||||
"description": "uni小程序"
|
||||
},
|
||||
"Push": {},
|
||||
"Barcode": {},
|
||||
"Camera": {}
|
||||
},
|
||||
"safearea": {
|
||||
"bottom": {
|
||||
"offset": "none"
|
||||
}
|
||||
},
|
||||
/* 应用发布信息 */
|
||||
"distribute": {
|
||||
/* android打包配置 */
|
||||
"android": {
|
||||
"permissions": [
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_COARSE_LOCATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_FINE_LOCATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_LOCATION_EXTRA_COMMANDS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BATTERY_STATS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BLUETOOTH\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CALL_PHONE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CAMERA\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_CONFIGURATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.MODIFY_AUDIO_SETTINGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.READ_LOGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.VIBRATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>",
|
||||
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BLUETOOTH_ADMIN\"/>"
|
||||
],
|
||||
"abiFilters": ["armeabi-v7a", "arm64-v8a", "x86"],
|
||||
"permissionExternalStorage": {
|
||||
"request": "none",
|
||||
"prompt": "应用保存运行状态等信息,需要获取读写手机存储(系统提示为访问设备上的照片、媒体内容和文件)权限,请允许。"
|
||||
},
|
||||
"permissionPhoneState": {
|
||||
"request": "none" //拨打电话权限关闭
|
||||
},
|
||||
"minSdkVersion": 23,
|
||||
"targetSdkVersion": 30
|
||||
},
|
||||
/* ios打包配置 */
|
||||
"ios": {
|
||||
"privacyDescription": {
|
||||
"NSPhotoLibraryUsageDescription": "上传用户头像保存分享海报",
|
||||
"NSPhotoLibraryAddUsageDescription": "上传用户头像保存分享海报",
|
||||
"NSCameraUsageDescription": "上传用户头像保存分享海报",
|
||||
"NSLocationWhenInUseUsageDescription": "根据客户地理位置推荐最近门店",
|
||||
"NSLocationAlwaysUsageDescription": "根据客户地理位置推荐最近门店"
|
||||
},
|
||||
"idfa": false,
|
||||
"dSYMs": false
|
||||
},
|
||||
/* SDK配置 */
|
||||
"sdkConfigs": {
|
||||
"maps": {
|
||||
"amap": {
|
||||
"appkey_ios": "0a3202688624938fd5d2f37b52c30d5d",
|
||||
"appkey_android": "0354f5ddc11e2ea76c5aac647f44d945",
|
||||
"name": "amapIvoVHpJR"
|
||||
}
|
||||
},
|
||||
"payment": {
|
||||
"weixin": {
|
||||
"__platform__": ["ios", "android"],
|
||||
"appid": "wx2e8f79ff281284f5",
|
||||
"UniversalLinks": "https://shop.lihaink.cn/"
|
||||
}
|
||||
},
|
||||
"share": {
|
||||
"weixin": {
|
||||
"appid": "wx2e8f79ff281284f5",
|
||||
"UniversalLinks": "https://shop.lihaink.cn/"
|
||||
}
|
||||
},
|
||||
"geolocation": {
|
||||
"amap": {
|
||||
"name": "amapIvoVHpJR",
|
||||
"__platform__": ["ios", "android"],
|
||||
"appkey_ios": "0a3202688624938fd5d2f37b52c30d5d",
|
||||
"appkey_android": "0354f5ddc11e2ea76c5aac647f44d945"
|
||||
}
|
||||
},
|
||||
"push": {},
|
||||
"oauth": {
|
||||
"weixin": {
|
||||
"appid": "wx2e8f79ff281284f5",
|
||||
"UniversalLinks": "https://shop.lihaink.cn/"
|
||||
}
|
||||
},
|
||||
"ad": {}
|
||||
},
|
||||
"icons": {
|
||||
"android": {
|
||||
"hdpi": "unpackage/res/icons/72x72.png",
|
||||
"xhdpi": "unpackage/res/icons/96x96.png",
|
||||
"xxhdpi": "unpackage/res/icons/144x144.png",
|
||||
"xxxhdpi": "unpackage/res/icons/192x192.png"
|
||||
},
|
||||
"ios": {
|
||||
"appstore": "unpackage/res/icons/1024x1024.png",
|
||||
"ipad": {
|
||||
"app": "unpackage/res/icons/76x76.png",
|
||||
"app@2x": "unpackage/res/icons/152x152.png",
|
||||
"notification": "unpackage/res/icons/20x20.png",
|
||||
"notification@2x": "unpackage/res/icons/40x40.png",
|
||||
"proapp@2x": "unpackage/res/icons/167x167.png",
|
||||
"settings": "unpackage/res/icons/29x29.png",
|
||||
"settings@2x": "unpackage/res/icons/58x58.png",
|
||||
"spotlight": "unpackage/res/icons/40x40.png",
|
||||
"spotlight@2x": "unpackage/res/icons/80x80.png"
|
||||
},
|
||||
"iphone": {
|
||||
"app@2x": "unpackage/res/icons/120x120.png",
|
||||
"app@3x": "unpackage/res/icons/180x180.png",
|
||||
"notification@2x": "unpackage/res/icons/40x40.png",
|
||||
"notification@3x": "unpackage/res/icons/60x60.png",
|
||||
"settings@2x": "unpackage/res/icons/58x58.png",
|
||||
"settings@3x": "unpackage/res/icons/87x87.png",
|
||||
"spotlight@2x": "unpackage/res/icons/80x80.png",
|
||||
"spotlight@3x": "unpackage/res/icons/120x120.png"
|
||||
}
|
||||
}
|
||||
},
|
||||
"splashscreen": {
|
||||
"useOriginalMsgbox": true,
|
||||
"androidStyle": "common"
|
||||
}
|
||||
},
|
||||
"nativePlugins": {
|
||||
"JG-JCore": {
|
||||
"JPUSH_APPKEY_IOS": "8a5efd65cda14fafa6e64ad3",
|
||||
"JPUSH_CHANNEL_IOS": "8a5efd65cda14fafa6e64ad3",
|
||||
"JPUSH_APPKEY_ANDROID": "b5f679f4357018605ea6fd2e",
|
||||
"JPUSH_CHANNEL_ANDROID": "",
|
||||
"__plugin_info__": {
|
||||
"name": "JG-JCore",
|
||||
"description": "极光推送JCore插件",
|
||||
"platforms": "Android,iOS",
|
||||
"url": "",
|
||||
"android_package_name": "uni.UNI3A527D1",
|
||||
"ios_bundle_id": "",
|
||||
"isCloud": false,
|
||||
"bought": -1,
|
||||
"pid": "",
|
||||
"parameters": {
|
||||
"JPUSH_APPKEY_IOS": {
|
||||
"des": "[iOS]极光portal配置应用信息时分配的AppKey",
|
||||
"key": "JCore:APP_KEY",
|
||||
"value": "daebe19b547c43128796a078"
|
||||
},
|
||||
"JPUSH_CHANNEL_IOS": {
|
||||
"des": "[iOS]用于统计分发渠道,不需要可填默认值developer-default",
|
||||
"key": "JCore:CHANNEL",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_APPKEY_ANDROID": {
|
||||
"des": "[Android]极光portal配置应用信息时分配的AppKey",
|
||||
"key": "JPUSH_APPKEY",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_CHANNEL_ANDROID": {
|
||||
"des": "[Android]用于统计分发渠道,不需要可填默认值developer-default",
|
||||
"key": "JPUSH_CHANNEL",
|
||||
"value": ""
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"JG-JPush": {
|
||||
"JPUSH_ISPRODUCTION_IOS": "true",
|
||||
"JPUSH_ADVERTISINGID_IOS": "",
|
||||
"JPUSH_DEFAULTINITJPUSH_IOS": "true",
|
||||
"JPUSH_OPPO_APPKEY": "",
|
||||
"JPUSH_OPPO_APPID": "",
|
||||
"JPUSH_OPPO_APPSECRET": "",
|
||||
"JPUSH_VIVO_APPKEY": "",
|
||||
"JPUSH_VIVO_APPID": "",
|
||||
"JPUSH_MEIZU_APPKEY": "",
|
||||
"JPUSH_MEIZU_APPID": "",
|
||||
"JPUSH_XIAOMI_APPKEY": "",
|
||||
"JPUSH_XIAOMI_APPID": "",
|
||||
"__plugin_info__": {
|
||||
"name": "JG-JPush",
|
||||
"description": "极光推送Hbuilder插件",
|
||||
"platforms": "Android,iOS",
|
||||
"url": "",
|
||||
"android_package_name": "uni.UNI3A527D1",
|
||||
"ios_bundle_id": "",
|
||||
"isCloud": false,
|
||||
"bought": -1,
|
||||
"pid": "",
|
||||
"parameters": {
|
||||
"JPUSH_ISPRODUCTION_IOS": {
|
||||
"des": "[iOS]是否是生产环境,是填true,不是填false或者不填",
|
||||
"key": "JPush:ISPRODUCTION",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_ADVERTISINGID_IOS": {
|
||||
"des": "[iOS]广告标识符(IDFA)如果不需要使用IDFA,可不填",
|
||||
"key": "JPush:ADVERTISINGID",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_DEFAULTINITJPUSH_IOS": {
|
||||
"des": "[iOS]是否默认初始化,是填true,不是填false或者不填",
|
||||
"key": "JPush:DEFAULTINITJPUSH",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_OPPO_APPKEY": {
|
||||
"des": "厂商OPPO-appkey,示例:OP-12345678",
|
||||
"key": "OPPO_APPKEY",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_OPPO_APPID": {
|
||||
"des": "厂商OPPO-appId,示例:OP-12345678",
|
||||
"key": "OPPO_APPID",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_OPPO_APPSECRET": {
|
||||
"des": "厂商OPPO-appSecret,示例:OP-12345678",
|
||||
"key": "OPPO_APPSECRET",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_VIVO_APPKEY": {
|
||||
"des": "厂商VIVO-appkey,示例:12345678",
|
||||
"key": "com.vivo.push.api_key",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_VIVO_APPID": {
|
||||
"des": "厂商VIVO-appId,示例:12345678",
|
||||
"key": "com.vivo.push.app_id",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_MEIZU_APPKEY": {
|
||||
"des": "厂商MEIZU-appKey,示例:MZ-12345678",
|
||||
"key": "MEIZU_APPKEY",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_MEIZU_APPID": {
|
||||
"des": "厂商MEIZU-appId,示例:MZ-12345678",
|
||||
"key": "MEIZU_APPID",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_XIAOMI_APPKEY": {
|
||||
"des": "厂商XIAOMI-appKey,示例:MI-12345678",
|
||||
"key": "XIAOMI_APPKEY",
|
||||
"value": ""
|
||||
},
|
||||
"JPUSH_XIAOMI_APPID": {
|
||||
"des": "厂商XIAOMI-appId,示例:MI-12345678",
|
||||
"key": "XIAOMI_APPID",
|
||||
"value": ""
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
/* 快应用特有相关 */
|
||||
"quickapp": {},
|
||||
/* 小程序特有相关 */
|
||||
"mp-weixin": {
|
||||
"appid": "wx5fb1cc8edb3f8baa",
|
||||
"setting": {
|
||||
"urlCheck": false,
|
||||
"minified": true,
|
||||
"postcss": true,
|
||||
"es6": true
|
||||
},
|
||||
"permission": {
|
||||
"scope.userLocation": {
|
||||
"desc": "获取您的位置"
|
||||
}
|
||||
},
|
||||
"requiredPrivateInfos": ["getLocation", "chooseAddress"],
|
||||
"usingComponents": true,
|
||||
"__usePrivacyCheck__": true
|
||||
},
|
||||
"mp-alipay": {
|
||||
"usingComponents": true
|
||||
},
|
||||
"mp-baidu": {
|
||||
"usingComponents": true
|
||||
},
|
||||
"mp-toutiao": {
|
||||
"usingComponents": true
|
||||
},
|
||||
"h5": {
|
||||
"devServer": {
|
||||
"https": false
|
||||
},
|
||||
"router": {
|
||||
"mode": "history",
|
||||
"base": ""
|
||||
},
|
||||
"domain": "",
|
||||
"sdkConfigs": {
|
||||
"maps": {
|
||||
"qqmap": {
|
||||
"key": "SMJBZ-WCHK4-ZPZUA-DSIXI-XDDVQ-XWFX7"
|
||||
}
|
||||
}
|
||||
},
|
||||
"title": "加载中...",
|
||||
"template": "template.h5.html",
|
||||
"optimization": {
|
||||
"treeShaking": {
|
||||
"enable": true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
"name" : "惠农生活",
|
||||
"appid" : "__UNI__3A527D1",
|
||||
"description" : "",
|
||||
"versionName" : "2.0.37",
|
||||
"versionCode" : 2037,
|
||||
"transformPx" : false,
|
||||
/* 5+App特有相关 */
|
||||
"app-plus" : {
|
||||
"titleNView" : true,
|
||||
"usingComponents" : true,
|
||||
"nvueCompiler" : "uni-app",
|
||||
"nvueStyleCompiler" : "uni-app",
|
||||
"compilerVersion" : 3,
|
||||
"compatible" : {
|
||||
"ignoreVersion" : true //true表示忽略版本检查提示框,HBuilderX1.9.0及以上版本支持
|
||||
},
|
||||
"splashscreen" : {
|
||||
"alwaysShowBeforeRender" : false,
|
||||
"waiting" : false,
|
||||
"autoclose" : true,
|
||||
"delay" : 0
|
||||
},
|
||||
/* 模块配置 */
|
||||
"modules" : {
|
||||
"VideoPlayer" : {},
|
||||
"OAuth" : {},
|
||||
"Payment" : {},
|
||||
"Share" : {},
|
||||
"iBeacon" : {},
|
||||
"Maps" : {},
|
||||
"Geolocation" : {},
|
||||
"UniMP" : {
|
||||
"description" : "uni小程序"
|
||||
},
|
||||
"Push" : {},
|
||||
"Barcode" : {},
|
||||
"Camera" : {}
|
||||
},
|
||||
"safearea" : {
|
||||
"bottom" : {
|
||||
"offset" : "none"
|
||||
}
|
||||
},
|
||||
/* 应用发布信息 */
|
||||
"distribute" : {
|
||||
/* android打包配置 */
|
||||
"android" : {
|
||||
"permissions" : [
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_COARSE_LOCATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_FINE_LOCATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_LOCATION_EXTRA_COMMANDS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BATTERY_STATS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BLUETOOTH\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CALL_PHONE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CAMERA\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_CONFIGURATION\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.MODIFY_AUDIO_SETTINGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.READ_LOGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.VIBRATE\"/>",
|
||||
"<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>",
|
||||
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>",
|
||||
"<uses-permission android:name=\"android.permission.BLUETOOTH_ADMIN\"/>"
|
||||
],
|
||||
"abiFilters" : [ "armeabi-v7a", "arm64-v8a", "x86" ],
|
||||
"permissionExternalStorage" : {
|
||||
"request" : "none",
|
||||
"prompt" : "应用保存运行状态等信息,需要获取读写手机存储(系统提示为访问设备上的照片、媒体内容和文件)权限,请允许。"
|
||||
},
|
||||
"permissionPhoneState" : {
|
||||
"request" : "none" //拨打电话权限关闭
|
||||
},
|
||||
"minSdkVersion" : 23,
|
||||
"targetSdkVersion" : 30
|
||||
},
|
||||
/* ios打包配置 */
|
||||
"ios" : {
|
||||
"privacyDescription" : {
|
||||
"NSPhotoLibraryUsageDescription" : "上传用户头像保存分享海报",
|
||||
"NSPhotoLibraryAddUsageDescription" : "上传用户头像保存分享海报",
|
||||
"NSCameraUsageDescription" : "上传用户头像保存分享海报",
|
||||
"NSLocationWhenInUseUsageDescription" : "根据客户地理位置推荐最近门店",
|
||||
"NSLocationAlwaysUsageDescription" : "根据客户地理位置推荐最近门店"
|
||||
},
|
||||
"idfa" : false,
|
||||
"dSYMs" : false
|
||||
},
|
||||
/* SDK配置 */
|
||||
"sdkConfigs" : {
|
||||
"maps" : {
|
||||
"amap" : {
|
||||
"appkey_ios" : "0a3202688624938fd5d2f37b52c30d5d",
|
||||
"appkey_android" : "0354f5ddc11e2ea76c5aac647f44d945",
|
||||
"name" : "amapIvoVHpJR"
|
||||
}
|
||||
},
|
||||
"payment" : {
|
||||
"weixin" : {
|
||||
"__platform__" : [ "ios", "android" ],
|
||||
"appid" : "wx2e8f79ff281284f5",
|
||||
"UniversalLinks" : "https://shop.lihaink.cn/"
|
||||
}
|
||||
},
|
||||
"share" : {
|
||||
"weixin" : {
|
||||
"appid" : "wx2e8f79ff281284f5",
|
||||
"UniversalLinks" : "https://shop.lihaink.cn/"
|
||||
}
|
||||
},
|
||||
"geolocation" : {
|
||||
"amap" : {
|
||||
"name" : "amapIvoVHpJR",
|
||||
"__platform__" : [ "ios", "android" ],
|
||||
"appkey_ios" : "0a3202688624938fd5d2f37b52c30d5d",
|
||||
"appkey_android" : "0354f5ddc11e2ea76c5aac647f44d945"
|
||||
}
|
||||
},
|
||||
"push" : {},
|
||||
"oauth" : {
|
||||
"weixin" : {
|
||||
"appid" : "wx2e8f79ff281284f5",
|
||||
"UniversalLinks" : "https://shop.lihaink.cn/"
|
||||
},
|
||||
"univerify" : {}
|
||||
},
|
||||
"ad" : {}
|
||||
},
|
||||
"icons" : {
|
||||
"android" : {
|
||||
"hdpi" : "unpackage/res/icons/72x72.png",
|
||||
"xhdpi" : "unpackage/res/icons/96x96.png",
|
||||
"xxhdpi" : "unpackage/res/icons/144x144.png",
|
||||
"xxxhdpi" : "unpackage/res/icons/192x192.png"
|
||||
},
|
||||
"ios" : {
|
||||
"appstore" : "unpackage/res/icons/1024x1024.png",
|
||||
"ipad" : {
|
||||
"app" : "unpackage/res/icons/76x76.png",
|
||||
"app@2x" : "unpackage/res/icons/152x152.png",
|
||||
"notification" : "unpackage/res/icons/20x20.png",
|
||||
"notification@2x" : "unpackage/res/icons/40x40.png",
|
||||
"proapp@2x" : "unpackage/res/icons/167x167.png",
|
||||
"settings" : "unpackage/res/icons/29x29.png",
|
||||
"settings@2x" : "unpackage/res/icons/58x58.png",
|
||||
"spotlight" : "unpackage/res/icons/40x40.png",
|
||||
"spotlight@2x" : "unpackage/res/icons/80x80.png"
|
||||
},
|
||||
"iphone" : {
|
||||
"app@2x" : "unpackage/res/icons/120x120.png",
|
||||
"app@3x" : "unpackage/res/icons/180x180.png",
|
||||
"notification@2x" : "unpackage/res/icons/40x40.png",
|
||||
"notification@3x" : "unpackage/res/icons/60x60.png",
|
||||
"settings@2x" : "unpackage/res/icons/58x58.png",
|
||||
"settings@3x" : "unpackage/res/icons/87x87.png",
|
||||
"spotlight@2x" : "unpackage/res/icons/80x80.png",
|
||||
"spotlight@3x" : "unpackage/res/icons/120x120.png"
|
||||
}
|
||||
}
|
||||
},
|
||||
"splashscreen" : {
|
||||
"useOriginalMsgbox" : true,
|
||||
"androidStyle" : "common"
|
||||
}
|
||||
},
|
||||
"nativePlugins" : {
|
||||
"JG-JCore" : {
|
||||
"JPUSH_APPKEY_IOS" : "8a5efd65cda14fafa6e64ad3",
|
||||
"JPUSH_CHANNEL_IOS" : "8a5efd65cda14fafa6e64ad3",
|
||||
"JPUSH_APPKEY_ANDROID" : "b5f679f4357018605ea6fd2e",
|
||||
"JPUSH_CHANNEL_ANDROID" : "",
|
||||
"__plugin_info__" : {
|
||||
"name" : "JG-JCore",
|
||||
"description" : "极光推送JCore插件",
|
||||
"platforms" : "Android,iOS",
|
||||
"url" : "",
|
||||
"android_package_name" : "uni.UNI3A527D1",
|
||||
"ios_bundle_id" : "",
|
||||
"isCloud" : false,
|
||||
"bought" : -1,
|
||||
"pid" : "",
|
||||
"parameters" : {
|
||||
"JPUSH_APPKEY_IOS" : {
|
||||
"des" : "[iOS]极光portal配置应用信息时分配的AppKey",
|
||||
"key" : "JCore:APP_KEY",
|
||||
"value" : "daebe19b547c43128796a078"
|
||||
},
|
||||
"JPUSH_CHANNEL_IOS" : {
|
||||
"des" : "[iOS]用于统计分发渠道,不需要可填默认值developer-default",
|
||||
"key" : "JCore:CHANNEL",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_APPKEY_ANDROID" : {
|
||||
"des" : "[Android]极光portal配置应用信息时分配的AppKey",
|
||||
"key" : "JPUSH_APPKEY",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_CHANNEL_ANDROID" : {
|
||||
"des" : "[Android]用于统计分发渠道,不需要可填默认值developer-default",
|
||||
"key" : "JPUSH_CHANNEL",
|
||||
"value" : ""
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"JG-JPush" : {
|
||||
"JPUSH_ISPRODUCTION_IOS" : "true",
|
||||
"JPUSH_ADVERTISINGID_IOS" : "",
|
||||
"JPUSH_DEFAULTINITJPUSH_IOS" : "true",
|
||||
"JPUSH_OPPO_APPKEY" : "",
|
||||
"JPUSH_OPPO_APPID" : "",
|
||||
"JPUSH_OPPO_APPSECRET" : "",
|
||||
"JPUSH_VIVO_APPKEY" : "",
|
||||
"JPUSH_VIVO_APPID" : "",
|
||||
"JPUSH_MEIZU_APPKEY" : "",
|
||||
"JPUSH_MEIZU_APPID" : "",
|
||||
"JPUSH_XIAOMI_APPKEY" : "",
|
||||
"JPUSH_XIAOMI_APPID" : "",
|
||||
"__plugin_info__" : {
|
||||
"name" : "JG-JPush",
|
||||
"description" : "极光推送Hbuilder插件",
|
||||
"platforms" : "Android,iOS",
|
||||
"url" : "",
|
||||
"android_package_name" : "uni.UNI3A527D1",
|
||||
"ios_bundle_id" : "",
|
||||
"isCloud" : false,
|
||||
"bought" : -1,
|
||||
"pid" : "",
|
||||
"parameters" : {
|
||||
"JPUSH_ISPRODUCTION_IOS" : {
|
||||
"des" : "[iOS]是否是生产环境,是填true,不是填false或者不填",
|
||||
"key" : "JPush:ISPRODUCTION",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_ADVERTISINGID_IOS" : {
|
||||
"des" : "[iOS]广告标识符(IDFA)如果不需要使用IDFA,可不填",
|
||||
"key" : "JPush:ADVERTISINGID",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_DEFAULTINITJPUSH_IOS" : {
|
||||
"des" : "[iOS]是否默认初始化,是填true,不是填false或者不填",
|
||||
"key" : "JPush:DEFAULTINITJPUSH",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_OPPO_APPKEY" : {
|
||||
"des" : "厂商OPPO-appkey,示例:OP-12345678",
|
||||
"key" : "OPPO_APPKEY",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_OPPO_APPID" : {
|
||||
"des" : "厂商OPPO-appId,示例:OP-12345678",
|
||||
"key" : "OPPO_APPID",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_OPPO_APPSECRET" : {
|
||||
"des" : "厂商OPPO-appSecret,示例:OP-12345678",
|
||||
"key" : "OPPO_APPSECRET",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_VIVO_APPKEY" : {
|
||||
"des" : "厂商VIVO-appkey,示例:12345678",
|
||||
"key" : "com.vivo.push.api_key",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_VIVO_APPID" : {
|
||||
"des" : "厂商VIVO-appId,示例:12345678",
|
||||
"key" : "com.vivo.push.app_id",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_MEIZU_APPKEY" : {
|
||||
"des" : "厂商MEIZU-appKey,示例:MZ-12345678",
|
||||
"key" : "MEIZU_APPKEY",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_MEIZU_APPID" : {
|
||||
"des" : "厂商MEIZU-appId,示例:MZ-12345678",
|
||||
"key" : "MEIZU_APPID",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_XIAOMI_APPKEY" : {
|
||||
"des" : "厂商XIAOMI-appKey,示例:MI-12345678",
|
||||
"key" : "XIAOMI_APPKEY",
|
||||
"value" : ""
|
||||
},
|
||||
"JPUSH_XIAOMI_APPID" : {
|
||||
"des" : "厂商XIAOMI-appId,示例:MI-12345678",
|
||||
"key" : "XIAOMI_APPID",
|
||||
"value" : ""
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
/* 快应用特有相关 */
|
||||
"quickapp" : {},
|
||||
/* 小程序特有相关 */
|
||||
"mp-weixin" : {
|
||||
"appid" : "wx5fb1cc8edb3f8baa",
|
||||
"setting" : {
|
||||
"urlCheck" : false,
|
||||
"minified" : true,
|
||||
"postcss" : true,
|
||||
"es6" : true
|
||||
},
|
||||
"permission" : {
|
||||
"scope.userLocation" : {
|
||||
"desc" : "获取您的位置"
|
||||
}
|
||||
},
|
||||
"requiredPrivateInfos" : [ "getLocation", "chooseAddress" ],
|
||||
"usingComponents" : true,
|
||||
"__usePrivacyCheck__" : true
|
||||
},
|
||||
"mp-alipay" : {
|
||||
"usingComponents" : true
|
||||
},
|
||||
"mp-baidu" : {
|
||||
"usingComponents" : true
|
||||
},
|
||||
"mp-toutiao" : {
|
||||
"usingComponents" : true
|
||||
},
|
||||
"h5" : {
|
||||
"devServer" : {
|
||||
"https" : false
|
||||
},
|
||||
"router" : {
|
||||
"mode" : "history",
|
||||
"base" : ""
|
||||
},
|
||||
"domain" : "",
|
||||
"sdkConfigs" : {
|
||||
"maps" : {
|
||||
"qqmap" : {
|
||||
"key" : "SMJBZ-WCHK4-ZPZUA-DSIXI-XDDVQ-XWFX7"
|
||||
}
|
||||
}
|
||||
},
|
||||
"title" : "加载中...",
|
||||
"template" : "template.h5.html",
|
||||
"optimization" : {
|
||||
"treeShaking" : {
|
||||
"enable" : true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
69
pages/users/login/oneClickStyle.js
Normal file
69
pages/users/login/oneClickStyle.js
Normal file
@ -0,0 +1,69 @@
|
||||
let customStyle = {
|
||||
"fullScreen": true, // 是否全屏显示,默认值: false
|
||||
"backgroundColor": "#ffffff", // 授权页面背景颜色,默认值:#ffffff
|
||||
"backgroundImage": "", // 全屏显示的背景图片,默认值:"" (仅支持本地图片,只有全屏显示时支持)
|
||||
"icon": {
|
||||
"path": "static/xxx.png", // 自定义显示在授权框中的logo,仅支持本地图片 默认显示App logo
|
||||
"width": "60px", //图标宽度 默认值:60px
|
||||
"height": "60px" //图标高度 默认值:60px
|
||||
},
|
||||
"closeIcon": {
|
||||
"path": "static/xxx.png", // 自定义显示在授权框中的logo,仅支持本地图片 默认显示App logo
|
||||
"width": "60px", //图标宽度 默认值:60px (HBuilderX 4.0支持)
|
||||
"height": "60px" //图标高度 默认值:60px (HBuilderX 4.0支持)
|
||||
},
|
||||
"phoneNum": {
|
||||
"color": "#202020" // 手机号文字颜色 默认值:#202020
|
||||
},
|
||||
"slogan": {
|
||||
"color": "#BBBBBB" // slogan 字体颜色 默认值:#BBBBBB
|
||||
},
|
||||
"authButton": {
|
||||
"normalColor": "#3CB625", // 授权按钮正常状态背景颜色 默认值:#3479f5
|
||||
"highlightColor": "#2861c5", // 授权按钮按下状态背景颜色 默认值:#2861c5(仅ios支持)
|
||||
"disabledColor": "#73aaf5", // 授权按钮不可点击时背景颜色 默认值:#73aaf5(仅ios支持)
|
||||
"textColor": "#ffffff", // 授权按钮文字颜色 默认值:#ffffff
|
||||
"title": "本机号码一键登录", // 授权按钮文案 默认值:“本机号码一键登录”
|
||||
"borderRadius": "24px" // 授权按钮圆角 默认值:"24px" (按钮高度的一半)
|
||||
},
|
||||
"otherLoginButton": {
|
||||
"visible": true, // 是否显示其他登录按钮,默认值:true
|
||||
"normalColor": "", // 其他登录按钮正常状态背景颜色 默认值:透明
|
||||
"highlightColor": "", // 其他登录按钮按下状态背景颜色 默认值:透明
|
||||
"textColor": "#656565", // 其他登录按钮文字颜色 默认值:#656565
|
||||
"title": "其他登录方式", // 其他登录方式按钮文字 默认值:“其他登录方式”
|
||||
"borderColor": "", //边框颜色 默认值:透明(仅iOS支持)
|
||||
"borderRadius": "0px" // 其他登录按钮圆角 默认值:"24px" (按钮高度的一半)
|
||||
},
|
||||
"privacyTerms": {
|
||||
"defaultCheckBoxState": true, // 条款勾选框初始状态 默认值: true
|
||||
"isCenterHint": false, //未勾选服务条款时点击登录按钮的提示是否居中显示 默认值: false (3.7.13+ 版本支持)
|
||||
"uncheckedImage": "", // 可选 条款勾选框未选中状态图片(仅支持本地图片 建议尺寸 24x24px)(3.2.0+ 版本支持)
|
||||
"checkedImage": "", // 可选 条款勾选框选中状态图片(仅支持本地图片 建议尺寸24x24px)(3.2.0+ 版本支持)
|
||||
"checkBoxSize": 12, // 可选 条款勾选框大小
|
||||
"textColor": "#BBBBBB", // 文字颜色 默认值:#BBBBBB
|
||||
"termsColor": "#5496E3", // 协议文字颜色 默认值: #5496E3
|
||||
"prefix": "我已阅读并同意", // 条款前的文案 默认值:“我已阅读并同意”
|
||||
"suffix": "并使用本机号码登录", // 条款后的文案 默认值:“并使用本机号码登录”
|
||||
"privacyItems": [ // 自定义协议条款,最大支持2个,需要同时设置url和title. 否则不生效
|
||||
{
|
||||
"url": "https://", // 点击跳转的协议详情页面
|
||||
"title": "用户服务协议" // 协议名称
|
||||
}
|
||||
]
|
||||
},
|
||||
"buttons": { // 自定义页面下方按钮仅全屏模式生效(3.1.14+ 版本支持)
|
||||
"iconWidth": "45px", // 图标宽度(高度等比例缩放) 默认值:45px
|
||||
"list": [{
|
||||
"provider": "apple",
|
||||
"iconPath": "/static/apple.png" // 图标路径仅支持本地图片
|
||||
},
|
||||
{
|
||||
"provider": "weixin",
|
||||
"iconPath": "/static/wechat.png" // 图标路径仅支持本地图片
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
|
||||
export default customStyle
|
@ -15,9 +15,9 @@
|
||||
</block>
|
||||
<block v-if="type == 2">
|
||||
<view class="txt">请等待商家收货并退款</view>
|
||||
<view class="time">还剩:
|
||||
<countDown :is-day="true" :tip-text="' '" :day-text="'天'" :hour-text="':'" :minute-text="':'" :second-text="' '"
|
||||
:datatime="datatime"></countDown>
|
||||
<view class="time">还剩:
|
||||
<countDown :is-day="true" :tip-text="' '" :day-text="'天'" :hour-text="':'" :minute-text="':'"
|
||||
:second-text="' '" :datatime="datatime"></countDown>
|
||||
</view>
|
||||
</block>
|
||||
<block v-if="type == 3">
|
||||
@ -59,12 +59,17 @@
|
||||
<view class="title">退款信息</view>
|
||||
<view class="product-box">
|
||||
<view class="product-item" v-for="(item,index) in detail.refundProduct" :key="index">
|
||||
<image class="img-box" :src="item.product.cart_info.productAttr.image" mode="" v-if="item.product.cart_info.productAttr.image"></image>
|
||||
<image class="img-box" :src="item.product.cart_info.productAttr.image" mode=""
|
||||
v-if="item.product.cart_info.productAttr.image"></image>
|
||||
<image class="img-box" :src="item.product.cart_info.product.image" mode="" v-else></image>
|
||||
<view class="msg">
|
||||
<view class="name line1"><text v-if="item.product.cart_info.product_type === 2" class="event_name event_bg">预售</text>{{item.product.cart_info.product.store_name}}</view>
|
||||
<view class="name line1"><text v-if="item.product.cart_info.product_type === 2"
|
||||
class="event_name event_bg">预售</text>{{item.product.cart_info.product.store_name}}
|
||||
</view>
|
||||
<view class="des">{{item.product.cart_info.productAttr.sku}}</view>
|
||||
<view class="price">¥{{item.product.product_type == 3 ? item.product.cart_info.productAssistAttr.assist_price : item.product.product_type == 4 ? item.product.cart_info.activeSku.active_price : item.product.cart_info.productAttr.price}}</view>
|
||||
<view class="price">
|
||||
¥{{item.product.product_type == 3 ? item.product.cart_info.productAssistAttr.assist_price : item.product.product_type == 4 ? item.product.cart_info.activeSku.active_price : item.product.cart_info.productAttr.price}}
|
||||
</view>
|
||||
<view class="num">x {{item.refund_num}}</view>
|
||||
</view>
|
||||
</view>
|
||||
@ -76,10 +81,10 @@
|
||||
<view class="txt flex">
|
||||
<text>{{detail.refund_order_sn}}</text>
|
||||
<!-- #ifdef H5 -->
|
||||
<text class='copy copy-data' :data-clipboard-text="detail.refund_order_sn">复制</text>
|
||||
<text class='copy copy-data' :data-clipboard-text="detail.refund_order_sn">复制</text>
|
||||
<!-- #endif -->
|
||||
<!-- #ifdef MP -->
|
||||
<text class='copy' @tap='copy'>复制</text>
|
||||
<text class='copy' @tap='copy'>复制</text>
|
||||
<!-- #endif -->
|
||||
</view>
|
||||
</view>
|
||||
@ -95,6 +100,13 @@
|
||||
<text>{{detail.refund_num}}</text>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<view class="item">
|
||||
<view class="label">退款原因:</view>
|
||||
<view class="txt flex">
|
||||
<text>{{ getRightText(detail.refund_message) }}</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="item">
|
||||
<view class="label">申请时间:</view>
|
||||
<view class="txt flex">
|
||||
@ -116,7 +128,7 @@
|
||||
</view>
|
||||
</view>
|
||||
<view class="btn-wrapper">
|
||||
<block v-if="type==-1">
|
||||
<block v-if="type==-1">
|
||||
<view class="btn" @click="applyAgain(detail)">再次申请</view>
|
||||
</block>
|
||||
<block v-else-if="type==1">
|
||||
@ -148,35 +160,40 @@
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: CRMEB Team <admin@crmeb.com>
|
||||
// +----------------------------------------------------------------------
|
||||
import { refundDetail, refundCancelApi } from '@/api/order.js'
|
||||
import {
|
||||
refundDetail,
|
||||
refundCancelApi
|
||||
} from '@/api/order.js'
|
||||
import ClipboardJS from "@/plugin/clipboard/clipboard.js";
|
||||
import countDown from '@/components/countDown'
|
||||
import { mapGetters } from "vuex";
|
||||
export default{
|
||||
import {
|
||||
mapGetters
|
||||
} from "vuex";
|
||||
export default {
|
||||
components: {
|
||||
countDown,
|
||||
countDown,
|
||||
},
|
||||
computed:{
|
||||
computed: {
|
||||
...mapGetters(['viewColor']),
|
||||
},
|
||||
data(){
|
||||
return{
|
||||
type:0,
|
||||
refund_order_id:0,
|
||||
detail:'',
|
||||
data() {
|
||||
return {
|
||||
type: 0,
|
||||
refund_order_id: 0,
|
||||
detail: '',
|
||||
datatime: 0
|
||||
}
|
||||
},
|
||||
onLoad(options) {
|
||||
uni.setNavigationBarTitle({
|
||||
title:'退款详情'
|
||||
title: '退款详情'
|
||||
})
|
||||
this.refund_order_id = options.id
|
||||
this.getDetail()
|
||||
},
|
||||
onReady: function() {
|
||||
// #ifdef H5
|
||||
this.$nextTick(function() {
|
||||
this.$nextTick(function() {
|
||||
const clipboard = new ClipboardJS(".copy-data");
|
||||
clipboard.on("success", () => {
|
||||
this.$util.Tips({
|
||||
@ -186,57 +203,65 @@
|
||||
});
|
||||
// #endif
|
||||
},
|
||||
methods:{
|
||||
methods: {
|
||||
getRightText(str) {
|
||||
const index = str.indexOf("@");
|
||||
if (index !== -1) {
|
||||
return str.substring(0, index);
|
||||
} else {
|
||||
return str
|
||||
}
|
||||
},
|
||||
//联系客服拨打电话
|
||||
call: function(service){
|
||||
call: function(service) {
|
||||
let that = this;
|
||||
if(service){
|
||||
if (service) {
|
||||
uni.showModal({
|
||||
title: '提示',
|
||||
content: '暂无在线客服,确定拨打客服电话:'+that.detail.merchant.service_phone+'吗?',
|
||||
content: '暂无在线客服,确定拨打客服电话:' + that.detail.merchant.service_phone + '吗?',
|
||||
success: function(res) {
|
||||
if (res.confirm) {
|
||||
uni.makePhoneCall({
|
||||
phoneNumber: that.detail.merchant.service_phone,
|
||||
phoneNumber: that.detail.merchant.service_phone,
|
||||
});
|
||||
}
|
||||
}
|
||||
})
|
||||
}else{
|
||||
} else {
|
||||
return that.$util.Tips({
|
||||
title: '暂无可用客服'
|
||||
})
|
||||
}
|
||||
},
|
||||
//查看大图
|
||||
loookImg(item,index){
|
||||
loookImg(item, index) {
|
||||
uni.previewImage({
|
||||
urls: this.detail.pics,
|
||||
current: this.detail.pics[index]
|
||||
});
|
||||
},
|
||||
getDetail(){
|
||||
refundDetail(this.refund_order_id).then(res=>{
|
||||
getDetail() {
|
||||
refundDetail(this.refund_order_id).then(res => {
|
||||
// status 0审核中 1待发货 2待收货 3已退款 -1已拒绝
|
||||
this.type = res.data.status
|
||||
this.detail = res.data
|
||||
this.datatime = res.data.auto_refund_time;
|
||||
|
||||
|
||||
})
|
||||
},
|
||||
goPage(){
|
||||
goPage() {
|
||||
uni.navigateTo({
|
||||
url:'/pages/users/refund/goods/index?id='+this.detail.refund_order_id
|
||||
url: '/pages/users/refund/goods/index?id=' + this.detail.refund_order_id
|
||||
})
|
||||
},
|
||||
applyAgain(item){
|
||||
applyAgain(item) {
|
||||
uni.navigateTo({
|
||||
url:`/pages/order_details/index?order_id=${item.refundProduct[0].product.order_id}`
|
||||
url: `/pages/order_details/index?order_id=${item.refundProduct[0].product.order_id}`
|
||||
})
|
||||
},
|
||||
go(){
|
||||
go() {
|
||||
uni.navigateTo({
|
||||
url:`/pages/users/refund/logistics?orderId=${this.detail.refund_order_id}`
|
||||
url: `/pages/users/refund/logistics?orderId=${this.detail.refund_order_id}`
|
||||
})
|
||||
},
|
||||
/**
|
||||
@ -252,30 +277,30 @@
|
||||
},
|
||||
// #endif
|
||||
// 客服
|
||||
goService(){
|
||||
goService() {
|
||||
uni.navigateTo({
|
||||
url:`/pages/chat/customer_list/chat?mer_id=${this.detail.mer_id}&uid=${this.detail.uid}&refund_order_id=${this.detail.refund_order_id}`
|
||||
url: `/pages/chat/customer_list/chat?mer_id=${this.detail.mer_id}&uid=${this.detail.uid}&refund_order_id=${this.detail.refund_order_id}`
|
||||
})
|
||||
},
|
||||
// 取消售后
|
||||
cancelSales(){
|
||||
cancelSales() {
|
||||
let that = this;
|
||||
uni.showModal({
|
||||
content: '确定要取消售后?',
|
||||
success: function(res) {
|
||||
if(res.confirm) {
|
||||
refundCancelApi(that.detail.refund_order_id).then(res => {
|
||||
if (res.status === 200) {
|
||||
that.$util.Tips({
|
||||
title: res.message
|
||||
});
|
||||
content: '确定要取消售后?',
|
||||
success: function(res) {
|
||||
if (res.confirm) {
|
||||
refundCancelApi(that.detail.refund_order_id).then(res => {
|
||||
if (res.status === 200) {
|
||||
that.$util.Tips({
|
||||
title: res.message
|
||||
});
|
||||
that.getDetail();
|
||||
}
|
||||
})
|
||||
} else if(res.cancel) {
|
||||
console.log('用户点击取消');
|
||||
}
|
||||
}
|
||||
})
|
||||
} else if (res.cancel) {
|
||||
console.log('用户点击取消');
|
||||
}
|
||||
}
|
||||
});
|
||||
},
|
||||
}
|
||||
@ -283,217 +308,251 @@
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
.refund-detail{
|
||||
.head{
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
height: 150rpx;
|
||||
padding: 0 30rpx;
|
||||
color: #fff;
|
||||
background-color: #666666;
|
||||
font-size: 30rpx;
|
||||
.txt{
|
||||
font-weight: bold;
|
||||
.refund-detail {
|
||||
.head {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
height: 150rpx;
|
||||
padding: 0 30rpx;
|
||||
color: #fff;
|
||||
background-color: #666666;
|
||||
font-size: 30rpx;
|
||||
|
||||
.txt {
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.time {
|
||||
margin-top: 10rpx;
|
||||
font-size: 24rpx;
|
||||
opacity: .8;
|
||||
|
||||
.time {
|
||||
display: inline-block;
|
||||
width: 600rpx;
|
||||
|
||||
/deep/ .red {
|
||||
color: #fff;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
.time{
|
||||
margin-top: 10rpx;
|
||||
font-size: 24rpx;
|
||||
opacity: .8;
|
||||
.time{
|
||||
display: inline-block;
|
||||
width: 600rpx;
|
||||
/deep/ .red{
|
||||
|
||||
.info-box {
|
||||
margin-top: 12rpx;
|
||||
background-color: #fff;
|
||||
|
||||
.title {
|
||||
padding: 0 32rpx;
|
||||
line-height: 86rpx;
|
||||
border-bottom: 1px solid #F0F0F0;
|
||||
color: #282828;
|
||||
}
|
||||
|
||||
.product-box {
|
||||
.product-item {
|
||||
display: flex;
|
||||
padding: 25rpx 30rpx;
|
||||
|
||||
.img-box {
|
||||
width: 130rpx;
|
||||
height: 130rpx;
|
||||
border-radius: 16rpx;
|
||||
}
|
||||
|
||||
.msg {
|
||||
position: relative;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
width: 440rpx;
|
||||
margin-left: 26rpx;
|
||||
|
||||
.name {
|
||||
font-size: 28rpx;
|
||||
color: #282828;
|
||||
}
|
||||
|
||||
.des {
|
||||
font-size: 20rpx;
|
||||
color: #868686;
|
||||
}
|
||||
|
||||
.price {
|
||||
font-size: 26rpx;
|
||||
color: var(--view-priceColor);
|
||||
}
|
||||
|
||||
.num {
|
||||
position: absolute;
|
||||
right: -80rpx;
|
||||
top: 4rpx;
|
||||
color: #868686;
|
||||
font-size: 26rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.event_name {
|
||||
display: inline-block;
|
||||
margin-right: 9rpx;
|
||||
color: #fff;
|
||||
font-size: 20rpx;
|
||||
padding: 0 8rpx;
|
||||
line-height: 30rpx;
|
||||
text-align: center;
|
||||
border-radius: 6rpx;
|
||||
}
|
||||
}
|
||||
|
||||
.event_bg {
|
||||
background: #FF7F00;
|
||||
}
|
||||
|
||||
.store-info {
|
||||
padding: 30rpx;
|
||||
|
||||
.des {
|
||||
margin-top: 10rpx;
|
||||
font-size: 26rpx;
|
||||
color: #868686;
|
||||
}
|
||||
|
||||
.red-txt {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-top: 10rpx;
|
||||
color: $theme-color;
|
||||
font-size: 24rpx;
|
||||
|
||||
.iconfont {
|
||||
font-size: 30rpx;
|
||||
margin-right: 5rpx;
|
||||
margin-top: 6rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.content {
|
||||
margin-top: 12rpx;
|
||||
padding: 30rpx 30rpx 0;
|
||||
background-color: #FFFFFF;
|
||||
|
||||
.item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 30rpx;
|
||||
|
||||
.txt {
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
width: 450rpx;
|
||||
color: #868686;
|
||||
text-align: right;
|
||||
|
||||
.copy {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 80rpx;
|
||||
height: 34rpx;
|
||||
margin-left: 20rpx;
|
||||
border: 1px solid #666666;
|
||||
border-radius: 17rpx;
|
||||
font-size: 20rpx;
|
||||
color: #333;
|
||||
}
|
||||
}
|
||||
|
||||
&:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.upload-img {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
margin-top: 20rpx;
|
||||
|
||||
.img-item {
|
||||
position: relative;
|
||||
width: 156rpx;
|
||||
height: 156rpx;
|
||||
margin-right: 23rpx;
|
||||
margin-top: 20rpx;
|
||||
|
||||
&:nth-child(4n) {
|
||||
margin-right: 0;
|
||||
}
|
||||
|
||||
image {
|
||||
width: 156rpx;
|
||||
height: 156rpx;
|
||||
border-radius: 8rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.btn-wrapper {
|
||||
position: relative;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
height: 100rpx;
|
||||
|
||||
button {
|
||||
font-size: 28rpx;
|
||||
}
|
||||
|
||||
.btn {
|
||||
width: 176rpx;
|
||||
height: 60rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
margin-left: 20rpx;
|
||||
background: var(--view-theme);
|
||||
border-radius: 30rpx;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.gray {
|
||||
background: transparent;
|
||||
border: 1px solid #eee;
|
||||
color: #AAAAAA;
|
||||
}
|
||||
|
||||
&:after {
|
||||
content: ' ';
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 50%;
|
||||
width: 690rpx;
|
||||
height: 1px;
|
||||
margin-left: -345rpx;
|
||||
background-color: #f0f0f0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.box {
|
||||
margin-top: 12rpx;
|
||||
padding: 25rpx 30rpx;
|
||||
background-color: #fff;
|
||||
font-size: 30rpx;
|
||||
color: #282828;
|
||||
|
||||
.des {
|
||||
margin-top: 5rpx;
|
||||
font-size: 26rpx;
|
||||
color: #868686;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.info-box{
|
||||
margin-top: 12rpx;
|
||||
background-color: #fff;
|
||||
.title{
|
||||
padding: 0 32rpx;
|
||||
line-height: 86rpx;
|
||||
border-bottom: 1px solid #F0F0F0;
|
||||
color: #282828;
|
||||
}
|
||||
.product-box{
|
||||
.product-item{
|
||||
display: flex;
|
||||
padding: 25rpx 30rpx;
|
||||
.img-box{
|
||||
width:130rpx;
|
||||
height:130rpx;
|
||||
border-radius:16rpx;
|
||||
}
|
||||
.msg{
|
||||
position: relative;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
width: 440rpx;
|
||||
margin-left: 26rpx;
|
||||
.name{
|
||||
font-size: 28rpx;
|
||||
color: #282828;
|
||||
}
|
||||
.des{
|
||||
font-size: 20rpx;
|
||||
color: #868686;
|
||||
}
|
||||
.price{
|
||||
font-size: 26rpx;
|
||||
color: var(--view-priceColor);
|
||||
}
|
||||
.num{
|
||||
position: absolute;
|
||||
right: -80rpx;
|
||||
top: 4rpx;
|
||||
color: #868686;
|
||||
font-size: 26rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.event_name{
|
||||
display: inline-block;
|
||||
margin-right: 9rpx;
|
||||
color: #fff;
|
||||
font-size: 20rpx;
|
||||
padding: 0 8rpx;
|
||||
line-height: 30rpx;
|
||||
text-align: center;
|
||||
border-radius: 6rpx;
|
||||
}
|
||||
}
|
||||
|
||||
.event_bg{
|
||||
background: #FF7F00;
|
||||
}
|
||||
.store-info{
|
||||
padding: 30rpx;
|
||||
.des{
|
||||
margin-top: 10rpx;
|
||||
font-size: 26rpx;
|
||||
color: #868686;
|
||||
}
|
||||
.red-txt{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-top: 10rpx;
|
||||
color: $theme-color;
|
||||
font-size: 24rpx;
|
||||
.iconfont{
|
||||
font-size: 30rpx;
|
||||
margin-right: 5rpx;
|
||||
margin-top: 6rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
/deep/.styleAll {
|
||||
color: #fff;
|
||||
}
|
||||
.content{
|
||||
margin-top: 12rpx;
|
||||
padding: 30rpx 30rpx 0;
|
||||
background-color: #FFFFFF;
|
||||
.item{
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 30rpx;
|
||||
.txt{
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
width: 450rpx;
|
||||
color: #868686;
|
||||
text-align: right;
|
||||
.copy{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width:80rpx;
|
||||
height:34rpx;
|
||||
margin-left: 20rpx;
|
||||
border:1px solid #666666;
|
||||
border-radius:17rpx;
|
||||
font-size: 20rpx;
|
||||
color: #333;
|
||||
}
|
||||
}
|
||||
&:last-child{
|
||||
margin-bottom: 0;
|
||||
}
|
||||
.upload-img{
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
margin-top: 20rpx;
|
||||
.img-item{
|
||||
position: relative;
|
||||
width: 156rpx;
|
||||
height: 156rpx;
|
||||
margin-right: 23rpx;
|
||||
margin-top: 20rpx;
|
||||
&:nth-child(4n){
|
||||
margin-right: 0;
|
||||
}
|
||||
image{
|
||||
width: 156rpx;
|
||||
height: 156rpx;
|
||||
border-radius: 8rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
.btn-wrapper{
|
||||
position: relative;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
height: 100rpx;
|
||||
button{
|
||||
font-size: 28rpx;
|
||||
}
|
||||
.btn{
|
||||
width:176rpx;
|
||||
height:60rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
margin-left: 20rpx;
|
||||
background: var(--view-theme);
|
||||
border-radius:30rpx;
|
||||
color: #fff;
|
||||
}
|
||||
.gray{
|
||||
background: transparent;
|
||||
border: 1px solid #eee;
|
||||
color: #AAAAAA;
|
||||
}
|
||||
&:after{
|
||||
content:' ';
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 50%;
|
||||
width: 690rpx;
|
||||
height:1px;
|
||||
margin-left: -345rpx;
|
||||
background-color: #f0f0f0;
|
||||
}
|
||||
}
|
||||
}
|
||||
.box{
|
||||
margin-top: 12rpx;
|
||||
padding:25rpx 30rpx;
|
||||
background-color: #fff;
|
||||
font-size: 30rpx;
|
||||
color: #282828;
|
||||
.des{
|
||||
margin-top: 5rpx;
|
||||
font-size: 26rpx;
|
||||
color: #868686;
|
||||
}
|
||||
}
|
||||
}
|
||||
/deep/.styleAll{
|
||||
color: #fff;
|
||||
}
|
||||
</style>
|
||||
</style>
|
35
uniCloud-aliyun/cloudfunctions/oneClickLoginFn/index.js
Normal file
35
uniCloud-aliyun/cloudfunctions/oneClickLoginFn/index.js
Normal file
@ -0,0 +1,35 @@
|
||||
// 下面仅展示客户端使用post方式发送content-type为application/json请求的场景
|
||||
exports.main = async (event) => {
|
||||
let body = event.body
|
||||
if (event.isBase64Encoded) {
|
||||
body = Buffer.from(body, 'base64')
|
||||
}
|
||||
const {
|
||||
access_token,
|
||||
openid
|
||||
} = JSON.parse(body)
|
||||
if (!access_token || !openid) {
|
||||
return { // 不建议把完整手机号返回给前端
|
||||
success: false,
|
||||
error: {
|
||||
code: 'params is not fund',
|
||||
message: '参数错误',
|
||||
}
|
||||
}
|
||||
}
|
||||
const res = await uniCloud.getPhoneNumber({
|
||||
provider: 'univerify',
|
||||
appid: '__UNI__3A527D1', // DCloud appid,不同于callFunction方式调用,使用云函数Url化需要传递DCloud appid参数!!!
|
||||
access_token: access_token,
|
||||
openid: openid
|
||||
})
|
||||
console.log(res); // res里包含手机号
|
||||
return { // 不建议把完整手机号返回给前端
|
||||
success: true,
|
||||
info: {
|
||||
code: 0,
|
||||
message: '获取手机号成功',
|
||||
data: res,
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,8 @@
|
||||
{
|
||||
"name": "oneClickLoginFn",
|
||||
"dependencies": {},
|
||||
"extensions": {
|
||||
"uni-cloud-jql": {},
|
||||
"uni-cloud-verify": {}
|
||||
}
|
||||
}
|
12
uniCloud-aliyun/database/JQL查询.jql
Normal file
12
uniCloud-aliyun/database/JQL查询.jql
Normal file
@ -0,0 +1,12 @@
|
||||
// 本文件用于,使用JQL语法操作项目关联的uniCloud空间的数据库,方便开发调试和远程数据库管理
|
||||
// 编写clientDB的js API(也支持常规js语法,比如var),可以对云数据库进行增删改查操作。不支持uniCloud-db组件写法
|
||||
// 可以全部运行,也可以选中部分代码运行。点击工具栏上的运行按钮或者按下【F5】键运行代码
|
||||
// 如果文档中存在多条JQL语句,只有最后一条语句生效
|
||||
// 如果混写了普通js,最后一条语句需是数据库操作语句
|
||||
// 此处代码运行不受DB Schema的权限控制,移植代码到实际业务中注意在schema中配好permission
|
||||
// 不支持clientDB的action
|
||||
// 数据库查询有最大返回条数限制,详见:https://uniapp.dcloud.net.cn/uniCloud/cf-database.html#limit
|
||||
// 详细JQL语法,请参考:https://uniapp.dcloud.net.cn/uniCloud/jql.html
|
||||
|
||||
// 下面示例查询uni-id-users表的所有数据
|
||||
db.collection('uni-id-users').get();
|
6
uni_modules/uni-config-center/changelog.md
Normal file
6
uni_modules/uni-config-center/changelog.md
Normal file
@ -0,0 +1,6 @@
|
||||
## 0.0.3(2022-11-11)
|
||||
- 修复 config 方法获取根节点为数组格式配置时错误的转化为了对象的Bug
|
||||
## 0.0.2(2021-04-16)
|
||||
- 修改插件package信息
|
||||
## 0.0.1(2021-03-15)
|
||||
- 初始化项目
|
81
uni_modules/uni-config-center/package.json
Normal file
81
uni_modules/uni-config-center/package.json
Normal file
@ -0,0 +1,81 @@
|
||||
{
|
||||
"id": "uni-config-center",
|
||||
"displayName": "uni-config-center",
|
||||
"version": "0.0.3",
|
||||
"description": "uniCloud 配置中心",
|
||||
"keywords": [
|
||||
"配置",
|
||||
"配置中心"
|
||||
],
|
||||
"repository": "",
|
||||
"engines": {
|
||||
"HBuilderX": "^3.1.0"
|
||||
},
|
||||
"dcloudext": {
|
||||
"sale": {
|
||||
"regular": {
|
||||
"price": "0.00"
|
||||
},
|
||||
"sourcecode": {
|
||||
"price": "0.00"
|
||||
}
|
||||
},
|
||||
"contact": {
|
||||
"qq": ""
|
||||
},
|
||||
"declaration": {
|
||||
"ads": "无",
|
||||
"data": "无",
|
||||
"permissions": "无"
|
||||
},
|
||||
"npmurl": "",
|
||||
"type": "unicloud-template-function"
|
||||
},
|
||||
"directories": {
|
||||
"example": "../../../scripts/dist"
|
||||
},
|
||||
"uni_modules": {
|
||||
"dependencies": [],
|
||||
"encrypt": [],
|
||||
"platforms": {
|
||||
"cloud": {
|
||||
"tcb": "y",
|
||||
"aliyun": "y"
|
||||
},
|
||||
"client": {
|
||||
"App": {
|
||||
"app-vue": "u",
|
||||
"app-nvue": "u"
|
||||
},
|
||||
"H5-mobile": {
|
||||
"Safari": "u",
|
||||
"Android Browser": "u",
|
||||
"微信浏览器(Android)": "u",
|
||||
"QQ浏览器(Android)": "u"
|
||||
},
|
||||
"H5-pc": {
|
||||
"Chrome": "u",
|
||||
"IE": "u",
|
||||
"Edge": "u",
|
||||
"Firefox": "u",
|
||||
"Safari": "u"
|
||||
},
|
||||
"小程序": {
|
||||
"微信": "u",
|
||||
"阿里": "u",
|
||||
"百度": "u",
|
||||
"字节跳动": "u",
|
||||
"QQ": "u"
|
||||
},
|
||||
"快应用": {
|
||||
"华为": "u",
|
||||
"联盟": "u"
|
||||
},
|
||||
"Vue": {
|
||||
"vue2": "y",
|
||||
"vue3": "u"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
93
uni_modules/uni-config-center/readme.md
Normal file
93
uni_modules/uni-config-center/readme.md
Normal file
@ -0,0 +1,93 @@
|
||||
# 为什么使用uni-config-center
|
||||
|
||||
实际开发中很多插件需要配置文件才可以正常运行,如果每个插件都单独进行配置的话就会产生下面这样的目录结构
|
||||
|
||||
```bash
|
||||
cloudfunctions
|
||||
└─────common 公共模块
|
||||
├─plugin-a // 插件A对应的目录
|
||||
│ ├─index.js
|
||||
│ ├─config.json // plugin-a对应的配置文件
|
||||
│ └─other-file.cert // plugin-a依赖的其他文件
|
||||
└─plugin-b // plugin-b对应的目录
|
||||
├─index.js
|
||||
└─config.json // plugin-b对应的配置文件
|
||||
```
|
||||
|
||||
假设插件作者要发布一个项目模板,里面使用了很多需要配置的插件,无论是作者发布还是用户使用都是一个大麻烦。
|
||||
|
||||
uni-config-center就是用了统一管理这些配置文件的,使用uni-config-center后的目录结构如下
|
||||
|
||||
```bash
|
||||
cloudfunctions
|
||||
└─────common 公共模块
|
||||
├─plugin-a // 插件A对应的目录
|
||||
│ └─index.js
|
||||
├─plugin-b // plugin-b对应的目录
|
||||
│ └─index.js
|
||||
└─uni-config-center
|
||||
├─index.js // config-center入口文件
|
||||
├─plugin-a
|
||||
│ ├─config.json // plugin-a对应的配置文件
|
||||
│ └─other-file.cert // plugin-a依赖的其他文件
|
||||
└─plugin-b
|
||||
└─config.json // plugin-b对应的配置文件
|
||||
```
|
||||
|
||||
使用uni-config-center后的优势
|
||||
|
||||
- 配置文件统一管理,分离插件主体和配置信息,更新插件更方便
|
||||
- 支持对config.json设置schema,插件使用者在HBuilderX内编写config.json文件时会有更好的提示(后续HBuilderX会提供支持)
|
||||
|
||||
# 用法
|
||||
|
||||
在要使用uni-config-center的公共模块或云函数内引入uni-config-center依赖,请参考:[使用公共模块](https://uniapp.dcloud.net.cn/uniCloud/cf-common)
|
||||
|
||||
```js
|
||||
const createConfig = require('uni-config-center')
|
||||
|
||||
const uniIdConfig = createConfig({
|
||||
pluginId: 'uni-id', // 插件id
|
||||
defaultConfig: { // 默认配置
|
||||
tokenExpiresIn: 7200,
|
||||
tokenExpiresThreshold: 600,
|
||||
},
|
||||
customMerge: function(defaultConfig, userConfig) { // 自定义默认配置和用户配置的合并规则,不设置的情况侠会对默认配置和用户配置进行深度合并
|
||||
// defaudltConfig 默认配置
|
||||
// userConfig 用户配置
|
||||
return Object.assign(defaultConfig, userConfig)
|
||||
}
|
||||
})
|
||||
|
||||
|
||||
// 以如下配置为例
|
||||
// {
|
||||
// "tokenExpiresIn": 7200,
|
||||
// "passwordErrorLimit": 6,
|
||||
// "bindTokenToDevice": false,
|
||||
// "passwordErrorRetryTime": 3600,
|
||||
// "app-plus": {
|
||||
// "tokenExpiresIn": 2592000
|
||||
// },
|
||||
// "service": {
|
||||
// "sms": {
|
||||
// "codeExpiresIn": 300
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
|
||||
// 获取配置
|
||||
uniIdConfig.config() // 获取全部配置,注意:uni-config-center内不存在对应插件目录时会返回空对象
|
||||
uniIdConfig.config('tokenExpiresIn') // 指定键值获取配置,返回:7200
|
||||
uniIdConfig.config('service.sms.codeExpiresIn') // 指定键值获取配置,返回:300
|
||||
uniIdConfig.config('tokenExpiresThreshold', 600) // 指定键值获取配置,如果不存在则取传入的默认值,返回:600
|
||||
|
||||
// 获取文件绝对路径
|
||||
uniIdConfig.resolve('custom-token.js') // 获取uni-config-center/uni-id/custom-token.js文件的路径
|
||||
|
||||
// 引用文件(require)
|
||||
uniIDConfig.requireFile('custom-token.js') // 使用require方式引用uni-config-center/uni-id/custom-token.js文件。文件不存在时返回undefined,文件内有其他错误导致require失败时会抛出错误。
|
||||
|
||||
// 判断是否包含某文件
|
||||
uniIDConfig.hasFile('custom-token.js') // 配置目录是否包含某文件,true: 文件存在,false: 文件不存在
|
||||
```
|
File diff suppressed because one or more lines are too long
@ -0,0 +1 @@
|
||||
{"name":"uni-config-center","version":"0.0.3","description":"配置中心","main":"index.js","keywords":[],"author":"DCloud","license":"Apache-2.0"}
|
34
uni_modules/uni-id-common/changelog.md
Normal file
34
uni_modules/uni-id-common/changelog.md
Normal file
@ -0,0 +1,34 @@
|
||||
## 1.0.17(2024-04-26)
|
||||
- 兼容uni-app-x对客户端uniPlatform的调整(uni-app-x内uniPlatform区分app-android、app-ios)
|
||||
## 1.0.16(2023-04-25)
|
||||
- 新增maxTokenLength配置,用于限制数据库用户记录token数组的最大长度
|
||||
## 1.0.15(2023-04-06)
|
||||
- 修复部分语言国际化出错的Bug
|
||||
## 1.0.14(2023-03-07)
|
||||
- 修复 admin用户包含其他角色时未包含在token的Bug
|
||||
## 1.0.13(2022-07-21)
|
||||
- 修复 创建token时未传角色权限信息生成的token不正确的bug
|
||||
## 1.0.12(2022-07-15)
|
||||
- 提升与旧版本uni-id的兼容性(补充读取配置文件时回退平台app-plus、h5),但是仍推荐使用新平台名进行配置(app、web)
|
||||
## 1.0.11(2022-07-14)
|
||||
- 修复 部分情况下报`read property 'reduce' of undefined`的错误
|
||||
## 1.0.10(2022-07-11)
|
||||
- 将token存储在用户表的token字段内,与旧版本uni-id保持一致
|
||||
## 1.0.9(2022-07-01)
|
||||
- checkToken兼容token内未缓存角色权限的情况,此时将查库获取角色权限
|
||||
## 1.0.8(2022-07-01)
|
||||
- 修复clientDB默认依赖时部分情况下获取不到uni-id配置的Bug
|
||||
## 1.0.7(2022-06-30)
|
||||
- 修复config文件不合法时未抛出具体错误的Bug
|
||||
## 1.0.6(2022-06-28)
|
||||
- 移除插件内的数据表schema
|
||||
## 1.0.5(2022-06-27)
|
||||
- 修复使用多应用配置时报`Cannot read property 'appId' of undefined`的Bug
|
||||
## 1.0.4(2022-06-27)
|
||||
- 修复使用自定义token内容功能报错的Bug [详情](https://ask.dcloud.net.cn/question/147945)
|
||||
## 1.0.2(2022-06-23)
|
||||
- 对齐旧版本uni-id默认配置
|
||||
## 1.0.1(2022-06-22)
|
||||
- 补充对uni-config-center的依赖
|
||||
## 1.0.0(2022-06-21)
|
||||
- 提供uni-id token创建、校验、刷新接口,简化旧版uni-id公共模块
|
85
uni_modules/uni-id-common/package.json
Normal file
85
uni_modules/uni-id-common/package.json
Normal file
@ -0,0 +1,85 @@
|
||||
{
|
||||
"id": "uni-id-common",
|
||||
"displayName": "uni-id-common",
|
||||
"version": "1.0.17",
|
||||
"description": "包含uni-id token生成、校验、刷新功能的云函数公共模块",
|
||||
"keywords": [
|
||||
"uni-id-common",
|
||||
"uniCloud",
|
||||
"token",
|
||||
"权限"
|
||||
],
|
||||
"repository": "https://gitcode.net/dcloud/uni-id-common",
|
||||
"engines": {
|
||||
"HBuilderX": "^3.1.0"
|
||||
},
|
||||
"dcloudext": {
|
||||
"sale": {
|
||||
"regular": {
|
||||
"price": "0.00"
|
||||
},
|
||||
"sourcecode": {
|
||||
"price": "0.00"
|
||||
}
|
||||
},
|
||||
"contact": {
|
||||
"qq": ""
|
||||
},
|
||||
"declaration": {
|
||||
"ads": "无",
|
||||
"data": "无",
|
||||
"permissions": "无"
|
||||
},
|
||||
"npmurl": "",
|
||||
"type": "unicloud-template-function"
|
||||
},
|
||||
"uni_modules": {
|
||||
"dependencies": ["uni-config-center"],
|
||||
"encrypt": [],
|
||||
"platforms": {
|
||||
"cloud": {
|
||||
"tcb": "y",
|
||||
"aliyun": "y",
|
||||
"alipay": "n"
|
||||
},
|
||||
"client": {
|
||||
"Vue": {
|
||||
"vue2": "u",
|
||||
"vue3": "u"
|
||||
},
|
||||
"App": {
|
||||
"app-vue": "u",
|
||||
"app-nvue": "u"
|
||||
},
|
||||
"H5-mobile": {
|
||||
"Safari": "u",
|
||||
"Android Browser": "u",
|
||||
"微信浏览器(Android)": "u",
|
||||
"QQ浏览器(Android)": "u"
|
||||
},
|
||||
"H5-pc": {
|
||||
"Chrome": "u",
|
||||
"IE": "u",
|
||||
"Edge": "u",
|
||||
"Firefox": "u",
|
||||
"Safari": "u"
|
||||
},
|
||||
"小程序": {
|
||||
"微信": "u",
|
||||
"阿里": "u",
|
||||
"百度": "u",
|
||||
"字节跳动": "u",
|
||||
"QQ": "u",
|
||||
"钉钉": "u",
|
||||
"快手": "u",
|
||||
"飞书": "u",
|
||||
"京东": "u"
|
||||
},
|
||||
"快应用": {
|
||||
"华为": "u",
|
||||
"联盟": "u"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
3
uni_modules/uni-id-common/readme.md
Normal file
3
uni_modules/uni-id-common/readme.md
Normal file
@ -0,0 +1,3 @@
|
||||
# uni-id-common
|
||||
|
||||
文档请参考:[uni-id-common](https://uniapp.dcloud.net.cn/uniCloud/uni-id-common.html)
|
File diff suppressed because one or more lines are too long
@ -0,0 +1 @@
|
||||
{"name":"uni-id-common","version":"1.0.17","description":"uni-id token生成、校验、刷新","main":"index.js","homepage":"https://uniapp.dcloud.io/uniCloud/uni-id-common.html","repository":{"type":"git","url":"git+https://gitee.com/dcloud/uni-id-common.git"},"author":"DCloud","license":"Apache-2.0","dependencies":{"uni-config-center":"file:../../../../../uni-config-center/uniCloud/cloudfunctions/common/uni-config-center"}}
|
@ -0,0 +1,26 @@
|
||||
'use strict';
|
||||
|
||||
class BridgeError extends Error {
|
||||
|
||||
constructor(code, message) {
|
||||
super(message)
|
||||
|
||||
this._code = code
|
||||
}
|
||||
|
||||
get code() {
|
||||
return this._code
|
||||
}
|
||||
|
||||
get errCode() {
|
||||
return this._code
|
||||
}
|
||||
|
||||
get errMsg() {
|
||||
return this.message
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
BridgeError
|
||||
}
|
@ -0,0 +1,124 @@
|
||||
'use strict';
|
||||
|
||||
const {
|
||||
ProviderType
|
||||
} = require('./consts.js')
|
||||
|
||||
const configCenter = require('uni-config-center')
|
||||
|
||||
// 多维数据为兼容uni-id以前版本配置
|
||||
const OauthConfig = {
|
||||
'weixin-app': [
|
||||
['app', 'oauth', 'weixin'],
|
||||
['app-plus', 'oauth', 'weixin']
|
||||
],
|
||||
'weixin-mp': [
|
||||
['mp-weixin', 'oauth', 'weixin']
|
||||
],
|
||||
'weixin-h5': [
|
||||
['web', 'oauth', 'weixin-h5'],
|
||||
['h5-weixin', 'oauth', 'weixin'],
|
||||
['h5', 'oauth', 'weixin']
|
||||
],
|
||||
'weixin-web': [
|
||||
['web', 'oauth', 'weixin-web']
|
||||
],
|
||||
'qq-app': [
|
||||
['app', 'oauth', 'qq'],
|
||||
['app-plus', 'oauth', 'qq']
|
||||
],
|
||||
'qq-mp': [
|
||||
['mp-qq', 'oauth', 'qq']
|
||||
]
|
||||
}
|
||||
|
||||
const Support_Platforms = [
|
||||
ProviderType.WEIXIN_MP,
|
||||
ProviderType.WEIXIN_H5,
|
||||
ProviderType.WEIXIN_APP,
|
||||
ProviderType.WEIXIN_WEB,
|
||||
ProviderType.QQ_MP,
|
||||
ProviderType.QQ_APP
|
||||
]
|
||||
|
||||
class ConfigBase {
|
||||
|
||||
constructor() {
|
||||
const uniIdConfigCenter = configCenter({
|
||||
pluginId: 'uni-id'
|
||||
})
|
||||
|
||||
this._uniIdConfig = uniIdConfigCenter.config()
|
||||
}
|
||||
|
||||
getAppConfig(appid) {
|
||||
if (Array.isArray(this._uniIdConfig)) {
|
||||
return this._uniIdConfig.find((item) => {
|
||||
return (item.dcloudAppid === appid)
|
||||
})
|
||||
}
|
||||
return this._uniIdConfig
|
||||
}
|
||||
}
|
||||
|
||||
class AppConfig extends ConfigBase {
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
}
|
||||
|
||||
get(appid, platform) {
|
||||
if (!this.isSupport(platform)) {
|
||||
return null
|
||||
}
|
||||
|
||||
let appConfig = this.getAppConfig(appid)
|
||||
if (!appConfig) {
|
||||
return null
|
||||
}
|
||||
|
||||
return this.getOauthConfig(appConfig, platform)
|
||||
}
|
||||
|
||||
isSupport(platformName) {
|
||||
return (Support_Platforms.indexOf(platformName) >= 0)
|
||||
}
|
||||
|
||||
getOauthConfig(appConfig, platformName) {
|
||||
let treePath = OauthConfig[platformName]
|
||||
let node = this.findNode(appConfig, treePath)
|
||||
if (node && node.appid && node.appsecret) {
|
||||
return {
|
||||
appid: node.appid,
|
||||
secret: node.appsecret
|
||||
}
|
||||
}
|
||||
return null
|
||||
}
|
||||
|
||||
findNode(treeNode, arrayPath) {
|
||||
let node = treeNode
|
||||
for (let treePath of arrayPath) {
|
||||
for (let name of treePath) {
|
||||
const currentNode = node[name]
|
||||
if (currentNode) {
|
||||
node = currentNode
|
||||
} else {
|
||||
node = null
|
||||
break
|
||||
}
|
||||
}
|
||||
if (node === null) {
|
||||
node = treeNode
|
||||
} else {
|
||||
break
|
||||
}
|
||||
}
|
||||
return node
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
module.exports = {
|
||||
AppConfig
|
||||
};
|
@ -0,0 +1,30 @@
|
||||
'use strict';
|
||||
|
||||
const TAG = "UNI_OPEN_BRIDGE"
|
||||
|
||||
const HTTP_STATUS = {
|
||||
SUCCESS: 200
|
||||
}
|
||||
|
||||
const ProviderType = {
|
||||
WEIXIN_MP: 'weixin-mp',
|
||||
WEIXIN_H5: 'weixin-h5',
|
||||
WEIXIN_APP: 'weixin-app',
|
||||
WEIXIN_WEB: 'weixin-web',
|
||||
QQ_MP: 'qq-mp',
|
||||
QQ_APP: 'qq-app'
|
||||
}
|
||||
|
||||
// old
|
||||
const PlatformType = ProviderType
|
||||
|
||||
const ErrorCodeType = {
|
||||
SYSTEM_ERROR: TAG + "_SYSTEM_ERROR"
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
HTTP_STATUS,
|
||||
ProviderType,
|
||||
PlatformType,
|
||||
ErrorCodeType
|
||||
}
|
@ -0,0 +1,317 @@
|
||||
'use strict';
|
||||
|
||||
const {
|
||||
PlatformType,
|
||||
ProviderType,
|
||||
ErrorCodeType
|
||||
} = require('./consts.js')
|
||||
|
||||
const {
|
||||
AppConfig
|
||||
} = require('./config.js')
|
||||
|
||||
const {
|
||||
Storage
|
||||
} = require('./storage.js')
|
||||
|
||||
const {
|
||||
BridgeError
|
||||
} = require('./bridge-error.js')
|
||||
|
||||
const {
|
||||
WeixinServer
|
||||
} = require('./weixin-server.js')
|
||||
|
||||
const appConfig = new AppConfig()
|
||||
|
||||
class AccessToken extends Storage {
|
||||
|
||||
constructor() {
|
||||
super('access-token', ['provider', 'appid'])
|
||||
}
|
||||
|
||||
async update(key) {
|
||||
super.update(key)
|
||||
|
||||
const result = await this.getByWeixinServer(key)
|
||||
|
||||
return this.set(key, result.value, result.duration)
|
||||
}
|
||||
|
||||
async fallback(key) {
|
||||
return this.getByWeixinServer(key)
|
||||
}
|
||||
|
||||
async getByWeixinServer(key) {
|
||||
const oauthConfig = appConfig.get(key.dcloudAppid, key.provider)
|
||||
let methodName
|
||||
if (key.provider === ProviderType.WEIXIN_MP) {
|
||||
methodName = 'GetMPAccessTokenData'
|
||||
} else if (key.provider === ProviderType.WEIXIN_H5) {
|
||||
methodName = 'GetH5AccessTokenData'
|
||||
} else {
|
||||
throw new BridgeError(ErrorCodeType.SYSTEM_ERROR, "provider invalid")
|
||||
}
|
||||
|
||||
const responseData = await WeixinServer[methodName](oauthConfig)
|
||||
|
||||
const duration = responseData.expires_in || (60 * 60 * 2)
|
||||
delete responseData.expires_in
|
||||
|
||||
return {
|
||||
value: responseData,
|
||||
duration
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class UserAccessToken extends Storage {
|
||||
|
||||
constructor() {
|
||||
super('user-access-token', ['provider', 'appid', 'openid'])
|
||||
}
|
||||
}
|
||||
|
||||
class SessionKey extends Storage {
|
||||
|
||||
constructor() {
|
||||
super('session-key', ['provider', 'appid', 'openid'])
|
||||
}
|
||||
}
|
||||
|
||||
class Encryptkey extends Storage {
|
||||
|
||||
constructor() {
|
||||
super('encrypt-key', ['provider', 'appid', 'openid'])
|
||||
}
|
||||
|
||||
async update(key) {
|
||||
super.update(key)
|
||||
|
||||
const result = await this.getByWeixinServer(key)
|
||||
|
||||
return this.set(key, result.value, result.duration)
|
||||
}
|
||||
|
||||
getKeyString(key) {
|
||||
return `${super.getKeyString(key)}-${key.version}`
|
||||
}
|
||||
|
||||
getExpiresIn(value) {
|
||||
if (value <= 0) {
|
||||
return 60
|
||||
}
|
||||
return value
|
||||
}
|
||||
|
||||
async fallback(key) {
|
||||
return this.getByWeixinServer(key)
|
||||
}
|
||||
|
||||
async getByWeixinServer(key) {
|
||||
const accessToken = await Factory.Get(AccessToken, key)
|
||||
const userSession = await Factory.Get(SessionKey, key)
|
||||
|
||||
const responseData = await WeixinServer.GetUserEncryptKeyData({
|
||||
openid: key.openid,
|
||||
access_token: accessToken.access_token,
|
||||
session_key: userSession.session_key
|
||||
})
|
||||
|
||||
const keyInfo = responseData.key_info_list.find((item) => {
|
||||
return item.version === key.version
|
||||
})
|
||||
|
||||
if (!keyInfo) {
|
||||
throw new BridgeError(ErrorCodeType.SYSTEM_ERROR, 'key version invalid')
|
||||
}
|
||||
|
||||
const value = {
|
||||
encrypt_key: keyInfo.encrypt_key,
|
||||
iv: keyInfo.iv
|
||||
}
|
||||
|
||||
return {
|
||||
value,
|
||||
duration: keyInfo.expire_in
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class Ticket extends Storage {
|
||||
|
||||
constructor() {
|
||||
super('ticket', ['provider', 'appid'])
|
||||
}
|
||||
|
||||
async update(key) {
|
||||
super.update(key)
|
||||
|
||||
const result = await this.getByWeixinServer(key)
|
||||
|
||||
return this.set(key, result.value, result.duration)
|
||||
}
|
||||
|
||||
async fallback(key) {
|
||||
return this.getByWeixinServer(key)
|
||||
}
|
||||
|
||||
async getByWeixinServer(key) {
|
||||
const accessToken = await Factory.Get(AccessToken, {
|
||||
dcloudAppid: key.dcloudAppid,
|
||||
provider: ProviderType.WEIXIN_H5
|
||||
})
|
||||
|
||||
const responseData = await WeixinServer.GetH5TicketData(accessToken)
|
||||
|
||||
const duration = responseData.expires_in || (60 * 60 * 2)
|
||||
delete responseData.expires_in
|
||||
delete responseData.errcode
|
||||
delete responseData.errmsg
|
||||
|
||||
return {
|
||||
value: responseData,
|
||||
duration
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
const Factory = {
|
||||
|
||||
async Get(T, key, fallback) {
|
||||
Factory.FixOldKey(key)
|
||||
return Factory.MakeUnique(T).get(key, fallback)
|
||||
},
|
||||
|
||||
async Set(T, key, value, expiresIn) {
|
||||
Factory.FixOldKey(key)
|
||||
return Factory.MakeUnique(T).set(key, value, expiresIn)
|
||||
},
|
||||
|
||||
async Remove(T, key) {
|
||||
Factory.FixOldKey(key)
|
||||
return Factory.MakeUnique(T).remove(key)
|
||||
},
|
||||
|
||||
async Update(T, key) {
|
||||
Factory.FixOldKey(key)
|
||||
return Factory.MakeUnique(T).update(key)
|
||||
},
|
||||
|
||||
FixOldKey(key) {
|
||||
if (!key.provider) {
|
||||
key.provider = key.platform
|
||||
}
|
||||
|
||||
const configData = appConfig.get(key.dcloudAppid, key.provider)
|
||||
if (!configData) {
|
||||
throw new BridgeError(ErrorCodeType.SYSTEM_ERROR, 'appid or provider invalid')
|
||||
}
|
||||
key.appid = configData.appid
|
||||
},
|
||||
|
||||
MakeUnique(T) {
|
||||
return new T()
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// exports
|
||||
|
||||
async function getAccessToken(key, fallback) {
|
||||
return Factory.Get(AccessToken, key, fallback)
|
||||
}
|
||||
|
||||
async function setAccessToken(key, value, expiresIn) {
|
||||
return Factory.Set(AccessToken, key, value, expiresIn)
|
||||
}
|
||||
|
||||
async function removeAccessToken(key) {
|
||||
return Factory.Remove(AccessToken, key)
|
||||
}
|
||||
|
||||
async function updateAccessToken(key) {
|
||||
return Factory.Update(AccessToken, key)
|
||||
}
|
||||
|
||||
async function getUserAccessToken(key, fallback) {
|
||||
return Factory.Get(UserAccessToken, key, fallback)
|
||||
}
|
||||
|
||||
async function setUserAccessToken(key, value, expiresIn) {
|
||||
return Factory.Set(UserAccessToken, key, value, expiresIn)
|
||||
}
|
||||
|
||||
async function removeUserAccessToken(key) {
|
||||
return Factory.Remove(UserAccessToken, key)
|
||||
}
|
||||
|
||||
async function getSessionKey(key, fallback) {
|
||||
return Factory.Get(SessionKey, key, fallback)
|
||||
}
|
||||
|
||||
async function setSessionKey(key, value, expiresIn) {
|
||||
return Factory.Set(SessionKey, key, value, expiresIn)
|
||||
}
|
||||
|
||||
async function removeSessionKey(key) {
|
||||
return Factory.Remove(SessionKey, key)
|
||||
}
|
||||
|
||||
async function getEncryptKey(key, fallback) {
|
||||
return Factory.Get(Encryptkey, key, fallback)
|
||||
}
|
||||
|
||||
async function setEncryptKey(key, value, expiresIn) {
|
||||
return Factory.Set(Encryptkey, key, value, expiresIn)
|
||||
}
|
||||
|
||||
async function removeEncryptKey(key) {
|
||||
return Factory.Remove(Encryptkey, key)
|
||||
}
|
||||
|
||||
async function updateEncryptKey(key) {
|
||||
return Factory.Update(Encryptkey, key)
|
||||
}
|
||||
|
||||
async function getTicket(key, fallback) {
|
||||
return Factory.Get(Ticket, key, fallback)
|
||||
}
|
||||
|
||||
async function setTicket(key, value, expiresIn) {
|
||||
return Factory.Set(Ticket, key, value, expiresIn)
|
||||
}
|
||||
|
||||
async function removeTicket(key) {
|
||||
return Factory.Remove(Ticket, key)
|
||||
}
|
||||
|
||||
async function updateTicket(key) {
|
||||
return Factory.Update(Ticket, key)
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
getAccessToken,
|
||||
setAccessToken,
|
||||
removeAccessToken,
|
||||
updateAccessToken,
|
||||
getUserAccessToken,
|
||||
setUserAccessToken,
|
||||
removeUserAccessToken,
|
||||
getSessionKey,
|
||||
setSessionKey,
|
||||
removeSessionKey,
|
||||
getEncryptKey,
|
||||
setEncryptKey,
|
||||
removeEncryptKey,
|
||||
updateEncryptKey,
|
||||
getTicket,
|
||||
setTicket,
|
||||
removeTicket,
|
||||
updateTicket,
|
||||
ProviderType,
|
||||
PlatformType,
|
||||
WeixinServer,
|
||||
ErrorCodeType
|
||||
}
|
@ -0,0 +1,15 @@
|
||||
{
|
||||
"name": "uni-open-bridge-common",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"keywords": [],
|
||||
"author": "",
|
||||
"license": "ISC",
|
||||
"dependencies": {
|
||||
"uni-config-center": "file:../../../../../uni-config-center/uniCloud/cloudfunctions/common/uni-config-center"
|
||||
}
|
||||
}
|
@ -0,0 +1,111 @@
|
||||
'use strict';
|
||||
|
||||
const {
|
||||
Validator
|
||||
} = require('./validator.js')
|
||||
|
||||
const {
|
||||
CacheKeyCascade
|
||||
} = require('./uni-cloud-cache.js')
|
||||
|
||||
const {
|
||||
BridgeError
|
||||
} = require('./bridge-error.js')
|
||||
|
||||
class Storage {
|
||||
|
||||
constructor(type, keys) {
|
||||
this._type = type || null
|
||||
this._keys = keys || []
|
||||
}
|
||||
|
||||
async get(key, fallback) {
|
||||
this.validateKey(key)
|
||||
const result = await this.create(key, fallback).get()
|
||||
return result.value
|
||||
}
|
||||
|
||||
async set(key, value, expiresIn) {
|
||||
this.validateKey(key)
|
||||
this.validateValue(value)
|
||||
const expires_in = this.getExpiresIn(expiresIn)
|
||||
if (expires_in !== 0) {
|
||||
await this.create(key).set(this.getValue(value), expires_in)
|
||||
}
|
||||
}
|
||||
|
||||
async remove(key) {
|
||||
this.validateKey(key)
|
||||
await this.create(key).remove()
|
||||
}
|
||||
|
||||
// virtual
|
||||
async update(key) {
|
||||
this.validateKey(key)
|
||||
}
|
||||
|
||||
async ttl(key) {
|
||||
this.validateKey(key)
|
||||
// 后续考虑支持
|
||||
}
|
||||
|
||||
async fallback(key) {}
|
||||
|
||||
getKeyString(key) {
|
||||
const keyArray = [Storage.Prefix]
|
||||
this._keys.forEach((name) => {
|
||||
keyArray.push(key[name])
|
||||
})
|
||||
keyArray.push(this._type)
|
||||
return keyArray.join(':')
|
||||
}
|
||||
|
||||
getValue(value) {
|
||||
return value
|
||||
}
|
||||
|
||||
getExpiresIn(value) {
|
||||
if (value !== undefined) {
|
||||
return value
|
||||
}
|
||||
return -1
|
||||
}
|
||||
|
||||
validateKey(key) {
|
||||
Validator.Key(this._keys, key)
|
||||
}
|
||||
|
||||
validateValue(value) {
|
||||
Validator.Value(value)
|
||||
}
|
||||
|
||||
create(key, fallback) {
|
||||
const keyString = this.getKeyString(key)
|
||||
const options = {
|
||||
layers: [{
|
||||
type: 'database',
|
||||
key: keyString
|
||||
}, {
|
||||
type: 'redis',
|
||||
key: keyString
|
||||
}]
|
||||
}
|
||||
|
||||
const _this = this
|
||||
return new CacheKeyCascade({
|
||||
...options,
|
||||
fallback: async function() {
|
||||
if (fallback) {
|
||||
return fallback(key)
|
||||
} else if (_this.fallback) {
|
||||
return _this.fallback(key)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
Storage.Prefix = "uni-id"
|
||||
|
||||
module.exports = {
|
||||
Storage
|
||||
};
|
@ -0,0 +1,324 @@
|
||||
const db = uniCloud.database()
|
||||
|
||||
function getType(value) {
|
||||
return Object.prototype.toString.call(value).slice(8, -1).toLowerCase()
|
||||
}
|
||||
|
||||
const validator = {
|
||||
key: function(value) {
|
||||
const err = new Error('Invalid key')
|
||||
if (typeof value !== 'string') {
|
||||
throw err
|
||||
}
|
||||
const valueTrim = value.trim()
|
||||
if (!valueTrim || valueTrim !== value) {
|
||||
throw err
|
||||
}
|
||||
},
|
||||
value: function(value) {
|
||||
// 仅作简单校验
|
||||
const type = getType(value)
|
||||
const validValueType = ['null', 'number', 'string', 'array', 'object']
|
||||
if (validValueType.indexOf(type) === -1) {
|
||||
throw new Error('Invalid value type')
|
||||
}
|
||||
},
|
||||
duration: function(value) {
|
||||
const err = new Error('Invalid duration')
|
||||
if (value === undefined) {
|
||||
return
|
||||
}
|
||||
if (typeof value !== 'number' || value === 0) {
|
||||
throw err
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 入库时 expired 为过期时间对应的时间戳,永不过期用-1表示
|
||||
* 返回结果时 与redis对齐,-1表示永不过期,-2表示已过期或不存在
|
||||
*/
|
||||
class DatabaseCache {
|
||||
constructor({
|
||||
collection = 'opendb-open-data'
|
||||
} = {}) {
|
||||
this.type = 'db'
|
||||
this.collection = db.collection(collection)
|
||||
}
|
||||
|
||||
_serializeValue(value) {
|
||||
return value === undefined ? null : JSON.stringify(value)
|
||||
}
|
||||
|
||||
_deserializeValue(value) {
|
||||
return value ? JSON.parse(value) : value
|
||||
}
|
||||
|
||||
async set(key, value, duration) {
|
||||
validator.key(key)
|
||||
validator.value(value)
|
||||
validator.duration(duration)
|
||||
value = this._serializeValue(value)
|
||||
await this.collection.doc(key).set({
|
||||
value,
|
||||
expired: duration && duration !== -1 ? Date.now() + (duration * 1000) : -1
|
||||
})
|
||||
}
|
||||
|
||||
async _getWithDuration(key) {
|
||||
const getKeyRes = await this.collection.doc(key).get()
|
||||
const record = getKeyRes.data[0]
|
||||
if (!record) {
|
||||
return {
|
||||
value: null,
|
||||
duration: -2
|
||||
}
|
||||
}
|
||||
const value = this._deserializeValue(record.value)
|
||||
const expired = record.expired
|
||||
if (expired === -1) {
|
||||
return {
|
||||
value,
|
||||
duration: -1
|
||||
}
|
||||
}
|
||||
const duration = expired - Date.now()
|
||||
if (duration <= 0) {
|
||||
await this.remove(key)
|
||||
return {
|
||||
value: null,
|
||||
duration: -2
|
||||
}
|
||||
}
|
||||
return {
|
||||
value,
|
||||
duration: Math.floor(duration / 1000)
|
||||
}
|
||||
}
|
||||
|
||||
async get(key, {
|
||||
withDuration = true
|
||||
} = {}) {
|
||||
const result = await this._getWithDuration(key)
|
||||
if (!withDuration) {
|
||||
delete result.duration
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
async remove(key) {
|
||||
await this.collection.doc(key).remove()
|
||||
}
|
||||
}
|
||||
|
||||
class RedisCache {
|
||||
constructor() {
|
||||
this.type = 'redis'
|
||||
this.redis = uniCloud.redis()
|
||||
}
|
||||
|
||||
_serializeValue(value) {
|
||||
return value === undefined ? null : JSON.stringify(value)
|
||||
}
|
||||
|
||||
_deserializeValue(value) {
|
||||
return value ? JSON.parse(value) : value
|
||||
}
|
||||
|
||||
async set(key, value, duration) {
|
||||
validator.key(key)
|
||||
validator.value(value)
|
||||
validator.duration(duration)
|
||||
value = this._serializeValue(value)
|
||||
if (!duration || duration === -1) {
|
||||
await this.redis.set(key, value)
|
||||
} else {
|
||||
await this.redis.set(key, value, 'EX', duration)
|
||||
}
|
||||
}
|
||||
|
||||
async get(key, {
|
||||
withDuration = false
|
||||
} = {}) {
|
||||
let value = await this.redis.get(key)
|
||||
value = this._deserializeValue(value)
|
||||
if (!withDuration) {
|
||||
return {
|
||||
value
|
||||
}
|
||||
}
|
||||
const durationSecond = await this.redis.ttl(key)
|
||||
let duration
|
||||
switch (durationSecond) {
|
||||
case -1:
|
||||
duration = -1
|
||||
break
|
||||
case -2:
|
||||
duration = -2
|
||||
break
|
||||
default:
|
||||
duration = durationSecond
|
||||
break
|
||||
}
|
||||
return {
|
||||
value,
|
||||
duration
|
||||
}
|
||||
}
|
||||
|
||||
async remove(key) {
|
||||
await this.redis.del(key)
|
||||
}
|
||||
}
|
||||
|
||||
class Cache {
|
||||
constructor({
|
||||
type,
|
||||
collection
|
||||
} = {}) {
|
||||
if (type === 'database') {
|
||||
return new DatabaseCache({
|
||||
collection
|
||||
})
|
||||
} else if (type === 'redis') {
|
||||
return new RedisCache()
|
||||
} else {
|
||||
throw new Error('Invalid cache type')
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class CacheKey {
|
||||
constructor({
|
||||
type,
|
||||
collection,
|
||||
cache,
|
||||
key,
|
||||
fallback
|
||||
} = {}) {
|
||||
this.cache = cache || new Cache({
|
||||
type,
|
||||
collection
|
||||
})
|
||||
this.key = key
|
||||
this.fallback = fallback
|
||||
}
|
||||
|
||||
async set(value, duration) {
|
||||
await this.cache.set(this.key, value, duration)
|
||||
}
|
||||
|
||||
async setWithSync(value, duration, syncMethod) {
|
||||
await Promise.all([
|
||||
this.set(this.key, value, duration),
|
||||
syncMethod(value, duration)
|
||||
])
|
||||
}
|
||||
|
||||
async get() {
|
||||
let {
|
||||
value,
|
||||
duration
|
||||
} = await this.cache.get(this.key)
|
||||
if (value !== null && value !== undefined) {
|
||||
return {
|
||||
value,
|
||||
duration
|
||||
}
|
||||
}
|
||||
if (!this.fallback) {
|
||||
return {
|
||||
value: null,
|
||||
duration: -2
|
||||
}
|
||||
}
|
||||
const fallbackResult = await this.fallback()
|
||||
value = fallbackResult.value
|
||||
duration = fallbackResult.duration
|
||||
if (value !== null && duration !== undefined) {
|
||||
await this.cache.set(this.key, value, duration)
|
||||
}
|
||||
return {
|
||||
value,
|
||||
duration
|
||||
}
|
||||
}
|
||||
|
||||
async remove() {
|
||||
await this.cache.remove(this.key)
|
||||
}
|
||||
}
|
||||
|
||||
class CacheKeyCascade {
|
||||
constructor({
|
||||
layers, // [{cache, type, collection, key}] 从低级到高级排序,[DbCacheKey, RedisCacheKey]
|
||||
fallback
|
||||
} = {}) {
|
||||
this.layers = layers
|
||||
this.cacheLayers = []
|
||||
let lastCacheKey
|
||||
for (let i = 0; i < layers.length; i++) {
|
||||
const {
|
||||
type,
|
||||
cache,
|
||||
collection,
|
||||
key
|
||||
} = layers[i]
|
||||
const lastCacheKeyTemp = lastCacheKey
|
||||
try {
|
||||
const currentCacheKey = new CacheKey({
|
||||
type,
|
||||
collection,
|
||||
cache,
|
||||
key,
|
||||
fallback: i === 0 ? fallback : function() {
|
||||
return lastCacheKeyTemp.get()
|
||||
}
|
||||
})
|
||||
this.cacheLayers.push(currentCacheKey)
|
||||
lastCacheKey = currentCacheKey
|
||||
} catch (e) {}
|
||||
}
|
||||
this.highLevelCache = lastCacheKey
|
||||
}
|
||||
|
||||
async set(value, duration) {
|
||||
return Promise.all(
|
||||
this.cacheLayers.map(item => {
|
||||
return item.set(value, duration)
|
||||
})
|
||||
)
|
||||
}
|
||||
|
||||
async setWithSync(value, duration, syncMethod) {
|
||||
const setPromise = this.cacheLayers.map(item => {
|
||||
return item.set(value, duration)
|
||||
})
|
||||
return Promise.all(
|
||||
[
|
||||
...setPromise,
|
||||
syncMethod(value, duration)
|
||||
]
|
||||
)
|
||||
}
|
||||
|
||||
async get() {
|
||||
return this.highLevelCache.get()
|
||||
}
|
||||
|
||||
async remove() {
|
||||
await Promise.all(
|
||||
this.cacheLayers.map(cacheKeyItem => {
|
||||
return cacheKeyItem.remove()
|
||||
})
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
Cache,
|
||||
DatabaseCache,
|
||||
RedisCache,
|
||||
CacheKey,
|
||||
CacheKeyCascade
|
||||
}
|
@ -0,0 +1,31 @@
|
||||
const Validator = {
|
||||
|
||||
Key(keyArray, parameters) {
|
||||
for (let i = 0; i < keyArray.length; i++) {
|
||||
const keyName = keyArray[i]
|
||||
if (typeof parameters[keyName] !== 'string') {
|
||||
Validator.ThrowNewError(`Invalid ${keyName}`)
|
||||
}
|
||||
if (parameters[keyName].length < 1) {
|
||||
Validator.ThrowNewError(`Invalid ${keyName}`)
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
Value(value) {
|
||||
if (value === undefined) {
|
||||
Validator.ThrowNewError('Invalid Value')
|
||||
}
|
||||
if (typeof value !== 'object') {
|
||||
Validator.ThrowNewError('Invalid Value Type')
|
||||
}
|
||||
},
|
||||
|
||||
ThrowNewError(message) {
|
||||
throw new Error(message)
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
Validator
|
||||
}
|
@ -0,0 +1,203 @@
|
||||
'use strict';
|
||||
|
||||
const crypto = require('crypto')
|
||||
|
||||
const {
|
||||
HTTP_STATUS
|
||||
} = require('./consts.js')
|
||||
|
||||
const {
|
||||
BridgeError
|
||||
} = require('./bridge-error.js')
|
||||
|
||||
class WeixinServer {
|
||||
|
||||
constructor(options = {}) {
|
||||
this._appid = options.appid
|
||||
this._secret = options.secret
|
||||
}
|
||||
|
||||
getAccessToken() {
|
||||
return uniCloud.httpclient.request(WeixinServer.AccessToken_Url, {
|
||||
dataType: 'json',
|
||||
method: 'POST',
|
||||
contentType: 'json',
|
||||
data: {
|
||||
appid: this._appid,
|
||||
secret: this._secret,
|
||||
grant_type: "client_credential"
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
// 使用客户端获取的 code 从微信服务器换取 openid,code 仅可使用一次
|
||||
codeToSession(code) {
|
||||
return uniCloud.httpclient.request(WeixinServer.Code2Session_Url, {
|
||||
dataType: 'json',
|
||||
data: {
|
||||
appid: this._appid,
|
||||
secret: this._secret,
|
||||
js_code: code,
|
||||
grant_type: 'authorization_code'
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
getUserEncryptKey({
|
||||
access_token,
|
||||
openid,
|
||||
session_key
|
||||
}) {
|
||||
console.log(access_token, openid, session_key);
|
||||
const signature = crypto.createHmac('sha256', session_key).update('').digest('hex')
|
||||
return uniCloud.httpclient.request(WeixinServer.User_Encrypt_Key_Url, {
|
||||
dataType: 'json',
|
||||
method: 'POST',
|
||||
dataAsQueryString: true,
|
||||
data: {
|
||||
access_token,
|
||||
openid: openid,
|
||||
signature: signature,
|
||||
sig_method: 'hmac_sha256'
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
getH5AccessToken() {
|
||||
return uniCloud.httpclient.request(WeixinServer.AccessToken_H5_Url, {
|
||||
dataType: 'json',
|
||||
method: 'GET',
|
||||
data: {
|
||||
appid: this._appid,
|
||||
secret: this._secret,
|
||||
grant_type: "client_credential"
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
getH5Ticket(access_token) {
|
||||
return uniCloud.httpclient.request(WeixinServer.Ticket_Url, {
|
||||
dataType: 'json',
|
||||
dataAsQueryString: true,
|
||||
method: 'POST',
|
||||
data: {
|
||||
access_token
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
getH5AccessTokenForEip() {
|
||||
return uniCloud.httpProxyForEip.postForm(WeixinServer.AccessToken_H5_Url, {
|
||||
appid: this._appid,
|
||||
secret: this._secret,
|
||||
grant_type: "client_credential"
|
||||
}, {
|
||||
dataType: 'json'
|
||||
})
|
||||
}
|
||||
|
||||
getH5TicketForEip(access_token) {
|
||||
return uniCloud.httpProxyForEip.postForm(WeixinServer.Ticket_Url, {
|
||||
access_token
|
||||
}, {
|
||||
dataType: 'json',
|
||||
dataAsQueryString: true
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
WeixinServer.AccessToken_Url = 'https://api.weixin.qq.com/cgi-bin/stable_token'
|
||||
WeixinServer.Code2Session_Url = 'https://api.weixin.qq.com/sns/jscode2session'
|
||||
WeixinServer.User_Encrypt_Key_Url = 'https://api.weixin.qq.com/wxa/business/getuserencryptkey'
|
||||
WeixinServer.AccessToken_H5_Url = 'https://api.weixin.qq.com/cgi-bin/token'
|
||||
WeixinServer.Ticket_Url = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket?type=jsapi'
|
||||
|
||||
WeixinServer.GetMPAccessToken = function(options) {
|
||||
return new WeixinServer(options).getAccessToken()
|
||||
}
|
||||
|
||||
WeixinServer.GetCodeToSession = function(options) {
|
||||
return new WeixinServer(options).codeToSession(options.code)
|
||||
}
|
||||
|
||||
WeixinServer.GetUserEncryptKey = function(options) {
|
||||
return new WeixinServer(options).getUserEncryptKey(options)
|
||||
}
|
||||
|
||||
WeixinServer.GetH5AccessToken = function(options) {
|
||||
return new WeixinServer(options).getH5AccessToken()
|
||||
}
|
||||
|
||||
WeixinServer.GetH5Ticket = function(options) {
|
||||
return new WeixinServer(options).getH5Ticket(options.access_token)
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
|
||||
function isAliyun() {
|
||||
return (uniCloud.getCloudInfos()[0].provider === 'aliyun')
|
||||
}
|
||||
|
||||
WeixinServer.GetResponseData = function(response) {
|
||||
console.log("WeixinServer::response", response)
|
||||
|
||||
if (!(response.status === HTTP_STATUS.SUCCESS || response.statusCodeValue === HTTP_STATUS.SUCCESS)) {
|
||||
throw new BridgeError(response.status || response.statusCodeValue, response.status || response.statusCodeValue)
|
||||
}
|
||||
|
||||
const responseData = response.data || response.body
|
||||
|
||||
if (responseData.errcode !== undefined && responseData.errcode !== 0) {
|
||||
throw new BridgeError(responseData.errcode, responseData.errmsg)
|
||||
}
|
||||
|
||||
return responseData
|
||||
}
|
||||
|
||||
WeixinServer.GetMPAccessTokenData = async function(options) {
|
||||
const response = await new WeixinServer(options).getAccessToken()
|
||||
return WeixinServer.GetResponseData(response)
|
||||
}
|
||||
|
||||
WeixinServer.GetCodeToSessionData = async function(options) {
|
||||
const response = await new WeixinServer(options).codeToSession(options.code)
|
||||
return WeixinServer.GetResponseData(response)
|
||||
}
|
||||
|
||||
WeixinServer.GetUserEncryptKeyData = async function(options) {
|
||||
const response = await new WeixinServer(options).getUserEncryptKey(options)
|
||||
return WeixinServer.GetResponseData(response)
|
||||
}
|
||||
|
||||
WeixinServer.GetH5AccessTokenData = async function(options) {
|
||||
const ws = new WeixinServer(options)
|
||||
let response
|
||||
if (isAliyun()) {
|
||||
response = await ws.getH5AccessTokenForEip()
|
||||
if (typeof response === 'string') {
|
||||
response = JSON.parse(response)
|
||||
}
|
||||
} else {
|
||||
response = await ws.getH5AccessToken()
|
||||
}
|
||||
return WeixinServer.GetResponseData(response)
|
||||
}
|
||||
|
||||
WeixinServer.GetH5TicketData = async function(options) {
|
||||
const ws = new WeixinServer(options)
|
||||
let response
|
||||
if (isAliyun()) {
|
||||
response = await ws.getH5TicketForEip(options.access_token)
|
||||
if (typeof response === 'string') {
|
||||
response = JSON.parse(response)
|
||||
}
|
||||
} else {
|
||||
response = await ws.getH5Ticket(options.access_token)
|
||||
}
|
||||
return WeixinServer.GetResponseData(response)
|
||||
}
|
||||
|
||||
|
||||
module.exports = {
|
||||
WeixinServer
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user