赞
踩
import { useStore } from '@/store' import { InjectionKey } from 'vue' export type SetLoginCallback = (params: { loginCallbackMethod: Function loginCallbackParams: any[] }) => void export const SetLoginCallbackKey: InjectionKey<SetLoginCallback> = Symbol('setLoginCallback') type routerLink = 'navigateTo' | 'redirectTo' | 'reLaunch' | 'switchTab' export default function () { const _this = (getCurrentInstance() as any).ctx const store = useStore() const setLoginCallback = inject(SetLoginCallbackKey) const navigateTo = (url: string, events?: any): any => { return new Promise((resolve, reject) => { if (!url) { return reject(new Error('路由为空')) } else if (/^wx/.test(url)) { return navigateToMiniProgram({ appId: url, }) } else if (/^http/.test(url)) { url = `/pages/web-view/index?url=${url}` } uni.navigateTo({ url, events, success: (res) => { resolve(res.eventChannel) }, fail: () => { /*尝试跳转到tab页面*/ return switchTab(url) }, }) }) } /*跳转tab*/ const switchTab = (url: string): any => { return new Promise((resolve, reject) => { uni.switchTab({ url, success: (res) => resolve(res), fail: (err) => reject(err), }) }) } /*关闭当前跳转其他*/ const redirectTo = (url: string): any => { return new Promise((resolve, reject) => { uni.redirectTo({ url, success: (res) => resolve(res), fail: (err) => reject(err), }) }) } /*关闭所有跳转其他*/ const reLaunch = (url: string): any => { return new Promise((resolve, reject) => { uni.reLaunch({ url, success: (res) => resolve(res), fail: (err) => reject(err), }) }) } /*跳转到其他小程序*/ const navigateToMiniProgram = (options: UniApp.NavigateToMiniProgramOptions): any => { return new Promise((resolve, reject) => { uni.navigateToMiniProgram({ ...options, success: (res) => resolve(res), fail: (err) => reject(err), }) }) } /*检测登录 并跳转页面*/ const linkTo = (url: string, type: routerLink = 'navigateTo') => { if (!!store.token) { switch (type) { case 'navigateTo': navigateTo(url) break case 'redirectTo': redirectTo(url) break case 'reLaunch': reLaunch(url) break case 'switchTab': switchTab(url) break } } else { openLoginPopup({ loginCallbackMethod: linkTo, loginCallbackParams: [url, type], }) } } /*检测登录 登录成功后回调*/ const checkLogin = (callback: any) => { if (!!store.token) { callback() } else { openLoginPopup({ loginCallbackMethod: callback, loginCallbackParams: [], }) } } const openLoginPopup: SetLoginCallback = (params) => { if (setLoginCallback) { setLoginCallback(params) } else { // 对于页面来说,登录框组件是页面的一个组件,所以登录框组件中的provide无法被页面获取到 // 所以这里直接通过页面实例,调用登录框组件的方法 // 切记将登录框组件作为整个页面的最顶级组件,使用方式下面有示例 if (!_this.parent && _this.$children.length > 0) { _this.$children[0].openPopup && _this.$children[0].openPopup(params) } } } return { navigateTo, switchTab, redirectTo, navigateToMiniProgram, linkTo, checkLogin, } }
<template> <slot></slot> <uni-popup ref="popupRef" type="center" :is-mask-click="false"> <view class="sww-login-box"> <view class="clb-popup-box clb-popup-info"> <view class="clb-popup-content"> <view class="clb-info-avatar"> <view class="clb-info-icon" /> </view> <view class="clb-info-title">您还未登录</view> <view class="clb-info-desc">请先登录再进行此操作</view> <view class="clb-btn-list"> <button :loading="loading" class="clb-btn-item" @click="getUserInfo">立即登录</button> <button :disabled="loading" class="clb-btn-item" @click="closePopup">暂不登录</button> </view> </view> </view> </view> </uni-popup> </template> <script lang="ts" setup> import { SetLoginCallback, SetLoginCallbackKey } from '@/hooks/routerLink' import { useStore } from '@/store' defineOptions({ name: 'DslLogin', }) /** * @desc: Types */ /** * @desc: Ref */ const popupRef = ref<any>() /** * @desc: Hooks */ const store = useStore() /** * @desc: Data */ const loginCallbackMethod = ref<any>(null) const loginCallbackParams = ref<any[]>([]) const loading = ref(false) /** * @desc: Computed */ provide(SetLoginCallbackKey, (params) => { openPopup(params) }) /** * @desc: 方法 */ const closePopup = () => { loginCallbackMethod.value = null loginCallbackParams.value = [] popupRef.value?.close() } const openPopup: SetLoginCallback = (params) => { loginCallbackMethod.value = params.loginCallbackMethod loginCallbackParams.value = params.loginCallbackParams popupRef.value?.open() } const getUserInfo = () => { loading.value = true store .login() .then(() => { loading.value = false try { loginCallbackMethod.value(...loginCallbackParams.value) } catch (e) { console.log(e) } closePopup() }) .catch((err) => { console.log(err) loading.value = false }) } /** * @desc: 生命周期 */ defineExpose({ openPopup, }) </script> <style scoped lang="scss"> .sww-login-box { display: flex; width: 100%; height: 100%; align-items: center; justify-content: center; } .clb-popup-box { position: relative; width: 626rpx; } .clb-popup-content { display: flex; width: 100%; padding: 0 47rpx; background-color: #fff; border-radius: 16rpx; flex-direction: column; align-items: center; } .clb-popup-title { padding-top: 40rpx; font-size: 28rpx; } .clb-popup-avatar { width: 90rpx; height: 90rpx; margin: 40rpx 0; overflow: hidden; background-size: 100%; border-radius: 50%; } .clb-popup-desc { font-size: 24rpx; font-weight: 400; color: #666; } .clb-phone-btn { display: flex; width: 100%; height: 82rpx; margin: 50rpx 0 30rpx; font-size: 30rpx; font-weight: 500; line-height: 1; color: #fff; background-color: #6cadff; border-radius: 41rpx; align-items: center; justify-content: center; } /* 用户登录 */ .clb-popup-info { padding-top: 120rpx; } .clb-popup-info .clb-popup-content { padding-top: 210rpx; } .clb-info-avatar { position: absolute; top: 0; left: 50%; padding: 20rpx; background-color: #fff; border-radius: 50%; transform: translate(-50%, 0); } .clb-info-icon { width: 287rpx; height: 287rpx; background-image: url(''); background-size: 100%; } .clb-info-title { margin-bottom: 10rpx; font-size: 32rpx; font-weight: bold; color: #000; } .clb-info-desc { font-size: 28rpx; font-weight: 400; color: #999; } .clb-btn-list { width: 100%; padding: 50rpx 0 30rpx; } .clb-btn-item { display: flex; width: 100%; height: 82rpx; padding: 0; margin: 0; font-size: 30rpx; font-weight: 500; line-height: 1; color: #fff; background-color: #6cadff; border-radius: 41rpx; align-items: center; justify-content: center; } .clb-btn-item::after { border: none; } .clb-btn-item:last-child { color: #6caeff; background-color: inherit; } </style>
<template> <dsl-login> <button @click="router.linkTo('/package/index/address/index')">跳转页面</button > </dsl-login> </template> <script setup lang="ts"> import { useRouter } from '@/hooks' /** * @desc: Types */ /** * @desc: Ref */ /** * @desc: Hooks */ const router = useRouter() </script>
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。