Vue.js入门教程

蚂蚁金服体验技术部招人啦,前端开发专家,Node.js开发专家,欢迎投递简历,长期有效。邮箱: chenglin.mcl@antfin.com,。

关于结缘的前端使用哪种框架,笔者之前有过Angular.js的开发经验,在React.js和Vue.js之间徘徊了一阵子最终选择了Vue,在我看来Vue和Angular有许多的共同点,在看Vue的文档的时候有很熟悉的味道,但和React类似只关注View这一块。Vue没有像React那样什么都放到js中,简洁方便的api设计以及组件式开发是我选择它的主要原因。

实现前端大抵有两种思路,一个是由后端拼接页面后返回完整的网页,然后由浏览器渲染。另一种是由后端返回json之类的数据,然后由前端拼接数据进行展示,当然也可以两者混搭。实现前后端分离是很多web开发人员的梦想,有兴趣的可以看看Web系统开发构架再思考-前后端的完全分离

结缘前端打算使用Vue.js做成SPA(Single Page Application)的形式,即一个入口页面,后续数据由后端Nodejs实现Restful风格的API调用,然后在前端呈现。使用Webpack作为前端工程解决方案解决资源管理,按需加载,实时更新等问题。本篇我们探索使用Vue.js实现结缘的登录界面。

代码

代码放在结缘的前端工程中,Vue-jieyuan,欢迎star,issue

安装部署Vue开发环境

为了方便大型应用的开发,尤大大开发了Vue-cli脚手架,提供了一系列的工具和库,方便我们快速的进行开发,具体功能包括单文件 Vue 组件,热加载,保存时检查代码,单元测试等,本质上和Express的express-generator是一样的。

因为vue-cli依赖webpack,所以首先安装webpack这个工具:

$ npm install -g webpack

关于webpack如果没了解过可以看基于webpack搭建前端工程解决方案探索,然后安装vue-cli:

$ npm install -g vue-cli

使用方法如下:

$ vue init webpack my-project
$ cd my-project
$ npm install
$ npm run dev

执行完成后在浏览器中localhost:8080查看。

尤大大目前提供了4套官方模板,如下:

  • browserify - A full-featured Browserify + vueify setup with hot-reload, linting & unit testing.
  • browserify-simple - A simple Browserify + vueify setup for quick prototyping.
  • webpack - A full-featured Webpack + vue-loader setup with hot reload, linting, testing & css extraction.
  • webpack-simple - A simple Webpack + vue-loader setup for quick prototyping.
    可以根据需求选择即可。

在使用过程中遇到两个问题,如果你也遇到了,可以在issues中查看。

学习ES6

由于vue-cli生成的文件中使用的是ES6的语法,而ES6是未来的趋势,所以ES6必须一学,目前浏览器和Nodejs对ES6的支持程度不断提高,不过要在所有的浏览器中使用es6代码目前还不可行,不过babel可以帮你提前体验新的语法而不需要等待浏览器支持。babel本质上是一个js的预编译器,可以把es6程序编译成es5,从而在支持ES5的环境中运行。

特地查了一下ES6和ES2016,ES2015的区别,实际上ES6===ES2015 < ES2016,由于ECMA委员会决定将标准每年一更,因此新推出的ES6被改名为ES2015,后面的标准将实行年制命名,如ES2016,ES2017...

学习ES6/ES2015可以参考如下资源

学习vue目前没有足够的教程可以帮我们快速了解如何构建我们的应用,所以只能一点点摸索,借鉴前人的经验,所幸尤大大有个使用Vue开发的Hacker News Clone,我们可以从这里吸收开发经验。另外Cnode社区也有个用Vue开发的客户端Vue-cnodejs也很不错。

这里首先简要介绍一些Hacker News 客户端中使用到的ES6特性:

箭头函数 => 和this

=>是匿名函数的一种简写,即lamda表达式,格式为( 形参列表 ) => { 函数体 },使用箭头函数,内部函数继承了外围作用域的this值,再也不用写var that=this这种hack代码了。直接上代码看:

// Expression bodies
var odds = evens.map(v => v + 1);
var nums = evens.map((v, i) => v + i);

// Statement bodies
nums.forEach(v => {
  if (v % 5 === 0)
    fives.push(v);
});

