uni-quick-login.vue 10.1 KB
Newer Older
1 2 3
<template>
	<view class="quick-login-box">
		<view class="item" v-for="(item,index) in servicesList" :key="index"
4
			@click="item.path?to(item.path):login_before(item.id,false)">
5 6 7 8 9 10 11 12 13 14
			<image class="logo" :src="item.logo" mode="widthFix"></image>
			<text class="login-title">{{item.text}}</text>
		</view>
	</view>
</template>
<script>
	import {
		mapGetters,
		mapMutations
	} from 'vuex';
DCloud_JSON's avatar
DCloud_JSON 已提交
15
	//前一个窗口的页面地址。控制点击切换快捷登录方式是创建还是返回
16 17
	import loginSuccess from '@/pages/ucenter/login-page/common/loginSuccess.js';
	export default {
DCloud_JSON's avatar
DCloud_JSON 已提交
18 19 20
		computed: {
			loginConfig() {
				return getApp().globalData.config.router.login
21 22 23
			},
			agreements() {
				return getApp().globalData.config.about.agreements || []
DCloud_JSON's avatar
DCloud_JSON 已提交
24 25
			}
		},
26 27 28
		data() {
			return {
				servicesList: [{
DCloud_JSON's avatar
DCloud_JSON 已提交
29
						"id": "username",
DCloud_JSON's avatar
DCloud_JSON 已提交
30
						"text": "账号登录",
31 32 33 34
						"logo": "/static/uni-quick-login/user.png",
						"path": "/pages/ucenter/login-page/pwd-login/pwd-login"
					},
					{
DCloud_JSON's avatar
DCloud_JSON 已提交
35
						"id": "smsCode",
36 37
						"text": "短信验证码",
						"logo": "/static/uni-quick-login/sms.png",
38
						"path": "/pages/ucenter/login-page/index/index?type=smsCode"
39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
					},
					{
						"id": "weixin",
						"text": "微信登录",
						"logo": "/static/uni-quick-login/wechat.png",
					},
					{
						"id": "apple",
						"text": "苹果登录",
						"logo": "/static/uni-quick-login/apple.png",
					},
					{
						"id": "univerify",
						"text": "一键登录",
						"logo": "/static/uni-quick-login/univerify.png",
					},
					{
						"id": "qq",
						"text": "QQ登录", //暂未提供该登录方式的接口示例
						"logo": "/static/uni-quick-login/univerify.png",
					},
					{
						"id": "xiaomi",
						"text": "小米登录", //暂未提供该登录方式的接口示例
						"logo": "/static/uni-quick-login/univerify.png",
					},
					{
						"id": "sinaweibo",
						"text": "微博登录", //暂未提供该登录方式的接口示例
						"logo": "/static/uni-quick-login/univerify.png",
69
					}
DCloud_JSON's avatar
DCloud_JSON 已提交
70
				],
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
				oauthServices: [],
				config: {},
				univerifyStyle: { //一键登录弹出窗的样式配置参数
					"fullScreen": true, // 是否全屏显示,true表示全屏模式,false表示非全屏模式,默认值为false。
					"backgroundColor": "#ffffff", // 授权页面背景颜色,默认值:#ffffff
					"buttons": { // 自定义登陆按钮
						"iconWidth": "45px", // 图标宽度(高度等比例缩放) 默认值:45px
						"list": []
					},
					"privacyTerms": {
						"defaultCheckBoxState": false, // 条款勾选框初始状态 默认值: true   
						"textColor": "#BBBBBB", // 文字颜色 默认值:#BBBBBB  
						"termsColor": "#5496E3", //  协议文字颜色 默认值: #5496E3  
						"prefix": "我已阅读并同意", // 条款前的文案 默认值:“我已阅读并同意”  
						"suffix": "并使用本机号码登录", // 条款后的文案 默认值:“并使用本机号码登录”  
						"privacyItems": []
					}
88
				}
89
			}
90 91 92 93 94 95
		},
		watch: {
			agree(agree) {
				this.univerifyStyle.privacyTerms.defaultCheckBoxState = agree
			}
		},
96
		props: {
97 98 99
			agree: {
				type: Boolean,
				default () {
100 101
					return false
				}
102 103
			}
		},
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
		async created() {
			let servicesList = this.servicesList
			//去掉配置中不存在的
			servicesList = servicesList.filter(item => this.loginConfig.includes(item.id))
			//处理一键登录
			if(this.loginConfig.includes('univerify')){
				this.univerifyStyle.privacyTerms.privacyItems = this.agreements
				//设置一键登录功能底下的快捷登陆按钮
				servicesList.forEach(({id,logo}) => {
					if (id != 'univerify') {
						this.univerifyStyle.buttons.list.push({
							"iconPath": logo,
							"provider": id
						})
					}
119
				})
120
			}
121 122 123 124 125 126 127 128
			//如果当前页面为默认登陆界面。当前第一优先级的“微信和苹果登陆”要隐藏,因为他已经被渲染在默认登陆界面顶部
			if (
				this.getRoute(1) == '/pages/ucenter/login-page/index/index' &&
				['weixin','apple'].includes(this.loginConfig[0])
			) {
				servicesList = servicesList.filter(item => item.id != this.loginConfig[0])
			}
			//去掉当前页面对应的登录选项
129 130 131
			this.servicesList = servicesList.filter(item => {
				let path = item.path?item.path.split('?')[0]:'';
				return path != this.getRoute(1)
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
			})
			console.log('servicesList', servicesList, this.servicesList);
		},
		mounted() {
			// #ifdef APP-PLUS
			plus.oauth.getServices(oauthServices => {
				this.oauthServices = oauthServices
			}, err => {
				uni.showModal({
					title: '获取服务供应商失败:' + JSON.stringify(err),
					showCancel: false,
					confirmText: '知道了'
				});
				console.error('获取服务供应商失败:' + JSON.stringify(err));
			})
			// #endif
148 149 150 151 152 153 154
		},
		methods: {
			...mapMutations({
				setUserInfo: 'user/login'
			}),
			getRoute(n = 0) {
				let pages = getCurrentPages();
DCloud_JSON's avatar
DCloud_JSON 已提交
155
				// console.log('route-pages-length', pages.length);
156 157 158 159 160 161
				if (n > pages.length) {
					return ''
				}
				return '/' + pages[pages.length - n].route
			},
			to(path) {
162
				// console.log('比较', this.getRoute(2), path)
163 164 165 166
				if (this.getRoute(2) == path) { // 控制路由是重新打开还是返回,避免重复打开页面
					uni.navigateBack();
				} else {
					uni.navigateTo({
167 168
						url: path,
						animationType: 'slide-in-left'
169 170 171
					})
				}
			},
DCloud_JSON's avatar
DCloud_JSON 已提交
172
			login_before(type, navigateBack = true) {
173
				if (!this.agree && type != 'univerify') {
174 175 176 177 178
					return uni.showToast({
						title: '你未同意隐私政策协议',
						icon: 'none'
					});
				}
179 180 181
				uni.showLoading({
					mask: true
				})
182
				// console.log(arguments);
183
				let oauthService = this.oauthServices.find((service) => service.id == type)
184
				// console.log(type);
185 186

				// #ifdef APP-PLUS
DCloud_JSON's avatar
DCloud_JSON 已提交
187
				//请勿直接使用前端获取的unionid或openid直接用于登录,前端的数据都是不可靠的
188 189 190 191
				if (type == 'weixin') {
					return oauthService.authorize(({
							code
						}) => {
192
							// console.log(code);
193
							this.login({
194 195 196 197 198
								code
							}, type)
						},
						err => {
							uni.hideLoading()
199
							console.error(err);
200 201 202
							uni.showModal({
								content: JSON.stringify(err),
								showCancel: false
203
							});
204 205 206 207 208 209 210
						})
				}
				// #endif

				uni.login({
					"provider": type,
					"univerifyStyle": this.univerifyStyle,
DCloud_JSON's avatar
DCloud_JSON 已提交
211
					complete: (e) => {
212
						console.log(e);
DCloud_JSON's avatar
DCloud_JSON 已提交
213
					},
214 215 216 217 218
					success: async e => {
						console.log(e);
						if (type == 'apple') {
							let res = await this.getUserInfo({
								provider: "apple"
DCloud_JSON's avatar
DCloud_JSON 已提交
219
							})
220
							uni.hideLoading()
221 222
							Object.assign(e.authResult, res.userInfo)
						}
DCloud_JSON's avatar
DCloud_JSON 已提交
223
						// #ifdef MP-WEIXIN
DCloud_JSON's avatar
DCloud_JSON 已提交
224
						if (type == 'weixin') {
DCloud_JSON's avatar
DCloud_JSON 已提交
225 226 227 228 229
							return this.login({
								code: e.code
							}, type)
						}
						// #endif
230
						this.login(e.authResult, type)
231 232 233
					},
					fail: (err) => {
						uni.hideLoading()
234
						console.error(err);
235 236

						if (type == 'univerify') {
237
							if (err.metadata && err.metadata.error_data) {
238
								uni.showToast({
DCloud_JSON's avatar
DCloud_JSON 已提交
239
									title: "一键登录:" + err.metadata.error_data,
240 241 242
									icon: 'none'
								});
							}
243 244 245 246 247 248
							if (err.errMsg) {
								uni.showToast({
									title: "一键登录:" + err.errMsg,
									icon: 'none'
								});
							}
249 250
							switch (err.errCode) {
								case 30002:
DCloud_JSON's avatar
DCloud_JSON 已提交
251
									console.log('在一键登录界面,点击其他登录方式');
252 253
									break;
								case 30003:
DCloud_JSON's avatar
DCloud_JSON 已提交
254
									console.log('关闭了登录');
255 256 257 258
									if (navigateBack) {
										uni.navigateBack()
									}
									break;
259 260 261 262 263 264 265
								case 30006:
									uni.showModal({
										title: "登录服务初始化错误",
										content: err.metadata.error_data,
										showCancel: false,
										confirmText: '知道了',
									});
266
									break;
267 268 269 270 271
								case "30008":
									uni.showToast({
										title: '点击了第三方登陆',
										icon: 'none'
									});
272 273 274 275 276 277 278 279 280
									console.log('点击了第三方登陆,provider:', err.provider);
									let {
										path
									} = this.servicesList.find(item => item.id == err.provider) || {}
									console.log('path', path);
									if (path && path != this.getRoute(1)) { //存在路径,且并不是当前已经打开的路径
										this.to(path)
									} else {
										this.login_before(err.provider)
281 282 283 284
									}
									break;
								default:
									console.log(9527, err);
285 286 287 288 289 290
									break;
							}
						}
					}
				})
			},
DCloud_JSON's avatar
DCloud_JSON 已提交
291
			login(params, type) { //联网验证登录
292
				console.log({params,type});
293
				let action = 'loginBy' + type.trim().toLowerCase().replace(type[0], type[0].toUpperCase())
294 295 296
				uniCloud.callFunction({
					name: 'uni-id-cf',
					data: {
297 298
						action,
						params
299 300 301 302
					},
					success: ({
						result
					}) => {
303
						console.log("login-result",result);
304 305 306 307 308 309 310 311
						if (result.code === 0) {
							if (type == 'univerify') {
								uni.closeAuthView()
							}
							uni.hideLoading()
							loginSuccess(result)
							delete result.userInfo.token
							this.setUserInfo(result.userInfo)
312 313 314 315 316
						} else {
							uni.showModal({
								content: result.msg,
								showCancel: false
							});
317 318 319 320 321
						}
					},
					complete: () => {
						uni.hideLoading()
					}
322
				})
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344
			},
			async getUserInfo(e) {
				return new Promise((resolve, reject) => {
					uni.getUserInfo({
						...e,
						success: (res) => {
							resolve(res);
						},
						fail: (err) => {
							uni.showModal({
								content: JSON.stringify(err),
								showCancel: false
							});
							reject(err);
						}
					})
				})
			}
		}
	}
</script>

DCloud_JSON's avatar
DCloud_JSON 已提交
345 346
<style lang="scss" scoped>
	@import '@/common/all-flex.css';
347 348 349
	.quick-login-box {
		flex-direction: row;
		width: 750rpx;
DCloud_JSON's avatar
DCloud_JSON 已提交
350 351 352
		justify-content: space-around;
		position: fixed;
		bottom: 10rpx;
DCloud_JSON's avatar
DCloud_JSON 已提交
353
		left: 0;
354 355 356 357 358 359 360 361 362 363 364 365 366 367
	}

	.item {
		flex-direction: column;
		justify-content: center;
		align-items: center;
		height: 200rpx;
	}

	.logo {
		width: 60rpx;
		height: 60rpx;
	}

DCloud_JSON's avatar
DCloud_JSON 已提交
368
	.login-title {
369 370 371
		margin-top: 4px;
		font-size: 26rpx;
	}
372
</style>