学习目标:

完成用户注册功能

完成用户登陆功能,掌握js-cookie的使用

完成微信扫码登陆的功能

完成用户中心嵌套布局,掌握nuxt嵌套路由的使用

1用户注册

1.1页面构建

创建pages/login.vue

<template> <div class="wrapper loginsign"> <div class="item signup"> <div class="form"> <h3 class="loginsign‐title">注册新账号</h3> <form class="sui‐form"> <div class="control‐group"> <label for="inputname" class="control‐label">名字</label> <div class="controls"> <input type="text" id="inputname" placeholder="真实姓名或常用昵称" class="input‐xlarge" data‐rules="required" /> </div> </div> <div class="different"> <div class="radio‐content"> <div id="a1" class="phone"> <div class="control‐group number"> <input type="text" placeholder="仅支持大陆手机号" class="input‐ xlarge" data‐rules="required|mobile" /> </div> <div class="control‐group code"> <div class="input‐append"> <input id="appendedInputButton" type="text" placeholder="短信验证" class="span2 input‐large msg‐input" /> <button type="button" class="sui‐btn msg‐btn">获取验证 码</button> </div> </div> <div class="control‐group"> <label for="inputpassword" class="control‐label">密码</label> <div class="controls"> <input type="text" id="inputpassword" placeholder="请输入6‐16位密码" class="input‐xlarge" /> </div> </div> </div> <div id="a2" class="email"> <div class="control‐group inputemail"> <input type="text" placeholder="输入手机号" class="input‐xlarge" /> </div> <div class="control‐group"> <label for="inputpassword" class="control‐label">密码:</label> <div class="controls"> <input type="text" id="inputpassword" placeholder="请输入6‐16位 字符" class="input‐xlarge" /> </div> </div> </div> </div> </div> <div class="control‐group btn‐signup"> <label class="control‐label"></label> <div class="controls"> <label> <input type="checkbox" /><span class="type‐text" style="font‐size:12px;">同意协议并接受《服务条款》</span> </label> <button type="submit" class="sui‐btn btn‐danger btn‐yes">注 册 </button> </div> </div> </form> </div> </div> <div class="item"> <div class="form"> <h3 class="loginsign‐title">用户登录</h3> <form class="sui‐form login‐form"> <div class="control‐group"> <label for="inputname" class="control‐label">手机号或Email:</label> <div class="controls"> <input type="text" id="inputname" placeholder="11位手机号或Email" class="input‐xlarge" data‐rules="required" /> </div> </div> <div class="control‐group"> <label for="inputpassword" class="control‐label">密码:</label> <div class="controls"> <input type="text" id="inputpassword" placeholder="输入登录密码" class="input‐xlarge" /> </div> </div> <div class="controls"> <label> <input type="checkbox" name="remember‐me" /><span class="type‐text" style="font‐size:12px;">记住登录状态</span> </label> <button type="submit" class="sui‐btn btn‐danger btn‐yes">登录</button> </div> <div class="other‐methods"> </div> </form> </div> </div> </div> </template> <script> import '~/assets/css/page‐sj‐person‐loginsign.css' export default { } </script>

其中的内容参见person-loginsign.html

1.2获取验证码

1.2.1模拟数据与API

(1)将user.yml 导入easymock

(2)修改easy-mock 数据

url: /user/user/sendsms/{mobile} method:put

{ "code": 20000, "flag": true, "message": "验证码发送成功" }

(3)编写API 创建api/user.js

import request from '@/utils/request' const api_group = 'user' const api_name = 'user' export default { sendsms(mobile) { return request({ url: `/${api_group}/${api_name}/sendsms/${mobile}`, method: 'put' }) } }

1.2.2调用API

(1)修改pages/login.vue脚本部分

<script> import '~/assets/css/page‐sj‐person‐loginsign.css' import userApi from '@/api/user' export default { data(){ return { pojo: {} } }, methods: { sendsms(){ userApi.sendsms( this.pojo.mobile ).then(res => { alert(res.data.message) }) } } } </script>

(2)修改pages/login.vue 绑定变量

<input type="text" v‐model="pojo.mobile" placeholder="仅支持大陆手机号" class="input‐xlarge" />

(3)修改pages/login.vue 按钮绑定方法

