123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394 |
- <template>
- <div class="wechat-auth-container">
- <div class="auth-content">
- <!-- <div class="logo">
- <img src="@/assets/images/logo.png" alt="Logo" />
- </div> -->
- <div class="auth-status" v-if="authStatus === 'loading'">
- <van-loading type="spinner" color="#1989fa" />
- <p>正在授权中...</p>
- </div>
- <!--
- <div class="auth-status" v-else-if="authStatus === 'success'">
- <van-icon name="success" color="#07c160" size="48" />
- <p>授权成功</p>
- <p class="user-info">{{ userInfo.nickname }}</p>
- </div> -->
- <div class="auth-status" v-else-if="authStatus === 'error'">
- <van-icon name="cross" color="#ee0a24" size="48" />
- <p>授权失败</p>
- <p class="error-message">{{ errorMessage }}</p>
- <van-button type="primary" @click="retryAuth">重新授权</van-button>
- </div>
- <div class="auth-status" v-else>
- <van-icon name="wechat" color="#07c160" size="48" />
- <p>微信授权</p>
- <p class="auth-desc">请点击下方按钮进行微信授权</p>
- <van-button type="primary" @click="startAuth">{{ Loading ? '授权中...' : '开始授权' }}</van-button>
- </div>
- </div>
- </div>
- </template>
- <script>
- import { mapActions } from 'vuex'
- import { wechatAuth, wechatRegister } from '@/api/wechat'
- import Cookies from 'js-cookie'
- export default {
- name: 'WechatAuth',
- data() {
- return {
- authStatus: 'init', // init, loading, success, error
- userInfo: {},
- errorMessage: '',
- redirect: '',
- code: '',
- Loading: false
- }
- },
- computed: {
- // 微信授权配置
- wechatConfig() {
- // 直接返回对象,不需要再返回函数
- const redirectUri = this.$route.query.url ? decodeURIComponent(this.$route.query.url) : null
- console.log('redirectUri from query:', redirectUri)
- if (redirectUri && redirectUri !== 'undefined') {
- Cookies.set('redirectUri', redirectUri)
- }
- return {
- appId: 'wx2d69589899b7ecd6',
- // appId: 'wx3a8dd28881c2c41f',
- scope: 'snsapi_userinfo', // snsapi_base 或 snsapi_userinfo
- redirectUri: encodeURIComponent(window.location.origin + '/login'),
- state: 'wechat_auth'
- }
- }
- },
- watch: {
- $route: {
- handler: function(route) {
- // 从URL中提取参数,包括#之前的参数
- const url = window.location.href
- const urlParams = new URLSearchParams(url.split('?')[1]?.split('#')[0] || '')
- this.code = urlParams.get('code') || route.query.code
- this.state = urlParams.get('state') || route.query.state
- // redirect参数可能在#之后
- this.redirect = route.query.redirect || this.extractRedirectFromHash()
- console.log('URL参数解析:', {
- url: url,
- code: this.code,
- state: this.state,
- redirect: this.redirect,
- routeQuery: route.query
- })
- },
- immediate: true
- }
- },
- methods: {
- ...mapActions({
- login: 'user/login',
- getInfo: 'user/getInfo'
- }),
- // 开始微信授权
- startAuth() {
- if (this.Loading) {
- return
- }
- this.authStatus = 'loading'
- // 构建微信授权URL
- const authUrl = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${this.wechatConfig.appId}&redirect_uri=${this.wechatConfig.redirectUri}&response_type=code&scope=${this.wechatConfig.scope}&state=${this.wechatConfig.state}#wechat_redirect`
- // 跳转到微信授权页面
- window.location.href = authUrl
- },
- // 处理微信授权回调
- async handleAuthCallback() {
- if (!this.code) {
- this.authStatus = 'error'
- this.errorMessage = '未获取到授权码'
- return
- }
- try {
- this.authStatus = 'loading'
- this.Loading = true
- // 调用后端接口,用code换取access_token和用户信息
- const result = await this.exchangeCodeForToken(this.code)
- console.log(result)
- if (result.openid) {
- this.userInfo = result
- this.authStatus = 'success'
- const res = await wechatRegister({
- username: result.userInfo.nickname,
- mail: result.openid + '@wechat.com',
- openid: result.openid + '-wechat'
- })
- console.log(res)
- // console.log(res)
- const loginData = JSON.stringify({
- openid: result.openid + '-wechat',
- edu: true
- })
- const myHeaders = new Headers()
- myHeaders.append('Content-Type', 'application/json')
- const requestOptionsLogin = {
- method: 'POST',
- headers: myHeaders,
- body: loginData,
- redirect: 'follow',
- credentials: 'include'
- }
- fetch('https://beta.api.cocorobo.cn/api/user', requestOptionsLogin)
- .then(response => response.text())
- .then(async() => {
- // this.$router.replace('/login')
- await this.login()
- let redirect_uri = Cookies.get('redirectUri')
- if (!redirect_uri || redirect_uri === 'undefined') {
- redirect_uri = ''
- }
- this.Loading = false
- console.log(redirect_uri)
- if (redirect_uri) {
- // Cookies.remove('redirectUri')
- window.location.href = redirect_uri
- } else {
- this.$router.replace('/appStoreCopy')
- }
- })
- .catch(error => {
- this.Loading = false
- console.log('error', error)
- setTimeout(() => {
- this.$router.replace('/login2')
- }, 2000)
- })
- } else {
- this.Loading = false
- throw new Error(result.message || '授权失败')
- }
- } catch (error) {
- console.error('微信授权失败:', error)
- this.authStatus = 'error'
- this.Loading = false
- this.errorMessage = error.message || '授权失败,请重试'
- // 如果授权失败,跳转到登录页面
- setTimeout(() => {
- this.$router.replace('/login2')
- }, 2000)
- }
- },
- // 用code换取access_token和用户信息
- async exchangeCodeForToken(code) {
- try {
- const response = await wechatAuth({
- code: code,
- state: this.state
- })
- // 根据实际API响应格式调整
- if (response && response.data) {
- return response.data
- } else if (response && response.success !== undefined) {
- return response
- } else {
- // 如果API还没有实现,模拟成功响应
- return {
- success: true,
- userInfo: {
- openid: 'test_openid_' + Date.now(),
- nickname: '微信用户',
- headimgurl: '',
- unionid: 'test_unionid_' + Date.now()
- }
- }
- }
- } catch (error) {
- console.error('API调用失败:', error)
- }
- },
- // 跳转到目标页面
- redirectToTarget() {
- let targetPath = '/'
- // 使用已解析的redirect参数
- if (this.redirect) {
- targetPath = this.redirect
- }
- console.log('跳转到目标页面:', targetPath)
- this.$router.replace(targetPath)
- },
- // 重试授权
- retryAuth() {
- this.authStatus = 'init'
- this.errorMessage = ''
- this.startAuth()
- },
- // 检查是否在微信浏览器中
- isWechatBrowser() {
- const ua = navigator.userAgent.toLowerCase()
- return ua.indexOf('micromessenger') !== -1
- },
- // 从URL hash中提取redirect参数
- extractRedirectFromHash() {
- const hash = window.location.hash
- if (hash && hash.includes('redirect=')) {
- const match = hash.match(/redirect=([^&]+)/)
- if (match) {
- return decodeURIComponent(match[1])
- }
- }
- return null
- },
- async getLogin() {
- const userid = await this.login()
- let redirect_uri = Cookies.get('redirectUri')
- if (!redirect_uri || redirect_uri === 'undefined') {
- redirect_uri = ''
- }
- console.log(redirect_uri)
- if (userid) {
- if (redirect_uri) {
- // Cookies.remove('redirectUri')
- window.location.href = redirect_uri
- } else {
- this.$router.replace('/appStoreCopy')
- }
- return '1'
- }
- return '2'
- // eduGet().then(res => {})
- }
- },
- mounted() {
- // 检查是否在微信浏览器中
- if (!this.isWechatBrowser()) {
- // this.authStatus = 'error'
- // this.errorMessage = '请在微信中打开此页面'
- Cookies.set('isWeChat', '1')
- this.$router.replace('/login2')
- return
- }else{
- Cookies.set('isWeChat', '2')
- }
- let type = this.getLogin();
- if(type == '1'){
- return;
- }
- // 如果有code参数,说明是从微信授权回调过来的
- if (this.code) {
- this.handleAuthCallback()
- } else {
- // 没有code参数,开始授权流程
- this.startAuth()
- }
- }
- }
- </script>
- <style lang="scss" scoped>
- .wechat-auth-container {
- min-height: 100vh;
- background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
- display: flex;
- align-items: center;
- justify-content: center;
- padding: 20px;
- .auth-content {
- background: white;
- border-radius: 16px;
- padding: 40px 30px;
- text-align: center;
- box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1);
- max-width: 400px;
- width: 100%;
- .logo {
- margin-bottom: 30px;
- img {
- width: 80px;
- height: 80px;
- border-radius: 50%;
- }
- }
- .auth-status {
- .van-loading,
- .van-icon {
- margin-bottom: 20px;
- }
- p {
- margin: 10px 0;
- font-size: 16px;
- color: #333;
- &.user-info {
- font-size: 14px;
- color: #666;
- margin-top: 10px;
- }
- &.auth-desc {
- font-size: 14px;
- color: #999;
- margin: 15px 0 25px;
- }
- &.error-message {
- font-size: 14px;
- color: #ee0a24;
- margin: 15px 0 25px;
- }
- }
- .van-button {
- margin-top: 20px;
- width: 200px;
- height: 44px;
- border-radius: 22px;
- }
- }
- }
- }
- // 响应式设计
- @media (max-width: 480px) {
- .wechat-auth-container {
- padding: 10px;
- .auth-content {
- padding: 30px 20px;
- .auth-status {
- .van-button {
- width: 100%;
- }
- }
- }
- }
- }
- </style>
|