Vue学习之路之登录注册

杭州数澜科技招聘Node.js工程师【可实习】UE/UI设计师【实习】前端React工程师【实习】,联系方式可见评论区

Vue.js + Element UI + MongoDB

P1 安装Vue-CLI

Vue.js文档
利用Vue.js提供的一个官方命令行工具

# 全局安装 vue-cli
$ npm install --global vue-cli
# 创建一个基于 webpack 模板的新项目
$ vue init webpack my-project
# 安装依赖,走你
$ cd my-project
$ npm install
$ npm run dev

Vue.js 主要目录结构

.
├── build           # 一些webpack的文件,配置参数什么的,一般不用动 
├── config          # vue项目的基本配置文件 
├── index.html      # 主页
├── node_modules    # 项目中安装的依赖模块 
├── package.json    # 项目文件,记载着一些命令和依赖还有简要的项目描述信息 
├── README.md 
├── server          # 自己创建的后端文件,可以忽视
├── src             # 源码文件夹,基本上文件都应该放在这里
    ├── App.vue     # App.vue组件 
    ├── assets      # 资源文件夹,里面放一些静态资源 
    ├── components  # 这里放的都是各个组件文件 
    ├── main.js     # 入口文件 
    └── router      # vue-router 路由配置
├── static          # 生成好的文件会放在这个目录下
├── test            # 测试文件夹,测试都写在这里
├── .babelrc        # babel编译参数,vue开发需要babel编译 
├── .gitignore
└── .eslintignore

