Record work income
做一个记账小程序
- 账号系统
- 登录/注册
- 持久登录
- 用户信息页
- 修改密码
- 路由鉴权
- 第三方登陆与绑定(GitHub)
- 账单图表
- 增删改查
- PWA
- 国际化
- Travis CI 持续集成
- 变量配置分离
- 谷歌统计
- Readme.md Convert About Page
唔,没什么可说的。
写给没什么网站开发基础的多多同学。
首先,开发一个网站,有多种开发方式。
在远古时代,网页常常是通过编写 Html 标签元素(如 <p></p>
),
并拼凑起来展示网页的内容。
后来有了 Css 来控制网页地布局,
再后来又有了 JavaScript,
可以实现与用户之间的交互和一些计算。
譬如点击按钮后,让网页的某部分内容发生变化,样式发生改变等等等等。
所以 Html + Css + JavaScript 成为了开发网页必备的三剑客。
这些都可以在 Mozilla 中更加深入地了解,在此略过不表。
后来 jQuery 横空出世,几乎一统江湖。那么 jQuery 又是什么? jQuery 名中含有一个 j, Query 则是查询的意思。顾名思义,jQuery 是由 JavaScript 编写的一个库。 用来更加快速地查找出 Html 文档中的标签元素,并操纵它。这也大大提高了前端开发的效率。
但天下大势,分久必合,合久必分。为了克服 HTML 在构建应用上的不足,由谷歌开发的 Angualar 诞生(2009)。相比 jQuery 而言,是一种更完备的 Web 应用开发的解决方案。 后来 Facebook 对市场上所有 JavaScript MVC 框架都不满意,就决定自己写一套。 React 就此诞生,并于 2013 年 5 月开源。 而 Vue 的第一个 Commit 可追溯至 2013 年 7 月 28 日,此后势如破竹,并于去年成功超过 React 在 Github 上的 Star 数。成为目前 GitHub 上最受欢迎的前端框架。 Angular/React/Vue 三足鼎立之势业已形成。
Vue.js 与 jQuery 不同,jQuery 操纵 DOM ((文档对象模型(Document Object Model)),譬如一个标签就是一个对象模型), 而 Vue 则由数据驱动,当数据发生改变时,再更新视图,并不直接操纵 DOM 。
说了这么多,本项目决定采用 Nuxt.js 来开发,Nuxt 正是基于 Vue.js 的通用应用框架。 好处就是,它可以提供一个良好的模板结构,以及服务端渲染、异步数据加载、中间件支持等等许多特性。 这些特性,通过 Vue 相关的插件与配置同样可以从零实现,而 Nuxt 则将其集成在一起,可以帮我们省去许多的工作。
项目地址: https://github.com/YunYouJun/duoduo-cashier
可参考 Nuxt 文档
运行 create-nuxt-app
此处我使用 yarn
yarn create nuxt-app duoduo-cashier
默认服务器为 None,因为后台打算使用 LeanCloud ,一是需求小可以使用开发版免费,二则无需额外搭建后台服务器。
UI 框架选择了 Element-UI ,因为过去经常使用,较为熟悉,且 Github 上星数最多,经常维护。
选择 SPA[single page web application] 模式,旨在开发单页应用,即只有一张Web页面的应用,当进行交互时,动态更新页面视图,也是现在较为流行的模式。
安装 axios 模块,以备发送 api 请求。 ( axios 是一个流行的基于 promise 的 HTTP 库,可以方便地发送请求并获取响应数据,是前端与后端建立连接的重要桥梁。)
进入文件夹,启动项目
cd duoduo-cashier
yarn dev
目录结构及各文件夹用途可参见此处文档
因为安装了 prettier 美化代码格式,所以有时会出现类似如下相关提示。
8:17 error Delete `␍⏎␍⏎` prettier/prettier
13:8 error Delete `␍⏎` prettier/prettier
✖ 2 problems (2 errors, 0 warnings)
2 errors and 0 warnings potentially fixable with the `--fix` option.
输入如下命令,自动修复。
yarn lint --fix
默认 prettier 配置文件 .prettierrc
如下
{
"semi": false,
"singleQuote": true
}
semi
为 false 代表 js 语句结尾不使用分号 ;
singleQuote
为 true 代表 js 中引号使用单引号
公共需要的 CSS 可以写在额外单独的文件里。 配置方法参见 CSS 配置
本项目配置在了 @/assets/css/main.scss
里 ( @ 代表项目根目录)
使用 scss 需要额外安装 node-sass
sass-loader
yarn add -S -D node-sass sass-loader
# 如果只有 -S 写入 package.json 的 dependencies 里, 有了 -D 则写入 devDependencied 里
sass 是一种成熟、稳定、强大的CSS预处理器, 可以看作一种用来编写 css 的编程语言。
SCSS 是 Sass3 版本当中引入的新语法特性,完全兼容CSS3的同时继承了Sass强大的动态功能。
与原生 css 相比的好处有如可以进行嵌套,使用变量等等。
$primary-color: #333;
body {
color: $primary-color;
.class {
...
}
}
图表部分的实现自然要依靠大家的智慧。框架必不可少. 此处我选用的是基于 Html5 Canvas 的 Chart.js, 以及用 vue 对其进行封装的 vue-chartjs。
# 安装依赖
yarn add vue-chartjs chart.js
因为采用 LeanCloud 作为后端,所以无需另外编写后台代码。
为了降低对 LeanCloud 的依赖(譬如日后迁移后台),用个专业名词来说就是解耦, 决定采用 LeanCloud 提供的 Rest API 服务。(这是一种通用的前后端分离的方法)
同时采用 Postman (一个 HTTP Client 工具,可以自定义 HTTP 请求)来进行接口的调试。 可参考使用 Postman 调试 REST API
可参考 nuxt-axios 文档
预先配置 axios 中请求的基础部分 browerBaseURL
// nuxt.config.js 中 axios 字段
module.exports = {
...
axios: {
// See https://github.com/nuxt-community/axios-module#options
browserBaseURL: 'https://sqgxqvmq.api.lncld.net/1.1/'
},
...
}
还需额外通过拦截器配置 axios 发送请求的全局 Headers 。可参见Extending axios
// nuxt.config.js
{
...
plugins: [
'~/plugins/axios'
]
...
}
// ~/plugins/axios.js
// 此处为 Leancloud api 对应应用 duoduo-cashier 的 Id 与 Key
export default function({ $axios, redirect }) {
$axios.setHeader('X-LC-Id', 'SqGXqvMqLFSGgxYpcXFbmBgR-gzGzoHsz')
$axios.setHeader('X-LC-Key', 'kUXrbjyUdvJV4i1FhsVayqa4')
...
}
配置完毕千万记得重启,泪流满面。
通过 axios 发送请求,在 Vue 的 methods 中即可类似如下调用。
...
login() {
this.$axios.$post('api/login', this.loginForm)
.then((res) => { // 此处使用箭头函数
...
this.$message({
type: 'success',
message: '登录成功!'
})
}
}
...
箭头函数有些类似于普通的 function 函数,譬如还可以这么写。
...
login() {
let _this = this
this.$axios.$post('api/login', this.loginForm)
.then(function(res) { // 此处使用箭头函数
...
_this.$message({
type: 'success',
message: '登录成功!'
})
}
}
...
显然使用箭头函数要更为方便,箭头函数的好处就是不会创建自己的 this ,它只会从自己的作用域链的上一层继承 this , 因此可以继续在其内部使用继承自外部的 this 。
Tip: 当箭头函数仅有一个参数的时候,可以省略掉括号。 (res) => {} res => {}
可参见箭头函数
当我们想要一些页面只允许登录后的用户才能访问,就需要用到中间件来拦截并过滤请求。
在 middleware 文件夹下,新建 auth.js
。
export default function({ store, redirect }) {
if (!store.state.username) {
return redirect('/login')
}
}
在显示页面前,会先去查找是否有用户信息存储下来,当无法获取登录后的用户信息时,便会重定向回登陆页面。
此外,我们想要一直保留用户的登录状态。可以依靠 Vuex 。 但 Vuex 当页面刷新时,会丢失数据。 想要实现保留用户登录状态,文中提供的方法是 nuxtServerInit ,每次从服务器获取到数据时都会执行一次。 但我们无法对 LeanCloud 的后台服务器内容进行修改,且编写的是静态化页面。 所以采用 Html 5 提供的 sessionStorage 来存储,并通过全局的中间件在每次页面开始时, 从 sessionStorage 中读取数据存储到 Vuex 的状态树中。
sessionStorage 当浏览器关闭时,便会消失。 而 localStorage 与 sessionStorage 有同样的 API 和功能,但是在浏览器关闭,然后重新打开后数据仍然存在。 所以想要下次打开页面也不用登录时,可以使用 localStorage 替换 sessionStorage 。
// middleware/status.js
export default function({ store }) {
if (!store.state.sessionStorage) {
if (sessionStorage.getItem('sessionToken') !== null) {
let sessionToken = sessionStorage.getItem('sessionToken')
store.commit('SET_SESSION_TOKEN', sessionToken)
}
}
if (!store.state.username) {
if (sessionStorage.getItem('username') !== null) {
let username = sessionStorage.getItem('username')
store.commit('SET_USER', username)
}
}
}
// store/index.js
export const state = () => ({
username: null,
sessionToken: null
})
export const mutations = {
SET_SESSION_TOKEN: function(state, sessionToken) {
state.sessionToken = sessionToken
if (sessionToken) {
sessionStorage.setItem('sessionToken', sessionToken)
} else {
sessionStorage.removeItem('sessionToken')
// 这里需要使用 removeItem
// 否则 sessionStorage.setItem('sessionToken', null) 会将其赋值变为 "null" 字符串
}
},
SET_USER: function(state, username) {
state.username = username
if (username) {
sessionStorage.setItem('username', username)
} else {
sessionStorage.removeItem('username')
}
}
}
// 实现登出 通过 this.$store.dispatch('logout) 调用
export const actions = {
logout({ commit }) {
commit('SET_USER', null)
commit('SET_SESSION_TOKEN', null)
}
}
想要连接 GitHub 第三方账户时,需先去 GitHub Developer Setting 新建
OAuth APP
。
连接 GitHub 时,编写页面脚本打开新窗口。
https://github.com/login/oauth/authorize?client_id=xxx&scope=user:email
client_id 可在注册好的 OAuth APP 中找到, scope 后参数为想要获取的 GitHub 的信息。
用户授权后,会跳转回注册时填写的 redirect_url
,并附加 code 参数。
譬如:http://localhost:3000/oauth?code=e7f48dcec821618e5350
再通过拼接如下 url,https://github.com/login/oauth/access_token?client_id=xxx&client_secret=xxx&code=xxx
获取 access_token
。
因为浏览器本身有 CORS 跨域限制。而我们有没有真正编写后端,而是直接使用 LeanCloud 服务,无法在后台进行操作 oauth 的回调。 所以作为前端页面想要获取 github 的 access_token 时,我使用了其他的代理服务 JsonBird。
我简要看了一下其在 GitHub 上的代码,大致是用了 Express 编写的后端作为代理发送请求并获得结果。
断开连接时,因为对数据对象操作,但是发现清除对象属性时,视图没有随之改变。 Vue 没有对对象的更深层属性进行监听,所以需要使用特殊的 API 进行修改。
this.$set(this.userInfo.authData, row.account, undefined)
_User
API-URL: users/me
Attributes | Type | Default | Description |
---|---|---|---|
bio | String | Bio | |
createdAt | |||
emailVerified | false | ||
mobilePhoneNumber | |||
mobilePhoneVerified | false | ||
objectId | |||
sessionToken | |||
updatedAt | |||
username |
不过有一个坑就是 GitHub Pages 似乎无法访问带有 /_nuxt/
链接的资源,与下划线有关。
所以需要修改配置 build
中 publicPath。
默认为 _nuxt
,修改为 nuxt
,也可以修改为其他不带下划线的名称。