// Lexical this
var bob = {
  _name: "Bob",
  _friends: [],
  printFriends() {
    this._friends.forEach(f =>
      console.log(this._name + " knows " + f));
  }
};

模块定义

在ES6之前js没有一个统一的模块定义方式,流行的定义方式有AMD,CommonJS等,而ES6从语言层面对定义模块的方式进行了统一。

// lib/math.js
export function sum(x, y) {
  return x + y;
}
export var pi = 3.141593;

// app.js
import * as math from "lib/math";
alert("2π = " + math.sum(math.pi, math.pi));

// otherApp.js
import {sum, pi} from "lib/math";
alert("2π = " + sum(pi, pi));

export default(默认加载)和 export *(整体加载)为:

// lib/mathplusplus.js
export * from "lib/math";
export var e = 2.71828182846;
export default function(x) {
    return Math.exp(x);
}

// app.js
import exp, {pi, e} from "lib/mathplusplus";
alert("2π = " + exp(pi, e));

默认加载的好处是我们不需要知道模块所要加载的变量名或函数名,输出时指定任意名字,且不需要大括号。更详细的可以查看阮一峰老师的module一节

const 和 let

const即常量,一旦定义了即不可变。let是更好的var,由于js的设计缺陷,var变量的作用域是函数体的全部,还有变量提升等怪异特性,导致诡异的错误,极难定位bug。而let拥有块级作用域,声明的全局变量不是全局对象的属性,形如for (let x...)的循环在每次迭代时都为x创建新的绑定.能用let尽量不用var,具体请看Is there any reason to use the “var” keyword in ES6?以及深入浅出ES6(十四):let和const

function f() {
  {
    let x;
    {
      // okay, block scoped name
      const x = "sneaky";
      // error, const
      x = "foo";
    }
    // okay, declared with `let`
    x = "bar";
    // error, already declared in block
    let x = "inner";
  }
}

promise

语言标准实现的异步编程解决方案:

function timeout(duration = 0) {
    return new Promise((resolve, reject) => {
        setTimeout(resolve, duration);
    })
}

var p = timeout(1000).then(() => {
    return timeout(2000);
}).then(() => {
    throw new Error("hmm");
}).catch(err => {
    return Promise.all([timeout(100), timeout(200)]);
})

关于Vue-cli的webpack模板的代码风格

Vue-cli本身是一套技术选型,本身有作者自己的设计偏好在里面,例如模板默认是ES6的语法,使用ESLint进行代码规范等。在我开始使用这个工具的过程中有两个纠结的地方,一个是缩进,一个分号。

关于缩进

模板中默认代码使用的2个空格进行缩进,这没问题,问题是在ESLint的配置文件.eslintrc.js中写死了indent的规则,于是各种缩进必须按照规范来,不然就会出现多处如下的错误

 error  indent  Expected indentation of 4 space characters but found 6
  /Users/Calvin/Develop/githubs/jieyuan/Vue-jieyuan/src/App.vue:35:7
        Hello,

用java习惯了,格式糟糕的话format一下就好了啊,然后各种查资料是用2个空格还是4个空格,可以看看知乎的这个回答为什么JS的规范说要用两个空格来缩进?,恩,看来写js代码用2个空格更流行一些。如果你用sublime,那么可以打开你一个js文件然后Preference -> Settings More -> Syntax Specfic-User,然后写入以下选项 :

{
  "tab_size": 2,
  "translate_tabs_to_spaces": true
}

同理对.vue也做一遍。不过萝卜青菜各有所爱,只要同意规范就好,如果还是希望使用4空格,可以编辑.eslintrc.js的indet项。

关于写不写分号

习惯了写分号,至今为止一直认为写分号会让代码清晰,不容易出错。不过在js这样不强制写分号的语言中需要另外考虑一番,可以看看知乎的这个问题:JavaScript 语句后应该加分号么?,于是又被尤大的答案折服了,ok,咱也不写分号了。更多的还是建议看尤大给的链接semicolons

单文件组件以及Vue-loader解惑

看Vue-cli中的src/componets文件夹有个Hello.vue的文件,这个是默认生成的单文件组件。如下:

<template>
  <div class="hello">
    <h1>{{ msg }}</h1>
  </div>
</template>
<script>
export default {
  data () {
    return {
      // note: changing this line won't causes changes
      // with hot-reload because the reloaded component
      // preserves its current state and we are modifying
      // its initial state.
      msg: 'Hello World!'
    }
  }
}
</script>

咦?不对啊,这玩意是组件?Vue文档中不是说组件要用如下形式声明吗:

var MyComponent = Vue.extend({
  // options...
})

// Globally register the component with tag: my-component
Vue.component('my-component', MyComponent)
<div id="example">
  <my-component></my-component>
</div>

好吧,在Building Large-Scale Apps中文档介绍了这种单文件的组件,它的特点是单文件组合HTML 模板,CSS和JS,并且可以使用自己想用的预处理器,并且css代码对于每个组件是隔离的,只能说Vue,就决定是你了!,例如:

直接官网盗图
,尤大创造了一个新的文件格式.vue,那这种文件咋解析啊,有没有文件解析器?于是有了vue-loader,它的官方介绍如下:

vue-loader is a loader for Webpack that can transform Vue components written in the following format into a plain JavaScript module

恩,Vue-loader会自动帮你把这种单文件组件转成组件使用,我们就不用操心啦。当然尤大也不强制你把代码都放在一个文件里,可以拆开放:

<template src="./template.html"></template>
<style src="./style.css"></style>
<script src="./script.js"></script>
或者从npm模块加载:
<!-- import a file from the installed "todomvc-app-css" npm package -->
<style src="todomvc-app-css/index.css">

更多内容可以看Vue-loader的官方文档,跟着做一遍可以加深印象,更了解webpack和vue的思想。

开发Vue组件

好吧,说了那么多,我们来开发一个基本的Login组件吧。目前网页设计水平还跟不上,直接使用Bootstrap的css库,在index.html的head标签中加入

<link rel="stylesheet" href="//cdn.bootcss.com/bootstrap/3.3.5/css/bootstrap.min.css">

compoents标签中添加Login.vue文件,输入如下内容:

<template>
<!-- Stack the columns on mobile by making one full-width and the other half-width -->
    <div class="container">
      <form class="form-signin">
        <h2 class="form-signin-heading">{{title}}</h2>
        <label for="inputEmail" class="sr-only">邮件地址</label>
        <input type="email" id="inputEmail" class="form-control" placeholder="邮件地址" required autofocus>
        <label for="inputPassword" class="sr-only">密码</label>
        <input type="password" id="inputPassword" class="form-control" placeholder="密码" required>
        <div class="checkbox">
          <label>
            <input type="checkbox" value="remember-me"> 记住我
          </label>
        </div>
        <button class="btn btn-lg btn-primary btn-block" type="submit">登录</button>
      </form>
    </div>

</template>

<script>
export default {
  data () {
    return {
      // note: changing this line won't causes changes
      // with hot-reload because the reloaded component
      // preserves its current state and we are modifying
      // its initial state.
      title: '登录结缘'
    }
  }
}
</script>

<style >
  body {
  padding-top: 40px;
  padding-bottom: 40px;
  background-color: #eee;
}

.form-signin {
  max-width: 330px;
  padding: 15px;
  margin: 0 auto;
}
.form-signin .form-signin-heading,
.form-signin .checkbox {
  margin-bottom: 10px;
}
.form-signin .checkbox {
  font-weight: normal;
}
.form-signin .form-control {
  position: relative;
  height: auto;
  -webkit-box-sizing: border-box;
     -moz-box-sizing: border-box;
          box-sizing: border-box;
  padding: 10px;
  font-size: 16px;
}
.form-signin .form-control:focus {
  z-index: 2;
}
.form-signin input[type="email"] {
  margin-bottom: -1px;
  border-bottom-right-radius: 0;
  border-bottom-left-radius: 0;
}
.form-signin input[type="password"] {
  margin-bottom: 10px;
  border-top-left-radius: 0;
  border-top-right-radius: 0;
}
</style>

App.vue中的Hello组件换成Login组件即可,最终效果如下:

代码

代码放在结缘的前端工程中,Vue-jieyuan,欢迎star,issue

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

推荐阅读更多精彩内容