完成后就可以在/src/components/*.vue模板中写代码,ctrl+s保存后页面会自动刷新,若无效请检查端口是否被占用

P2 安装Element UI

Element UI文档

npm i element-ui -S

完成后在main.js中添加如下代码完整引入Element,就能在/src/components/*.vue模板中使用Element UI的组件

// main.js
import ElementUI from 'element-ui'
import 'element-ui/lib/theme-default/index.css'

Vue.use(ElementUI)

P3 登录注册功能

登录界面

界面代码及注册略

思路

  • 得益于vue的数据双向绑定,让我们不用操作DOM就能边输入边验证字符串合法性
  • 利用axios实现前后端的数据交互
  • 利用jsonwebtoken实现登录验证,结合vue-routerbeforeEnter导航钩子在跳转前拦截验证access_token的有效性

验证输入的账号和密码是否合法(利用elementui的form表单)

export default {
  name: 'register',
  data () {
    var validateUser = (rule, value, cb) => {
      var pattern = /^[\w\u4e00-\u9fa5]{3,10}$/g
      if (value === '') {
        cb(new Error('请输入用户名'))
      } else if (!pattern.test(value)) {
        cb(new Error('请输入3-10个字母/汉字/数字/下划线'))
      } else {
        cb()
      }
    }
    var validatePwd = (rule, value, cb) => {
      var pattern = /^\S{3,20}$/g
      if (value === '') {
        cb(new Error('请输入密码'))
      } else if (!pattern.test(value)) {
        cb(new Error('请输入3-20个非空白字符'))
      } else {
        if (this.registerForm.checkPwd !== '') {
          this.$refs.registerForm.validateField('checkPwd')
        }
        cb()
      }
    }
    var validateCheckPwd = (rule, value, cb) => {
      if (value === '') {
        cb(new Error('请再次输入密码'))
      } else if (value !== this.registerForm.pwd) {
        cb(new Error('两次输入密码不一致!'))
      } else {
        cb()
      }
    }
    return {
      registerForm: {
        userName: '',
        pwd: '',
        checkPwd: ''
      },
      registerRule: {
        userName: [
        { validator: validateUser, trigger: 'blur' }
        ],
        pwd: [
        { validator: validatePwd, trigger: 'blur' }
        ],
        checkPwd: [
        { validator: validateCheckPwd, trigger: 'blur' }
        ]
      }
    }
  },
  methods: {
    submitForm (formName) {
      this.$refs[formName].validate((valid) => {
        if (valid) {
            ...
        } else {
          return false
        }
      })
    }
  }
}

利用axios实现与后端数据的交互

axios文档

Axios.post('http://localhost:3000/register', data)
.then(res => {
    console.log(res.data)
    if (res.data.code === 0) {
        this.$message({
            showClose: true,
            message: '注册成功',
            type: 'success'
        })
        router.push({name: 'Login'})
    } else {
        this.$message({
            showClose: true,
            message: '注册失败',
            type: 'error'
        })
    }
})

使用 JSON WEB Tokens 实现登录验证

由于node后端和vue前端是两个不同的端口(:3000和:8090),对于跨域(我已经允许跨域访问),session和cookie就不要想了,并不是设置一个什么就能解决的,花了一下午才发现,所以使用了token来做api请求,而且还能加密。

【翻译】在Nodejs中使用JSON WEB Tokens

后端处理登录

// sever/db/dbHelper.js
exports.findUser = function(data, cb) {
    User.findOne({
        username: data.usr
    }, function(err, doc) {
        // 用户密码都正确
        // jwt.encode({加密对象, 持续时间}, 密钥字符串)
        entries.data = user
        entries.code = 0
        var time = moment().add(1, 'days').valueOf()
        entries.access_token = jwt.encode({
            iss: user._id,
            exp: time
        }, jwtTokenSecret)
        cb(true, entries)
    })
}

加密后的entries.access_token:

access_token

前端获取到后端传递过来的access_token,将其保存进sessionStorage。这个导航钩子我放在/p路由独享的钩子下,在进入/p/:id前拦截导航,通过axios向后端传递access_token,根据后台返回值判断是否已经登录。

导航钩子传送门
注意router.beforeEach确保要调用next方法,否则钩子就不会被 resolved,但after钩子没有 next方法,不能改变导航

beforeEnter: (to, from, next) => {
    let pattern = /^(\/p)/g
    let token = sessionStorage.getItem('accessToken')   //保存token
    if (pattern.test(to.path)) {
        Axios.post('http://localhost:3000/isLogin', {access_token: token})
        .then(res => {
            if (res.data.code === 0) {
                console.log(from)
                console.log(to)
                next()
            } else {
                router.push({name: 'Login'})
                next()
            }
        })
        .catch(err => {
            console.log(err)
        })
    }
}

后端处理token是否合法

后台获取到传递的token值,利用jwt.decode(token, jwtTokenSecret)对其解码,解码结果就是当初我们加密的对象{iss, exp},首先根据exp判断token是否过期,然后根据_id查询数据库是否有这个用户

// 登录验证
exports.authority = function (req, cb) {
    // JWT 允许客户端使用一下3个方法附加token:
    // 作为请求链接(query)的参数,作为主体的参数(body),
    // 和作为请求头(Header)的参数。
    var token = (req.body && req.body.access_token) || (req.query && req.query.access_token) || req.headers['access-token']
    if (token) {
        try {
            var decoded = jwt.decode(token, jwtTokenSecret) // 解码
            if (decoded.exp <= Date.now()) {    // 判断token是否过期
                entries.code = 99
                cb(false, entries)
            } else {    // 之前加密对象是 user._id
                User.findOne({ _id: decoded.iss }, function(err, user) {
                    if (err) {
                        console.log(err)
                    } else if (user !== null) {
                        entries.code = 0
                        cb(true, entries)
                    }
                })
            }
        } catch (err) {
            console.log(err)
        }
    } else {
        entries.code = 99
        cb(false, entries)
    }
}

源码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • So you want to create an infographic, and you’ve stumbled...
    Nicebook阅读 591评论 0 3
  • 心中憋着一团怒火! “为了孩子,为了家人和睦,你就忍着点吧啊……” “年轻人!要多学习,学习包容,有啥好吵的...
    琼宝阅读 158评论 0 1
  • 苏轼仕途坎坷而壮心不已,蒲松龄面对落第却发奋创作,曹雪芹处境艰难仍不辍笔耕,安徒生屡遭失业而自强不息。 他们面对艰...
    熏莉阅读 397评论 0 0