822 lines
19 KiB
Vue
822 lines
19 KiB
Vue
<template>
|
||
<view class="login-container">
|
||
<view class="login-form">
|
||
<view class="logo-area">
|
||
<image class="logo" src="/packageRc/static/pageBg.png" mode="aspectFit"></image>
|
||
<view class="title">就业服务系统</view>
|
||
</view>
|
||
|
||
<view class="form-area">
|
||
<view class="form-item">
|
||
<text class="label">用户名</text>
|
||
<input
|
||
v-model="loginForm.username"
|
||
class="input"
|
||
placeholder="请输入用户名"
|
||
type="text"
|
||
auto-complete="off"
|
||
/>
|
||
</view>
|
||
|
||
<view class="form-item">
|
||
<text class="label">密码</text>
|
||
<input
|
||
v-model="loginForm.password"
|
||
class="input"
|
||
placeholder="请输入密码"
|
||
password
|
||
type="text"
|
||
/>
|
||
</view>
|
||
|
||
|
||
|
||
<button class="login-btn" @click="useVerify">登录</button>
|
||
</view>
|
||
</view>
|
||
|
||
<!-- 滑块验证组件 -->
|
||
<view v-if="showVerify" class="verify-mask">
|
||
<view class="verify-container">
|
||
<view class="verify-header">
|
||
<text class="verify-title">安全验证</text>
|
||
<text class="verify-close" @click="closeVerify">×</text>
|
||
</view>
|
||
<view class="verify-body">
|
||
<!-- 滑块图片区域 -->
|
||
<view class="verify-image-container">
|
||
<view class="verify-image-wrapper">
|
||
<image :src="verifyImage.backImgBase64" class="verify-background-image" mode="aspectFit"></image>
|
||
<image
|
||
v-if="verifyImage.blockImgBase64"
|
||
:src="verifyImage.blockImgBase64"
|
||
class="verify-block-image"
|
||
:style="{ left: sliderWidth + 'px' }"
|
||
mode="aspectFit"
|
||
></image>
|
||
</view>
|
||
<!-- 刷新按钮 -->
|
||
<view v-if="showRefresh" class="verify-refresh" @click.stop="getVerifyImage">
|
||
<text>↻</text>
|
||
</view>
|
||
<!-- 提示文字 -->
|
||
<view v-if="tipWords" class="verify-tip" :class="{ 'tip-error': !passFlag }">
|
||
{{ tipWords }}
|
||
</view>
|
||
</view>
|
||
<Verify
|
||
@success="handleSubmit"
|
||
:mode="'pop'"
|
||
:captchaType="'blockPuzzle'"
|
||
:blockSize="{ width: '47px' }"
|
||
:imgSize="{ width: '300px', height: '155px' }"
|
||
ref="verifyRef"
|
||
></Verify>
|
||
<!-- 滑块区域 -->
|
||
<view class="verify-slider">
|
||
<view class="slider-track">
|
||
<view
|
||
class="slider-fill"
|
||
:style="{ width: sliderWidth + 'px' }"
|
||
:class="{ 'slider-success': passFlag }"
|
||
></view>
|
||
<view
|
||
class="slider-thumb"
|
||
:style="{ left: sliderWidth + 'px' }"
|
||
:class="{ 'slider-success': passFlag }"
|
||
@touchstart="handleTouchStart"
|
||
@touchmove="handleTouchMove"
|
||
@touchend="handleTouchEnd"
|
||
@mousedown="handleMouseDown"
|
||
@mousemove="handleMouseMove"
|
||
@mouseup="handleMouseUp"
|
||
@mouseleave="handleMouseUp"
|
||
>
|
||
<text class="slider-icon">{{ sliderText }}</text>
|
||
</view>
|
||
</view>
|
||
<text
|
||
class="slider-text"
|
||
:class="{ 'slider-success': passFlag }"
|
||
>{{ sliderStatusText }}</text>
|
||
</view>
|
||
</view>
|
||
</view>
|
||
</view>
|
||
</view>
|
||
</template>
|
||
|
||
<script setup>
|
||
import Verify from '../../components/verifition/Verify.vue';
|
||
import { ref, onMounted, onUnmounted } from 'vue';
|
||
import { login } from '../../api/login.js';
|
||
import { setToken, saveUserInfo, getUserInfo } from '../../utils/auth.js';
|
||
import { encrypt, decrypt } from '../../utils/sm2Encrypt.js';
|
||
import { reqGet, reqCheck } from '../../utils/captchaApi.js';
|
||
|
||
// 登录表单数据
|
||
const loginForm = ref({
|
||
username: '',
|
||
password: '',
|
||
rememberMe: false,
|
||
code: '', // 用于存储滑块验证的code
|
||
captchaVerification: '' // 用于滑块验证结果
|
||
});
|
||
|
||
// 滑块验证相关状态
|
||
const showVerify = ref(false);
|
||
// Verify组件引用 - 在Composition API中使用ref来引用组件
|
||
const verifyRef = ref(null);
|
||
const verifyImage = ref({
|
||
backImgBase64: '',
|
||
blockImgBase64: '',
|
||
token: ''
|
||
});
|
||
const sliderWidth = ref(0);
|
||
const startX = ref(0);
|
||
const isDragging = ref(false);
|
||
const sliderText = ref('→');
|
||
const sliderStatusText = ref('向右滑动完成验证');
|
||
const clientUid = ref('');
|
||
const secretKey = ref('');
|
||
const spinning = ref(false);
|
||
const isEnd = ref(false);
|
||
const showRefresh = ref(true);
|
||
const passFlag = ref(false);
|
||
const tipWords = ref('');
|
||
|
||
// 生成clientUid
|
||
const generateClientUid = () => {
|
||
const s = [];
|
||
const hexDigits = '0123456789abcdef';
|
||
for (let i = 0; i < 36; i++) {
|
||
s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);
|
||
}
|
||
s[14] = '4';
|
||
s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1);
|
||
s[8] = s[13] = s[18] = s[23] = '-';
|
||
return 'slider' + '-' + s.join('');
|
||
};
|
||
|
||
// 打开滑块验证
|
||
const useVerify = async () => {
|
||
// 使用verifyRef.value访问组件实例,而不是this.$refs.verify
|
||
if (verifyRef.value) {
|
||
verifyRef.value.show();
|
||
} else {
|
||
console.error('Verify组件未正确加载');
|
||
}
|
||
// 简单表单验证
|
||
if (!loginForm.value.username) {
|
||
return uni.showToast({ title: '请输入用户名', icon: 'none' });
|
||
}
|
||
if (!loginForm.value.password) {
|
||
return uni.showToast({ title: '请输入密码', icon: 'none' });
|
||
}
|
||
|
||
try {
|
||
// 生成clientUid
|
||
if (!clientUid.value) {
|
||
clientUid.value = generateClientUid();
|
||
// 存储到localStorage
|
||
uni.setStorageSync('slider', clientUid.value);
|
||
}
|
||
|
||
// 获取验证图片
|
||
await getVerifyImage();
|
||
// 显示验证弹窗
|
||
// showVerify.value = true;
|
||
// 重置滑块状态
|
||
resetSlider();
|
||
} catch (error) {
|
||
console.error('获取验证图片失败:', error);
|
||
uni.showToast({
|
||
title: '获取验证图片失败,请稍后重试',
|
||
icon: 'none'
|
||
});
|
||
}
|
||
};
|
||
|
||
// 获取验证图片
|
||
const getVerifyImage = async () => {
|
||
try {
|
||
spinning.value = true;
|
||
// 从localStorage获取clientUid,如果没有则生成新的
|
||
const storedClientUid = uni.getStorageSync('slider') || generateClientUid();
|
||
if (!clientUid.value) {
|
||
clientUid.value = storedClientUid;
|
||
}
|
||
|
||
const res = await reqGet({
|
||
captchaType: 'blockPuzzle',
|
||
clientUid: clientUid.value,
|
||
ts: Date.now()
|
||
});
|
||
|
||
spinning.value = false;
|
||
|
||
// 处理不同的响应格式
|
||
const isSuccess = res.code === 200 || res.repCode === '0000';
|
||
const responseData = res.data || res.repData;
|
||
|
||
if (isSuccess && responseData) {
|
||
verifyImage.value = {
|
||
backImgBase64: 'data:image/png;base64,' + responseData.originalImageBase64,
|
||
blockImgBase64: 'data:image/png;base64,' + responseData.jigsawImageBase64,
|
||
token: responseData.token
|
||
};
|
||
// 保存secretKey用于加密
|
||
secretKey.value = responseData.secretKey || '';
|
||
} else {
|
||
throw new Error(res.msg || res.repMsg || '获取验证图片失败');
|
||
}
|
||
} catch (error) {
|
||
spinning.value = false;
|
||
console.error('获取验证图片错误:', error);
|
||
throw error;
|
||
}
|
||
};
|
||
|
||
// 关闭验证
|
||
const closeVerify = () => {
|
||
showVerify.value = false;
|
||
resetSlider();
|
||
};
|
||
|
||
// 重置滑块
|
||
const resetSlider = () => {
|
||
sliderWidth.value = 0;
|
||
sliderText.value = '→';
|
||
sliderStatusText.value = '向右滑动完成验证';
|
||
isDragging.value = false;
|
||
isEnd.value = false;
|
||
showRefresh.value = true;
|
||
passFlag.value = false;
|
||
tipWords.value = '';
|
||
};
|
||
|
||
// 处理触摸开始
|
||
const handleTouchStart = (e) => {
|
||
if (isEnd.value) return;
|
||
startX.value = e.touches[0].clientX;
|
||
isDragging.value = true;
|
||
sliderText.value = '';
|
||
};
|
||
|
||
// 处理触摸移动
|
||
const handleTouchMove = (e) => {
|
||
if (!isDragging.value || isEnd.value) return;
|
||
|
||
// 阻止默认行为,防止页面滚动
|
||
e.preventDefault();
|
||
|
||
const moveX = e.touches[0].clientX - startX.value;
|
||
|
||
// 限制滑块移动范围
|
||
if (moveX >= 0) {
|
||
// 获取滑块轨道宽度 - 修正选择器
|
||
const trackElement = document.querySelector('.verify-slider-track');
|
||
const trackWidth = trackElement ? trackElement.offsetWidth : 300; // 增大默认宽度
|
||
const maxMoveDistance = trackWidth - 60; // 减去滑块按钮宽度(60px)
|
||
|
||
sliderWidth.value = Math.min(moveX, maxMoveDistance);
|
||
} else {
|
||
sliderWidth.value = 0;
|
||
}
|
||
};
|
||
|
||
// 在移动端添加全局触摸事件处理,防止验证弹窗打开时页面滚动
|
||
document.addEventListener('touchmove', (e) => {
|
||
const verifyModal = document.querySelector('.verify-mask');
|
||
if (showVerify.value && verifyModal) {
|
||
e.preventDefault();
|
||
}
|
||
}, { passive: false });
|
||
|
||
// 处理触摸结束
|
||
const handleTouchEnd = async () => {
|
||
if (!isDragging.value) return;
|
||
|
||
isDragging.value = false;
|
||
|
||
// 验证滑块位置
|
||
await verifySlider();
|
||
};
|
||
|
||
// 处理鼠标按下
|
||
const handleMouseDown = (e) => {
|
||
startX.value = e.clientX;
|
||
isDragging.value = true;
|
||
};
|
||
|
||
// 处理鼠标移动
|
||
const handleMouseMove = (e) => {
|
||
if (!isDragging.value) return;
|
||
|
||
const moveX = e.clientX - startX.value;
|
||
if (moveX >= 0 && moveX <= 260) {
|
||
sliderWidth.value = moveX;
|
||
}
|
||
};
|
||
|
||
// 处理鼠标释放
|
||
const handleMouseUp = async () => {
|
||
if (!isDragging.value) return;
|
||
|
||
isDragging.value = false;
|
||
|
||
// 验证滑块位置
|
||
await verifySlider();
|
||
};
|
||
|
||
// AES加密函数
|
||
const aesEncrypt = (word, keyWord = 'XwKsGlMcdPMEhR1B') => {
|
||
// 由于uni-app环境,我们使用一个简化的加密实现
|
||
// 在实际项目中,应该引入crypto-js库
|
||
try {
|
||
// 这里只是为了保持接口一致,实际加密需要引入crypto-js
|
||
return word;
|
||
} catch (e) {
|
||
console.error('加密失败:', e);
|
||
return word;
|
||
}
|
||
};
|
||
|
||
// 验证滑块位置
|
||
const verifySlider = async () => {
|
||
try {
|
||
if (isEnd.value) return;
|
||
|
||
isDragging.value = false;
|
||
|
||
// 计算移动距离
|
||
const moveLeftDistance = sliderWidth.value;
|
||
|
||
// 准备验证数据
|
||
const pointJson = JSON.stringify({ x: moveLeftDistance, y: 5.0 });
|
||
|
||
const verifyData = {
|
||
captchaType: 'blockPuzzle',
|
||
pointJson: secretKey.value ? aesEncrypt(pointJson, secretKey.value) : pointJson,
|
||
token: verifyImage.value.token,
|
||
clientUid: clientUid.value,
|
||
ts: Date.now()
|
||
};
|
||
|
||
// 调用验证接口
|
||
const res = await reqCheck(verifyData);
|
||
|
||
// 处理不同的响应格式
|
||
const isSuccess = res.code === 200 || res.repCode === '0000';
|
||
const responseData = res.data || res.repData;
|
||
|
||
if (isSuccess) {
|
||
// 验证成功
|
||
sliderStatusText.value = '验证通过';
|
||
sliderText.value = '✓';
|
||
passFlag.value = true;
|
||
isEnd.value = true;
|
||
showRefresh.value = false;
|
||
|
||
// 生成captchaVerification
|
||
const captchaVerification = secretKey.value
|
||
? aesEncrypt(verifyImage.value.token + "---" + pointJson, secretKey.value)
|
||
: verifyImage.value.token + "---" + pointJson;
|
||
|
||
// 保存验证码信息
|
||
loginForm.value.code = responseData.code || '';
|
||
loginForm.value.captchaVerification = responseData.captchaVerification || captchaVerification;
|
||
|
||
// 延迟关闭验证弹窗并执行登录
|
||
setTimeout(() => {
|
||
showVerify.value = false;
|
||
handleLogin();
|
||
}, 1000);
|
||
} else {
|
||
// 验证失败
|
||
sliderStatusText.value = '验证失败,请重试';
|
||
passFlag.value = false;
|
||
tipWords.value = res.msg || res.repMsg || '验证失败';
|
||
|
||
setTimeout(() => {
|
||
resetSlider();
|
||
getVerifyImage(); // 重新获取验证图片
|
||
tipWords.value = '';
|
||
}, 1000);
|
||
}
|
||
} catch (error) {
|
||
console.error('滑块验证失败:', error);
|
||
sliderStatusText.value = '验证失败,请重试';
|
||
passFlag.value = false;
|
||
tipWords.value = '网络错误,请重试';
|
||
|
||
setTimeout(() => {
|
||
resetSlider();
|
||
getVerifyImage();
|
||
tipWords.value = '';
|
||
}, 1000);
|
||
}
|
||
};
|
||
|
||
// 登录处理
|
||
const handleLogin = async () => {
|
||
try {
|
||
// 显示加载状态
|
||
uni.showLoading({
|
||
title: '登录中',
|
||
mask: true
|
||
});
|
||
|
||
// 调用登录接口,添加clientUid参数
|
||
const loginData = {
|
||
username: loginForm.value.username,
|
||
password: encrypt(loginForm.value.password),
|
||
code: loginForm.value.code,
|
||
captchaVerification: loginForm.value.captchaVerification,
|
||
clientUid: clientUid.value
|
||
};
|
||
|
||
const res = await login(loginData);
|
||
|
||
// 保存token
|
||
setToken(res.token || res.data.token);
|
||
|
||
// 保存用户信息
|
||
saveUserInfo(loginForm.value.username, loginForm.value.password, loginForm.value.rememberMe);
|
||
|
||
// 登录成功提示
|
||
uni.showToast({
|
||
title: '登录成功',
|
||
icon: 'success'
|
||
});
|
||
|
||
// 跳转到首页或之前的页面
|
||
uni.navigateBack();
|
||
} catch (error) {
|
||
console.error('登录失败:', error);
|
||
uni.showToast({
|
||
title: error.response?.data?.msg || '登录失败,请检查账号密码',
|
||
icon: 'none'
|
||
});
|
||
} finally {
|
||
uni.hideLoading();
|
||
}
|
||
};
|
||
|
||
// 刷新滑块验证
|
||
const refreshVerify = () => {
|
||
resetSlider();
|
||
getVerifyImage();
|
||
};
|
||
|
||
// 页面加载时的初始化
|
||
onMounted(() => {
|
||
// 获取保存的用户信息
|
||
const userInfo = getUserInfo();
|
||
if (userInfo.rememberMe) {
|
||
loginForm.value.username = userInfo.username;
|
||
loginForm.value.password = userInfo.password;
|
||
loginForm.value.rememberMe = true;
|
||
}
|
||
|
||
// 从localStorage获取clientUid
|
||
const storedClientUid = uni.getStorageSync('slider');
|
||
if (storedClientUid) {
|
||
clientUid.value = storedClientUid;
|
||
}
|
||
});
|
||
|
||
// 页面卸载时清理事件监听器
|
||
onUnmounted(() => {
|
||
// 清理工作
|
||
});
|
||
</script>
|
||
|
||
<style scoped>
|
||
.login-container {
|
||
width: 100%;
|
||
height: 100vh;
|
||
display: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
background-color: #f5f5f5;
|
||
}
|
||
|
||
.login-form {
|
||
width: 80%;
|
||
max-width: 500rpx;
|
||
padding: 40rpx;
|
||
background-color: #fff;
|
||
border-radius: 16rpx;
|
||
box-shadow: 0 4rpx 20rpx rgba(0, 0, 0, 0.1);
|
||
}
|
||
|
||
.logo-area {
|
||
text-align: center;
|
||
margin-bottom: 40rpx;
|
||
}
|
||
|
||
.logo {
|
||
width: 160rpx;
|
||
height: 160rpx;
|
||
margin-bottom: 20rpx;
|
||
}
|
||
|
||
.title {
|
||
font-size: 36rpx;
|
||
font-weight: bold;
|
||
color: #1a62ce;
|
||
}
|
||
|
||
.form-area {
|
||
width: 100%;
|
||
}
|
||
|
||
.form-item {
|
||
margin-bottom: 30rpx;
|
||
}
|
||
|
||
.label {
|
||
display: block;
|
||
font-size: 28rpx;
|
||
color: #333;
|
||
margin-bottom: 10rpx;
|
||
}
|
||
|
||
.input {
|
||
width: 100%;
|
||
height: 80rpx;
|
||
padding: 0 20rpx;
|
||
border: 1rpx solid #ddd;
|
||
border-radius: 8rpx;
|
||
font-size: 28rpx;
|
||
color: #333;
|
||
}
|
||
|
||
.remember-area {
|
||
display: flex;
|
||
align-items: center;
|
||
margin-bottom: 30rpx;
|
||
justify-content: flex-end;
|
||
}
|
||
|
||
.remember-checkbox {
|
||
margin-right: 10rpx;
|
||
transform: scale(0.8);
|
||
}
|
||
|
||
.remember-text {
|
||
font-size: 28rpx;
|
||
color: #666;
|
||
}
|
||
|
||
.login-btn {
|
||
width: 100%;
|
||
height: 80rpx;
|
||
line-height: 80rpx;
|
||
background-color: #1a62ce;
|
||
color: #fff;
|
||
border-radius: 40rpx;
|
||
font-size: 32rpx;
|
||
margin-top: 10rpx;
|
||
border: none;
|
||
}
|
||
|
||
/* 滑块验证相关样式 */
|
||
.verify-mask {
|
||
position: fixed;
|
||
top: 0;
|
||
left: 0;
|
||
width: 100%;
|
||
height: 100%;
|
||
background-color: rgba(0, 0, 0, 0.5);
|
||
display: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
z-index: 9999;
|
||
}
|
||
|
||
.verify-container {
|
||
width: 80%;
|
||
max-width: 340px;
|
||
background-color: #fff;
|
||
border-radius: 12rpx;
|
||
overflow: hidden;
|
||
animation: fadeIn 0.3s ease-out;
|
||
}
|
||
|
||
.verify-header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
padding: 20rpx;
|
||
border-bottom: 1rpx solid #eee;
|
||
}
|
||
|
||
.verify-title {
|
||
font-size: 32rpx;
|
||
font-weight: 500;
|
||
color: #333;
|
||
}
|
||
|
||
.verify-close {
|
||
font-size: 48rpx;
|
||
color: #999;
|
||
cursor: pointer;
|
||
width: 40rpx;
|
||
height: 40rpx;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
border-radius: 50%;
|
||
transition: all 0.3s;
|
||
}
|
||
|
||
.verify-close:hover {
|
||
background-color: #f0f0f0;
|
||
color: #666;
|
||
}
|
||
|
||
.verify-body {
|
||
padding: 20rpx;
|
||
}
|
||
|
||
.verify-image-container {
|
||
position: relative;
|
||
width: 100%;
|
||
height: 200px;
|
||
margin-bottom: 20rpx;
|
||
border: 1rpx solid #eee;
|
||
border-radius: 8rpx;
|
||
overflow: hidden;
|
||
background-color: #fafafa;
|
||
}
|
||
|
||
.verify-image-wrapper {
|
||
position: relative;
|
||
width: 100%;
|
||
height: 100%;
|
||
}
|
||
|
||
.verify-background-image {
|
||
width: 100%;
|
||
height: 100%;
|
||
object-fit: cover;
|
||
}
|
||
|
||
.verify-block-image {
|
||
position: absolute;
|
||
top: 70px; /* 设置固定的垂直位置,确保与缺口对齐 */
|
||
width: 80px; /* 大幅增大宽度 */
|
||
height: 80px; /* 大幅增大高度 */
|
||
cursor: move;
|
||
transition: left 0.05s;
|
||
box-shadow: 0 0 8px rgba(0, 0, 0, 0.3);
|
||
border-radius: 4px;
|
||
z-index: 10;
|
||
}
|
||
|
||
/* 刷新按钮 */
|
||
.verify-refresh {
|
||
position: absolute;
|
||
top: 10px;
|
||
right: 10px;
|
||
width: 30px;
|
||
height: 30px;
|
||
background-color: rgba(255, 255, 255, 0.8);
|
||
border-radius: 50%;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
cursor: pointer;
|
||
transition: all 0.3s;
|
||
font-size: 18px;
|
||
color: #666;
|
||
}
|
||
|
||
.verify-refresh:hover {
|
||
background-color: #fff;
|
||
transform: rotate(180deg);
|
||
}
|
||
|
||
/* 提示文字 */
|
||
.verify-tip {
|
||
position: absolute;
|
||
bottom: 10px;
|
||
left: 50%;
|
||
transform: translateX(-50%);
|
||
padding: 5px 15px;
|
||
background-color: rgba(0, 0, 0, 0.6);
|
||
color: #fff;
|
||
border-radius: 15px;
|
||
font-size: 12px;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.verify-tip.tip-error {
|
||
background-color: #f56c6c;
|
||
}
|
||
|
||
.verify-slider {
|
||
width: 100%;
|
||
}
|
||
|
||
.slider-track {
|
||
position: relative;
|
||
width: 100%;
|
||
height: 40rpx;
|
||
background-color: #f5f5f5;
|
||
border-radius: 20rpx;
|
||
overflow: hidden;
|
||
transition: background-color 0.3s;
|
||
}
|
||
|
||
.slider-track:hover {
|
||
background-color: #e6e6e6;
|
||
}
|
||
|
||
.slider-fill {
|
||
position: absolute;
|
||
top: 0;
|
||
left: 0;
|
||
height: 100%;
|
||
background-color: #1a62ce;
|
||
transition: width 0.05s, background-color 0.3s;
|
||
}
|
||
|
||
.slider-thumb {
|
||
position: absolute;
|
||
top: -5px;
|
||
width: 48px;
|
||
height: 48px;
|
||
background-color: #fff;
|
||
border: 1rpx solid #ddd;
|
||
border-radius: 50%;
|
||
display: flex;
|
||
justify-content: center;
|
||
align-items: center;
|
||
cursor: pointer;
|
||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||
transition: left 0.05s, all 0.3s;
|
||
z-index: 1;
|
||
}
|
||
|
||
.slider-thumb:hover {
|
||
box-shadow: 0 3px 15px rgba(26, 98, 206, 0.3);
|
||
}
|
||
|
||
.slider-icon {
|
||
font-size: 24rpx;
|
||
color: #1a62ce;
|
||
}
|
||
|
||
.slider-text {
|
||
margin-top: 10rpx;
|
||
font-size: 24rpx;
|
||
color: #666;
|
||
text-align: center;
|
||
transition: color 0.3s;
|
||
}
|
||
|
||
/* 验证成功状态 */
|
||
.slider-fill.slider-success {
|
||
background-color: #67c23a;
|
||
}
|
||
|
||
.slider-thumb.slider-success {
|
||
border-color: #67c23a;
|
||
color: #67c23a;
|
||
}
|
||
|
||
.slider-text.slider-success {
|
||
color: #67c23a;
|
||
}
|
||
|
||
/* 动画效果 */
|
||
@keyframes fadeIn {
|
||
from {
|
||
opacity: 0;
|
||
transform: translateY(-20px);
|
||
}
|
||
to {
|
||
opacity: 1;
|
||
transform: translateY(0);
|
||
}
|
||
}
|
||
|
||
/* 适配移动端 */
|
||
@media (max-width: 480px) {
|
||
.verify-container {
|
||
width: 90%;
|
||
max-width: 320px;
|
||
}
|
||
|
||
.verify-image-container {
|
||
height: 180px;
|
||
}
|
||
|
||
.verify-body {
|
||
padding: 15px;
|
||
}
|
||
}
|
||
</style> |