<button type="button" class="sui‐btn msg‐btn" @click="sendsms" >获取验证码</button>

1.2.3使用ElementUI 的弹出框

(1)安装element-ui

cnpm install element‐ui ‐‐save

(2)plugins文件夹下创建element-ui.js

import Vue from 'vue' import ElementUI from 'element‐ui' Vue.use(ElementUI)

(3)修改nuxt.config.js,加入插件与样式

(2)开发者资质认证

郑州web教程(十次方专属社交平台前端系统开发)(1)

(3)创建网站应用

郑州web教程(十次方专属社交平台前端系统开发)(2)

填写应用的相关信息

郑州web教程(十次方专属社交平台前端系统开发)(3)

应用通过审核后,会得到AppID和 AppSecret,后边进行编码中会使用。

AppID : wx3bdb1192c22883f3

AppSecret : db9d6b88821df403e5ff11742e799105

3.2微信第三方登陆流程

1.第三方发起微信授权登录请求,微信用户允许授权第三方应用后,微信会拉起应用或重定向到第三方网站,并且带上授权临时票据code参数;

2.通过code参数加上AppID和AppSecret等,通过API换取access_token;

3.通过access_token进行接口调用,获取用户基本数据资源或帮助用户实现基本操作。

获取access_token时序图:

郑州web教程(十次方专属社交平台前端系统开发)(4)

3.3获取code

(1)修改login.vue,在登陆表单下方添加一个div, 用于显示微信登陆二维码

<div id="weixin"></div>

(2)修改login.vue ,引入微信登陆二维码js\

mounted(){ var obj = new WxLogin({ id: "weixin", appid: "wx3bdb1192c22883f3", scope: "snsapi_login", redirect_uri: "http://note.Java.itcast.cn/weixinlogin" }); }, head:{ script:[ {src:'http://res.wx.qq.com/connect/zh_CN/htmledition/js/wxLogin.js'} ] }

appid: 应用唯一标识

scope:应用授权作用于

redirect_uri:回调地址,是微信登陆成功后要跳转到的页面

(3)测试:http://localhost:3000/login 浏览器显示

郑州web教程(十次方专属社交平台前端系统开发)(5)

我们打开手机用微信扫二维码, 会出现以下提示

点击确认登陆按钮,浏览器会自动跳转到

http://note.java.itcast.cn/weixinlogin? code=02147Yff12Yhgz0ArCef1qabgf147Yf0&state=undefined

这个code是微信发给用户的临时令牌。我们可以根据code再次请求微信第三方登陆接口得到access_token(正式令牌)

3.4获取access_token

3.4.1API

通过code获取access_token 接口说明

通过code获取access_token的接口。

请求说明

http 请 求 方 式 : GET https://api.weixin.qq.com/sns/oauth2/access_token? appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code

参数说明

郑州web教程(十次方专属社交平台前端系统开发)(6)

返回说明

正确的返回:

{ "access_token":"ACCESS_TOKEN", "expires_in":7200, "refresh_token":"REFRESH_TOKEN","openid":"OPENID", "scope":"SCOPE" }

郑州web教程(十次方专属社交平台前端系统开发)(7)

3.4.2编写node服务

创建新的node工程 weixinlogin 用于调用微信第三方登陆接口 工程下创建server.js

var http = require('http'); var https = require('https'); var url = require('url'); http.createServer(function(request,response){ var params=url.parse(request.url, true).query; var appid='wx3bdb1192c22883f3'; var secret='db9d6b88821df403e5ff11742e799105'; if(params.operation==='token'){ https.get(`https://api.weixin.qq.com/sns/oauth2/access_token? appid=${appid}&secret=${secret}&code=${params.code}&grant_type=authorizat ion_code`, function (res) { res.on('data', function (chunk) { response.writeHead(200,{'Content‐ Type':'application/json;charset=utf‐8' ,"Access‐Control‐Allow‐Origin": "*" }); response.end(chunk); }); }) } }).listen(8888); // 终端打印如下信息 console.log('Server running at http://127.0.0.1:8888/');

在控制台输入 运行服务

地 址 栏 测 试 :http://localhost:8888/? code=02147Yff12Yhgz0ArCef1qabgf147Yf0&operation=token

