学习uni-app之微信登录

转载自知乎王大哥

效果

学习uni-app之微信登录_第1张图片

1.简介

  1. 前端:uniapp以及uniapp官网提供的模板
  2. 后端:springboot,mybatisplus,weixin-java-miniapp
  3. IDE:idea HbuilderX

2.微信登录步骤

  1. 使用button触发微信@getuserinfo获得用户同意
  2. 使用uni.login获取微信提供的code
  3. 再次调用uni.getUserInfo获取用户rawData数据
  4. 调用开发服务器登录接口,并传入code和rawData
  5. 后端使用code和rawData调用微信接口获得session_key、openid
  6. 开发者服务器自己生成一个key(自定义登录状态)与openid、session_key进行关联,并存到数据库中(mysql、redis等)。
  7. 开发者服务器返回生成key(自定义登录状态)到小程序。
  8. 小程序存储key(自定义登录状态)到本地。

3.前端code

3.1 login.vue`

<template>
	<view class="login">
		<view class="content">
			<!-- 头部logo -->
			<view class="header">
				<image :src="logoImage"></image>
			</view>
			<!-- 主体表单 -->
			<view class="main">
				<wInput
					v-model="phoneData"
					type="text"
					maxlength="11"
					placeholder="用户名/电话"
				></wInput>
				<wInput
					v-model="passData"
					type="password"
					maxlength="11"
					placeholder="密码"
				></wInput>
			</view>
			<wButton 
				text="登 录"
				:rotate="isRotate" 
				@click.native="startLogin()"
				class="wbutton"
			></wButton>
			
			<!-- 其他登录 -->
			<view class="other_login cuIcon">
				<view class="login_icon">
					<view class="cuIcon-weixin" @tap="login_weixin">
					</view>
				</view>
				<view class="login_icon">
					<view class="cuIcon-weibo" @tap="login_weibo"></view>
				</view>
				<view class="login_icon">
					<view class="cuIcon-github" @tap="login_github"></view>
				</view>
			</view>
			
			<!-- 底部信息 -->
			<view class="footer">
				<navigator url="forget" open-type="navigate">找回密码</navigator>
				<text>|</text>
				<navigator url="register" open-type="navigate">注册账号</navigator>
			</view>
		</view>
	</view>
</template>

<script>
import wInput from "../../components/watch-login/watch-input.vue"
import wButton from '../../components/watch-login/watch-button.vue'
	export default {
     
		components:{
     
			wInput,wButton
		},
		data() {
     
			return {
     
				//logo图片 base64
				logoImage: '',
				phoneData:'', //用户/电话
				passData:'', //密码
				isRotate: false, //是否加载旋转
			}
		},
		methods: {
     
			login_weixin() {
     
				//微信登录
				uni.showToast({
     
					icon: 'none',
					position: 'bottom',
					title: '...'
				});
				uni.navigateTo({
     
					url:"authorization"
				})
			}
		}
	}
</script>

<style>
	@import url("../../components/watch-login/css/icon.css");
	@import url("./css/main.css");
</style>

3.2 authorization.vue

<template>
	<view>
		<!-- #ifdef MP-WEIXIN -->
		<view v-if="isCanUse">
			<view>
				<view class="header"><image :src="logoImage"></image></view>
				<view class="content">
					<view>申请获取以下权限</view>
					<text>获得你的公开信息(昵称,头像、地区等)</text>
				</view>

				<button class="wbutton" open-type="getUserInfo" withCredentials="true" lang="zh_CN" @getuserinfo="wxLogin">授权登录</button>
			</view>
		</view>
		<!-- #endif -->
	</view>
</template>

<script>
export default {
     
	data() {
     
		return {
     
			logoImage:
				'',
			logining: false,
			isCanUse: true
		};
	},
	methods: {
     
		wxLogin(e) {
     
			
			const that = this;
			that.logining = true;
			let userInfo = e.detail.userInfo;
			uni.showLoading({
     
			    title: '登录中'
			});
			uni.login({
     
				provider: 'weixin',
				success: login_res => {
     
					let code = login_res.code;
					uni.getUserInfo({
     
						success(info_res) {
     
							console.log(info_res);
							uni.request({
     
								url: 'http://localhost:80/user/wxlogin',
								method: 'POST',
								header: {
     
									'content-type': 'application/x-www-form-urlencoded'
								},
								data: {
     
									code: code,
									rawData: info_res.rawData
								},
								success(res) {
     
									uni.hideLoading()
									if (res.data.code == 200) {
     
										uni.navigateTo({
     
											url:"../index/index"
										})
									} else {
     
										console.log('服务器异常');
									}
								},
								fail(error) {
     
									uni.hideLoading()
									console.log(error);
								}
							});
							//uni.hideLoading();
							//uni.navigateBack();
						}
					});
				}
			});
		}
	}
};
</script>

<style>
.header {
     
	margin: 90rpx 0 90rpx 50rpx;
	border-bottom: 1px solid #ccc;
	text-align: center;
	width: 650rpx;
	height: 300rpx;
	line-height: 450rpx;
}

.header image {
     
	width: 200rpx;
	height: 200rpx;
}

.content {
     
	margin-left: 50rpx;
	margin-bottom: 90rpx;
}

.content text {
     
	display: block;
	color: #9d9d9d;
	margin-top: 40rpx;
}

.bottom {
     
	border-radius: 80rpx;
	margin: 70rpx 50rpx;
	font-size: 35rpx;
}
</style>

4.后端

4.1 pom

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    4.0.0
    
        org.springframework.boot
        spring-boot-starter-parent
        2.2.7.RELEASE
         <!-- lookup parent from repository -->
    
    com.zhaoyu
    uniapp-demo
    0.0.1-SNAPSHOT
    uniapp-demo
    Demo project for Spring Boot

    
        1.8
    

    
        
            org.springframework.boot
            spring-boot-starter-web
        
        
            org.mybatis
            mybatis-typehandlers-jsr310
            1.0.1
        
        
            com.baomidou
            mybatis-plus-boot-starter
            2.3
        
        
            com.alibaba
            fastjson
            1.2.4
        
<!--        -->
<!--            org.mybatis.spring.boot-->
<!--            mybatis-spring-boot-starter-->
<!--            2.1.2-->
<!--        -->

        
            mysql
            mysql-connector-java
            runtime
        
        
            org.projectlombok
            lombok
            true
        
        
        
            com.github.binarywang
            weixin-java-miniapp
            3.7.0
        


        
            org.springframework.boot
            spring-boot-configuration-processor
            true
        
        
            org.springframework.boot
            spring-boot-starter-test
            test
            
                
                    org.junit.vintage
                    junit-vintage-engine
                
            
        
    

    
        
            
                org.springframework.boot
                spring-boot-maven-plugin
            
        
    



4.2 controller

package com.zhaoyu.uniappdemo.controller;

import cn.binarywang.wx.miniapp.api.WxMaService;
import cn.binarywang.wx.miniapp.bean.WxMaJscode2SessionResult;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.zhaoyu.uniappdemo.config.WxMaConfiguration;
import com.zhaoyu.uniappdemo.context.bean.User;
import com.zhaoyu.uniappdemo.context.builder.UserBuiler;
import com.zhaoyu.uniappdemo.context.response.R;
import com.zhaoyu.uniappdemo.mapper.UserMapper;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import me.chanjar.weixin.common.error.WxErrorException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import java.util.Date;
import java.util.UUID;

/**
 * 用于测试web项目
 *
 * @author zhaoyu
 * @version 1.0.0
 * @date 2020-05-12 14:59
 */
@RestController
@RequestMapping("/user")
@RequiredArgsConstructor(onConstructor = @__(@Autowired))
@Slf4j
public class HelloWorldController {
     
    private static final String APP_ID="wx87aaf8a080545584";
    private static final String APP_SECRET="c623c5facacfac51415ad3f14e7800cd";
    private final UserMapper userMapper;
    /**
     * Description: 登录接口
     * @param code 
     * @param rawData
     * @return java.lang.String
     * @author zhaoyu
     * @date 2020-05-12 16:07:43
     */
   @PostMapping("/wxlogin")
   public R wxlogin(
           @RequestParam(value = "code",required = false) String code,
           @RequestParam(value = "rawData",required = false) String rawData){
     
       final WxMaJscode2SessionResult session;
       final String sessionKey,openId,nickName,avatarUrl;
       final WxMaService wxService = WxMaConfiguration.getMaService(APP_ID);
       final String skey = UUID.randomUUID().toString();
       try {
     
           JSONObject rawDataJson = JSON.parseObject(rawData);
           nickName=rawDataJson.getString("nickName");
           avatarUrl=rawDataJson.getString("avatarUrl");
           session = wxService.getUserService().getSessionInfo(code);

       } catch (WxErrorException e) {
     
           throw new RuntimeException();
       }
       sessionKey=session.getSessionKey();
       openId=session.getOpenid();
       log.info(sessionKey);
       log.info(openId);
       User user = userMapper.selectById(openId);
       if(user == null){
     
           user = UserBuiler.of().with(userBuiler->{
     
               userBuiler.openId=openId;
               userBuiler.nickName=nickName;
               userBuiler.avatarUrl=avatarUrl;
               userBuiler.skey=skey;
               userBuiler.createTime= new Date();
               userBuiler.lastVisitTime=new Date();
               userBuiler.session_key=sessionKey;
           }).create();
           userMapper.insert(user);
       }else{
     
           user = UserBuiler.of().with(userBuiler->{
     
               userBuiler.nickName=nickName;
               userBuiler.avatarUrl=avatarUrl;
               userBuiler.skey=skey;
               userBuiler.lastVisitTime=new Date();
           }).create();
           userMapper.updateById(user);
       }
       return R.success("登录成功!");
   }
}

5. 数据库

CREATE TABLE `user`  (
  `open_id` varchar(100) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT 'open_id',
  `skey` varchar(100) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT 'skey',
  `create_time` timestamp(0) NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
  `last_visit_time` timestamp(0) NULL DEFAULT CURRENT_TIMESTAMP COMMENT '最后登录时间',
  `session_key` varchar(100) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT 'session_key',
  `avatar_url` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '头像',
  `nick_name` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '网名',
  PRIMARY KEY (`open_id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '微信用户信息' ;

6.现有疑问

  1. 不知道为啥第一次请求了getUserInfo后还要再请求一次获得rawData
  2. 微信的getUserInfo必须用button请求,有其他方式吗
  3. session_key还有什么用?

转载自知乎王大哥

你可能感兴趣的:(uniapp,vue,小程序,java,mysql)