我是靠谱客的博主 温婉皮卡丘,最近开发中收集的这篇文章主要介绍学习uni-app之微信登录,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

转载自知乎王大哥

效果

在这里插入图片描述

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">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.7.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.zhaoyu</groupId>
    <artifactId>uniapp-demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>uniapp-demo</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis-typehandlers-jsr310</artifactId>
            <version>1.0.1</version>
        </dependency>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>2.3</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.4</version>
        </dependency>
<!--        <dependency>-->
<!--            <groupId>org.mybatis.spring.boot</groupId>-->
<!--            <artifactId>mybatis-spring-boot-starter</artifactId>-->
<!--            <version>2.1.2</version>-->
<!--        </dependency>-->

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <!--引入微信小程序SDK-->
        <dependency>
            <groupId>com.github.binarywang</groupId>
            <artifactId>weixin-java-miniapp</artifactId>
            <version>3.7.0</version>
        </dependency>


        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

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还有什么用?

转载自知乎王大哥

最后

以上就是温婉皮卡丘为你收集整理的学习uni-app之微信登录的全部内容,希望文章能够帮你解决学习uni-app之微信登录所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(50)

评论列表共有 0 条评论

立即
投稿
返回
顶部