结果如下:

{ "access_token": "10_zSHADX2JGMivKFfa4nMbZV3ECzY21UY3qrF5ADyjpr_iiLUifo‐ nlN0GaRnUEN9T7BagiwSC07awplRFIO1Ghw", "expires_in": 7200, "refresh_token": "10 zl8gcJz0RXVDKtksbNTQJZ2uK1HiLJZ3I5PcSkA2VB3b6WXi2CR3R_htW6B8kKOmj‐ 91p08SJMfVKkL84vP1w", "openid": "oypcC1u9r‐mxVsRGSLFqE65lysVI", "scope": "snsapi_login", "unionid": "o6JuL1gaIwnVsZC5BpRYImTHKTm8" }

3.4.3调用node服务

node服务编写完成后,我们在十次方前台工程中调用node服务

(1)编写API ,创建api/weixin.js

import axios from 'axios' export default { getAccessToken(code){ return axios.get(`http://localhost:8888? operation=token&code=${code}`) } }

(2)创建utils/param.js (用于获取浏览器地址栏参数)

export function getUrlParam(name) { var reg = new RegExp("(^|&)" name "=([^&]*)(&|$)"); var r = window.location.search.substr(1).match(reg); if(r != null) return unescape(r[2]); return null; }

(3)创建pages/weixinlogin.vue

