01-登录-路由与组件
目的:完成登录页的路由与组件配置
组件:src/views/login/index.vue
<template>
<div class="page-login">
Login
</div>
</template>
<script>
export default {
name: 'PageLogin'
}
</script>
<style scoped lang="less"></style>
路由:src/router/index.js
一级路由规则
+const Login = () => import('@/views/login/index')
...
const routes = [
...
+ { path: '/login', component: Login }
]
链接:src/components/app-topnav.vue
+<li><RouterLink to="/login">请先登录</RouterLink></li>
02-登录-基础布局
目的:完成登录页基础结构布局。
结构分析:
- 登录头部 LoginHeader 需要留出副标题插槽
- 登录主体 Login
- 登录底部 LoginFooter
具体代码:
- 头部组件
src/views/login/components/login-header.vue
<template>
<header class="login-header">
<div class="container">
<h1 class="logo"><RouterLink to="/">小兔鲜</RouterLink></h1>
<h3 class="sub"><slot /></h3>
<RouterLink class="entry" to="/">
进入网站首页
<i class="iconfont icon-angle-right"></i>
<i class="iconfont icon-angle-right"></i>
</RouterLink>
</div>
</header>
</template>
<script>
export default {
name: 'LoginHeader'
}
</script>
<style scoped lang='less'>
.login-header {
background: #fff;
border-bottom: 1px solid #e4e4e4;
.container {
display: flex;
align-items: flex-end;
justify-content: space-between;
}
.logo {
width: 200px;
a {
display: block;
height: 132px;
width: 100%;
text-indent: -9999px;
background: url(../../../assets/images/logo.png) no-repeat center 18px / contain;
}
}
.sub {
flex: 1;
font-size: 24px;
font-weight: normal;
margin-bottom: 38px;
margin-left: 20px;
color: #666;
}
.entry {
width: 120px;
margin-bottom: 38px;
font-size: 16px;
i {
font-size: 14px;
color: @xtxColor;
letter-spacing: -5px;
}
}
}
</style>
- 底部组件
src/views/login/components/login-footer.vue
<template>
<footer class="login-footer">
<div class='container'>
<p>
<a href="javascript:;">关于我们</a>
<a href="javascript:;">帮助中心</a>
<a href="javascript:;">售后服务</a>
<a href="javascript:;">配送与验收</a>
<a href="javascript:;">商务合作</a>
<a href="javascript:;">搜索推荐</a>
<a href="javascript:;">友情链接</a>
</p>
<p>CopyRight © 小兔鲜儿</p>
</div>
</footer>
</template>
<script>
export default {
name: 'LoginFooter'
}
</script>
<style scoped lang='less'>
.login-footer {
padding: 30px 0 50px;
background: #fff;
p {
text-align: center;
color: #999;
padding-top: 20px;
a {
line-height: 1;
padding:0 10px;
color: #999;
display: inline-block;
~ a {
border-left: 1px solid #ccc;
}
}
}
}
</style>
- 主体组件
src/views/login/index.vue
<template>
<LoginHeader>欢迎登录</LoginHeader>
<section class="login-section">
<div class="wrapper">
<nav>
<a href="javascript:;">账户登录</a>
<a href="javascript:;">扫码登录</a>
</nav>
</div>
</section>
<LoginFooter />
</template>
<script>
import LoginHeader from './components/login-header'
import LoginFooter from './components/login-footer'
export default {
name: 'Login',
components: {
LoginHeader,
LoginFooter
}
}
</script>
<style scoped lang='less'>
.login-section {
background: url(../../assets/images/login-bg.png) no-repeat center / cover;
height: 488px;
position: relative;
.wrapper {
width: 380px;
background: #fff;
min-height: 400px;
position: absolute;
left: 50%;
top: 54px;
transform: translate3d(100px,0,0);
box-shadow: 0 0 10px rgba(0,0,0,.15);
nav {
height: 55px;
border-bottom: 1px solid #f5f5f5;
display: flex;
padding: 0 40px;
text-align: right;
align-items: center;
a {
flex: 1;
line-height: 1;
display: inline-block;
font-size: 18px;
position: relative;
&:first-child {
border-right: 1px solid #f5f5f5;
text-align: left;
}
&.active{
color: @xtxColor;
font-weight: bold;
}
}
}
}
}
</style>
03-登录-切换效果
目的:完成点击nav按钮切换,帐号登录和扫码登录。
大致步骤:
- 声明activeName数据提供模版使用
- 点击按钮设置activeName和绑定class属性
- 控制帐号登录盒子和扫码登录盒子显示隐藏
落的代码:
src/views/login/index.vue
+import { ref } from 'vue'
export default {
name: 'PageLogin',
components: { LoginHeader, LoginFooter },
setup () {
+ const activeName = ref('account')
+ return { activeName }
}
}
<div class="wrapper">
<nav>
<a @click="activeName='account'" :class="{active:activeName==='account'}" href="javascript:;">账户登录</a>
<a @click="activeName='qrcode'" :class="{active:activeName==='qrcode'}" href="javascript:;">扫码登录</a>
</nav>
<!-- 表单 -->
<div v-if="activeName==='account'" class="account-box">表单</div>
<!-- 二维码 -->
<div v-if="activeName==='qrcode'" class="qrcode-box">
<img src="@/assets/images/qrcode.jpg" alt="">
<p>打开 <a href="javascript:;">小兔鲜App</a> 扫码登录</p>
</div>
</div>
// 二维码容器
.qrcode-box {
text-align: center;
padding-top: 40px;
p {
margin-top: 20px;
a {
color: @xtxColor;
font-size: 16px;
}
}
}
04-登录-表单组件
目的:完成表单布局和帐号登录,短信登录切换。
大致步骤:
- 提取组件
- 通过isMsgLogin切换短信登录
- 通过form.isAgree绑定同意协议
- 使用组件
落的代码:
1.定义组件 src/views/login/component/login-form.vue
基础结构:
<template>
<div class="account-box">
<div class="toggle">
<a @click="isMsgLogin=false" href="javascript:;" v-if="isMsgLogin">
<i class="iconfont icon-user"></i> 使用账号登录
</a>
<a @click="isMsgLogin=true" href="javascript:;" v-else>
<i class="iconfont icon-msg"></i> 使用短信登录
</a>
</div>
<div class="form">
<template v-if="!isMsgLogin">
<div class="form-item">
<div class="input">
<i class="iconfont icon-user"></i>
<input type="text" placeholder="请输入用户名或手机号" />
</div>
<!-- <div class="error"><i class="iconfont icon-warning" />请输入手机号</div> -->
</div>
<div class="form-item">
<div class="input">
<i class="iconfont icon-lock"></i>
<input type="password" placeholder="请输入密码">
</div>
</div>
</template>
<template v-else>
<div class="form-item">
<div class="input">
<i class="iconfont icon-user"></i>
<input type="text" placeholder="请输入手机号" />
</div>
</div>
<div class="form-item">
<div class="input">
<i class="iconfont icon-code"></i>
<input type="password" placeholder="请输入验证码">
<span class="code">发送验证码</span>
</div>
</div>
</template>
<div class="form-item">
<div class="agree">
<XtxCheckbox v-model="form.isAgree" />
<span>我已同意</span>
<a href="javascript:;">《隐私条款》</a>
<span>和</span>
<a href="javascript:;">《服务条款》</a>
</div>
</div>
<a href="javascript:;" class="btn">登录</a>
</div>
<div class="action">
<img src="https://qzonestyle.gtimg.cn/qzone/vas/opensns/res/img/Connect_logo_7.png" alt="">
<div class="url">
<a href="javascript:;">忘记密码</a>
<a href="javascript:;">免费注册</a>
</div>
</div>
</div>
</template>
基础样式:
// 账号容器
.account-box {
.toggle {
padding: 15px 40px;
text-align: right;
a {
color: @xtxColor;
i {
font-size: 14px;
}
}
}
.form {
padding: 0 40px;
&-item {
margin-bottom: 28px;
.input {
position: relative;
height: 36px;
> i {
width: 34px;
height: 34px;
background: #cfcdcd;
color: #fff;
position: absolute;
left: 1px;
top: 1px;
text-align: center;
line-height: 34px;
font-size: 18px;
}
input {
padding-left: 44px;
border: 1px solid #cfcdcd;
height: 36px;
line-height: 36px;
width: 100%;
&.error {
border-color: @priceColor;
}
&.active,&:focus {
border-color: @xtxColor;
}
}
.code {
position: absolute;
right: 1px;
top: 1px;
text-align: center;
line-height: 34px;
font-size: 14px;
background: #f5f5f5;
color: #666;
width: 90px;
height: 34px;
cursor: pointer;
}
}
> .error {
position: absolute;
font-size: 12px;
line-height: 28px;
color: @priceColor;
i {
font-size: 14px;
margin-right: 2px;
}
}
}
.agree {
a {
color: #069;
}
}
.btn {
display: block;
width: 100%;
height: 40px;
color: #fff;
text-align: center;
line-height: 40px;
background: @xtxColor;
&.disabled {
background: #cfcdcd;
}
}
}
.action {
padding: 20px 40px;
display: flex;
justify-content: space-between;
align-items: center;
.url {
a {
color: #999;
margin-left: 10px;
}
}
}
}
依赖数据:
import { reactive, ref } from 'vue'
export default {
name: 'LoginForm',
setup () {
// 是否短信登录
const isMsgLogin = ref(false)
// 表单信息对象
const form = reactive({
isAgree: true
})
return { isMsgLogin, form }
}
}
2.使用组件 src/views/login/index.vue
<!-- 表单 -->
+ <LoginForm v-if="activeName==='account'"></LoginForm>
+import LoginForm from './components/login-form'
import { ref } from 'vue'
export default {
name: 'PageLogin',
components: {
LoginHeader,
LoginFooter,
+ LoginForm
},