<template> <div></div> </template> <script> import {getUrlParam} from '@/utils/param' import weixin from '@/api/weixin' import {setUser} from '@/utils/auth' export default { mounted(){ let code=getUrlParam('code')a if(code!==null){//如果是微信登陆 // 根 据 code 获 取 access_token weixin.getAccessToken(code).then( res=>{ let access_token= res.data.access_token let openid= res.data.openid console.log('access_token:' access_token 'openid:' openid) }) } } } </script>

3.5获取用户昵称与头像

3.5.1API

http 请 求 方 式 : GET https://api.weixin.qq.com/sns/userinfo? access_token=ACCESS_TOKEN&openid=OPENID

郑州web教程(十次方专属社交平台前端系统开发)(8)

返回说明

正确的Json返回结果:

{ "openid":"OPENID", "nickname":"NICKNAME", "sex":1, "province":"PROVINCE", "city":"CITY", "country":"COUNTRY", "headimgurl": "http://wx.qlogo.cn/mmopen/g3MonUZtNHkdmzicIlibx6iaFqAc56vxLSUfpb6n5WKSYVY 0ChQKkiaJSgQ1dZuTOgvLLrhJbERQQ4eMsv84eavHiaiceqxibJxCfHe/0", "privilege":[ "PRIVILEGE1", "PRIVILEGE2" ], "unionid": " o6_bmasdasdsad6_2sgVt7hMZOPfL" }

郑州web教程(十次方专属社交平台前端系统开发)(9)

3.5.2编写node服务

修改node工程 weixinlogin的server.js, 新增代码

..... if(params.operation==='userinfo'){ https.get(`https://api.weixin.qq.com/sns/userinfo? access_token=${params.access_token}&openid=${params.openid}`, function (res) { res.on('data', function (chunk) { // 发 送 响 应 数 据 "Hello World" response.writeHead(200,{'Content‐ Type':'application/json;charset=utf‐8' ,"Access‐Control‐Allow‐Origin": "*" }); response.end(chunk); }); }) } .....

3.5.3调用node服务

(1)编写API ,修改api/weixin.js 新增方法 用于根据access_token和openid获取用户信息

getUserinfo(access_token,openid){ return axios.get(`http://localhost:8888? operation=userinfo&access_token=${access_token}&openid=${openid}`) }

(2)修改pages/weixinlogin.vue

import {setUser} from '@/utils/auth'

在获取access_token和openid后,再次请求接口,获取昵称和头像,保存到cookie中

mounted(){ let code=getUrlParam('code') if(code!==null){//如果是微信登陆 // 根 据 code 获 取 access_token weixin.getAccessToken(code).then( res=>{ let access_token= res.data.access_token let openid= res.data.openid weixin.getUserinfo( access_token, openid ).then( res => { //提取用户昵称和头像 ********************** let nickname= res.data.nickname let headimgurl= res.data.headimgurl setUser(access_token,nickname,headimgurl) location.href='/' }) }) } }

3.6域名与端口设置

我们刚才都是要通过手动更改url才能完成测试,主要是因为回调地址是域名而我们的工程是本地地址。其实我们要想实现一气呵成的效果也不难,只要通过域名和端口设置即 可。

3.6.1域名指向

我们可以通过SwitchHosts 配置域名指向

127.0.0.1 note.java.itcast.cn

这样我们的工程就可以通过 http://note.java.itcast.cn:3000来访问了

3.6.2NUXT端口设置

修改package.json ,添加配置

"config": { "nuxt": { "port": "80" } },

重新启动工程,就可以通过http://note.java.itcast.cn 来访问了。

通过以上修改后,我们再次测试微信扫码登陆,就可以看到和生产环境一样的运行效果。

4用户中心嵌套布局

4.1子布局页

(1)创建pages/manager.vue ,这个是用户中心的布局页

<template>

<div>

<div class="myhome‐personinfo" style="background‐image: url('~/assets/img/widget‐homebg.png');">

<div class="wrapper">

<div class="person‐baseinfo">

<!‐‐头像信息‐‐>

<div class="photo">

<img src="http://img.studyofnet.com~/assets/img/widget‐myphoto.jpg" alt="" class="person" />

<div class="share">

<span><img src="http://img.studyofnet.com~/assets/img/asset‐QQ.png" alt="" width="34" height="28" /></span>

<span><img src="http://img.studyofnet.com~/assets/img/asset‐weixin.png" alt="" width="28" height="28" /></span>

<span><img src="http://img.studyofnet.com~/assets/img/asset‐weibo.png" alt="" width="28" height="28" /></span>

</div>

</div>

<!‐‐文字信息‐‐>

<div class="info">

<h1>Web爱好者<span class="allinfo"><a href="~/assets/person‐ myfile.html" target="_blank">查看完整档案</a></span></h1>

<ul class="fill">

<li> <i class="fa fa‐map‐marker" aria‐hidden="true"></i> <span class="edit‐item"> 填写现居城市</span>

<form action="" class="sui‐form form‐inline">

<input type="text" placeholder="现居城市" />

<button class="sui‐btn btn‐danger save‐btn">保存</button>

</form> </li>

<li> <i class="fa fa‐graduation‐cap" aria‐hidden="true"></i> <span class="edit‐item"> 填写毕业院校</span>

<form action="" class="sui‐form form‐inline">

<input type="text" placeholder="院校名称" />

<input type="text" placeholder="所学专业" />

<button class="sui‐btn btn‐danger save‐btn">保存</button>

</form> </li>

<li> <i class="fa fa‐shopping‐bag" aria‐hidden="true"></i> <span class="edit‐item"> 填写所在公司/组织</span>

<form action="" class="sui‐form form‐inline">

<input type="text" placeholder="公司/组织名称" />

<input type="text" placeholder="职位头衔" />

<button class="sui‐btn btn‐danger save‐btn">保存</button>

</form> </li>

<li> <i class="fa fa‐link" aria‐hidden="true"></i> <span class="edit‐item"> 填写个人网站</span>

<form action="" class="sui‐form form‐inline">

<input type="text" placeholder="个人网站" />

<button class="sui‐btn btn‐danger save‐btn">保存</button>

</form> </li>

</ul>

</div>

</div>

<!‐‐右侧编辑‐‐>

<div class="edit‐info">

<h4>个人简介<span class="addedit"><img src="http://img.studyofnet.com~/assets/img/widget‐ edit.png" width="12" height="12" />编辑</span></h4>

<div class="info‐box">

<div class="edit‐intro">

暂时没有个人简介

</div>

</div>

</div>

<div class="clearfix"></div>

</div>

</div>

<!‐‐两列布局‐‐>

<div class="wrapper myhome">

<div class="left‐list">

<div class="myhome‐list">

<ul class="home‐list">

<li class="active"><a href="~/assets/person‐homepage.html">我的主页</a></li>

<li><a href="~/assets/person‐myanswer.html">我的回答</a></li>

<li><a href="~/assets/person‐myquestion.html">我的提问</a></li>

<li><a href="~/assets/person‐myshare.html">我的分享</a></li>

</ul>

<ul class="home‐list bottom">

<li><a href="~/assets/person‐dynamic.html">个人动态</a></li>

<li><a href="~/assets/person‐myfocus.html">我的关注</a></li>

<li><a href="~/assets/person‐mycollect.html">我的收藏</a></li>

<li><a href="~/assets/person‐myreaded.html">浏览记录</a></li>

<li><a href="~/assets/person‐account.html">账户设置</a></li>

</ul>

</div>

</div>

<div class="right‐content">

<nuxt‐child/>

</div>

<div class="clearfix"></div>

</div>

</div>

</template>

<script>

import '~/assets/css/page‐sj‐person‐homepage.css'

</script>

注意:我们使用 标签

(2)在pages下创建manager文件夹,manager文件夹下创建index.vue(用户中心的默认首页)

<template> <div class="home‐content"> <ul class="sui‐nav nav‐tabs nav‐large"> <li class="active"><a href="#one" data‐toggle="tab">我的提问</a> </li> <li><a href="#two" data‐toggle="tab">我的回答</a></li> </ul> <div class="tab‐content tab‐wraped"> <div id="one" class="tab‐pane active"> <ul class="question‐list"> <li> <span class="fl good"><span class="num">12</span> 有用</span> <span class="title"><a href="#">有关PHP初级进阶的问题</a></span> <span class="fr date">4月6日</span> <span class="clearfix"></span> </li> <li> <span class="fl good"><span class="num">12</span> 有用</span> <span class="title"><a href="#">有关JAVA初级进阶的问题</a></span> <span class="fr date">4月6日</span> <span class="clearfix"></span> </li> <li> <span class="fl good"><span class="num">12</span> 有用</span> <span class="title"><a href="#">有关HTML5初级进阶的问题</a></span> <span class="fr date">4月6日</span> <span class="clearfix"></span> </li> <li> <span class="fl good"><span class="num">12</span> 有用</span> <span class="title"><a href="#">有关C 初级进阶的问题</a></span> <span class="fr date">4月6日</span> <span class="clearfix"></span> </li> <li> <span class="fl good"><span class="num">12</span> 有用</span> <span class="title"><a href="#">有关python初级进阶的问题</a></span> <span class="fr date">4月6日</span> <span class="clearfix"></span> </li> </ul> </div> <div id="two" class="tab‐pane"> <ul class="question‐list"> <li> <span class="fl good"> <span class="num">8</span> 有用</span> <span class="title"><a href="#">有关PHP初级进阶的问题</a></span> <span class="fr date">2017‐07‐05 15:08</span> <span class="clearfix"></span> </li> <li> <span class="fl good"> <span class="num">7</span> 有用</span> <span class="title"><a href="#">有关JAVA初级进阶的问题</a></span> <span class="fr date">2017‐07‐05 15:08</span> <span class="clearfix"></span> </li> <li> <span class="fl good"> <span class="num">6</span> 有用</span> <span class="title"><a href="#">有关HTML5初级进阶的问题</a></span> <span class="fr date">2017‐07‐05 15:08</span> <span class="clearfix"></span> </li> <li> <span class="fl good"> <span class="num">12</span> 有 用</span> <span class="title"><a href="#">有关C 初级进阶的问题</a></span> <span class="fr date">2017‐07‐05 15:08</span> <span class="clearfix"> </span> </li> <li> <span class="fl good"> <span class="num">12</span> 有用</span> <span class="title"><a href="#">有关python初级进阶的问题</a> </span> <span class="fr date">2017‐07‐05 15:08</span> <span class="clearfix"></span> </li> </ul> </div> </div> <div class="activities"> <h4 class="tit"><span>我的动态</span></h4> <ul class="activities‐content"> <li> <div class="index‐title"> <span class="author">本杰明</span> <span class="operate">关注了标签</span> · <span class="time">3小时前</span> </div> <div class="guanzhuname"> <span class="tag">php</span> <span class="tagnum">100</span> 关注 </div> <div class="intro"> PHP,是英文超文本预处理语言 Hypertext Preprocessor 的缩写。PHP 是一种开源的通用计算机脚本语言,尤其适用于网络开发并可嵌入HTML中使用。PHP 的语法借鉴吸收C语言、Java和Perl等流行计算机语言的特点,易于一般程序员学习。 </div> </li> <li> <div class="index‐title"> <span class="author">本杰明</span> <span class="operate">回答了问题</span> · <span class="time">3小时前</span> </div> <div class="question"> <p class="title">网页链接如何直接打开微信,并进入公众号关注页面</p> <p class="content">现在针对这个微信是屏蔽的,你可以选择通过连接到一个 其他的公众号文章中进行关注。</p> </div> <div class="qa‐num"> <span>关注<i>1</i></span> <span>回答<i>2</i></span> </div> </li> <li> <div class="index‐title"> <span class="author">本杰明</span> <span class="operate">收藏了文章</span> · <span class="time">3小时前</span> </div> <div class="question"> <p class="title">网页链接如何直接打开微信,并进入公众号关注页面</p> </div> <div class="qa‐num"> <span><a href="#">http://baidu.com</a></span> </div> </li> <li> <div class="index‐title"> <span class="author">本杰明</span> <span class="operate">收藏了文章</span> · <span class="time">3小时前</span> </div> <div class="question"> <p class="title">网页链接如何直接打开微信,并进入公众号关注页面</p> </div> <div class="qa‐num"> <span><a href="#">http://baidu.com</a></span> </div> </li> <li> <div class="index‐title"> <span class="author">本杰明</span> <span class="operate">回答了问题</span> · <span class="time">3小时前</span> </div> <div class="question"> <p class="title">网页链接如何直接打开微信,并进入公众号关注页面</p> <p class="content">现在针对这个微信是屏蔽的,你可以选择通过连接到一个其他的公众号文章中进行关注。</p> </div> <div class="qa‐num"> <span>关注<i>1</i></span> <span>回答<i>2</i></span> </div> </li> </ul> </div> </div> </template>

4.2用户中心各子页面

(1)创建pages/manager/myanswer.vue(我的问答)

(2)创建pages/manager/myquestion.vue(我的提问)

(3)创建pages/manager/myshare.vue(我的分享)

(4)创建pages/manager/dynamic.vue(个人动态)

(5)创建pages/manager/myfocus.vue(我的关注)

(6)创建pages/manager/mycollect.vue(我的收藏)

(7)创建pages/manager/myreaded.vue (浏览记录)

(8)创建pages/manager/account.vue(账户设置)

4.3菜单样式处理

修改pages/manager.vue中的链接地址

<div class="myhome‐list"> <ul class="home‐list"> <router‐link to="/manager" active‐class="active" tag="li" exact > <a>我的主页</a></router‐link> <router‐link to="/manager/myanswer" active‐class="active" tag="li" exact ><a>我的回答</a></router‐link> <router‐link to="/manager/myquestion" active‐class="active" tag="li" exact ><a>我的提问</a></router‐link> <router‐link to="/manager/myshare" active‐class="active" tag="li" exact ><a>我的分享</a></router‐link> </ul> <ul class="home‐list bottom"> <router‐link to="/manager/dynamic" active‐class="active" tag="li" exact ><a>个人动态</a></router‐link> <router‐link to="/manager/myfocus" active‐class="active" tag="li" exact ><a>我的关注</a></router‐link> <router‐link to="/manager/mycollect" active‐class="active" tag="li" exact ><a>我的收藏</a></router‐link> <router‐link to="/manager/myreaded" active‐class="active" tag="li" exact ><a>浏览记录</a></router‐link> <router‐link to="/manager/account" active‐class="active" tag="li" exact ><a>账户设置</a></router‐link> </ul> </div>

4.4用户中心鉴权

修改pages/manager.vue代码部分

import '~/assets/css/page‐sj‐person‐homepage.css' import {getUser} from '@/utils/auth' export default { created(){ if(getUser().name===undefined){ this.$router.push('/login') } } }

测试:在未登录的情况下在地址栏输入http://localhost:3000/manager 会自动跳转到登录页

修改layouts/default.vue的用户名与头像,修改链接到用户中心

<li><a href="/manager" class="notice">{{user.name}}</a></li> ... <li><a href="/manager" class="homego"><img :src="http://img.studyofnet.comuser.avatar" width="50px" height="50px" :alt="user.name" /></a></li>

,