自尤大去年9月推出vue对typescript的支持后,一直想开箱尝试,对于前端sr来讲,vue的顺滑加上ts的面向对象,想着就很是美好~ 终于在两个月前,找到了个机会尝试了一把vue+ts的组合。 开文记录下vue和ts整合之旅和遇到的一些坑。html
应该大部分人都知道vue,毕竟现在vue是与react肩并肩的存在,因此就不过多介绍啦。前端
vue中文官网 官网上的教程就是最好的入门教程vue
我在前几篇文章就一直有在宣传typescript,简单列举下ts的优势node
始于JavaScript,归于JavaScript,typescript是JavaScript的超集,因此它能够重用JavaScriptd代码,使用JavaScript的库react
JavaScript用的优势它都有,跨浏览器、跨操做系统等webpack
面向对象的编程思想,强大的类型检查web
开源大法好vuex
要说缺点的话,那就是不太适合过小的项目。vue-cli
就凭这些优势,足够咱们愉快的玩耍~typescript
先将node安装,而后在经过npm安装ts的包
npm install -g typescript
tsc -v
查看ts的版本号
咱们将经过官方提供的脚手架 vue-cli 来建立项目
npm install -g @vue/cli
安装完成后,能够经过 vue create
快速建立一个新项目的脚手架,或者直接经过 vue serve
构建新想法的原型。
vue create vue-ts
vue-ts是咱们的项目名称,执行后以下可看到有这么几个选项,xiaoli
这个选项是我以前建立的,稍后会介绍;default
这个后面写着 babel eslint ,表示若选择这个,那么只会引入babel和 ealint;manually select features
顾名思义,选择咱们想要的。那么咱们选择第三个
可看到列表里有不少选项,咱们以vue+ts为主,因此咱们选择 babel typescript router vuex
这几个,选择完后,以下
接下来会有好几个yes or no 的选项,你们根据本身项目的须要来选择就能够,最后一步,Save this as a preset for future projects?
若选择yes,就会将咱们以前的选择存储起来,做为一个预设选项,方便后续一键建立新项目。全部步骤选完,回车,便开始建立项目文件结构和拉取npm包
项目结构以下
index.html
入口文件就放在里面,这个文件夹下的文件不会归入webpack的打包中;
src:存放vue项目工程文件,其中已经帮咱们关联好router和vuex,文件结构很是简洁
其余:webpack、babel等配置文件
项目在构建中,已经引入 vue-class-component
,用于对ts的支持,或者使用 vue-property-decorator
,这个库是在以前那个的基础上扩展。
如下列举tsvue写法的各类变化
建立组件的方式变成以下
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
@Component
export default class Test extends Vue {
}
复制代码
经过构造函数建立data里的数据
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
@Component
export default class Test extends Vue {
private name: string;
constructor() {
super();
this.name = 'xiaoli';
}
}
复制代码
data里的数据使用方式以下
public getName(){
return this.name
}
复制代码
@Prop() public msg: string;
@Prop({ default: 'default value' }) propB: string
@Prop([String, Boolean]) propC: string | boolean
复制代码
public created(): void {
console.log('created');
}
public mounted():void{
console.log('mounted')
}
复制代码
js下是须要在method对象中声明方法,现变成以下
public clickFunc(): void {
console.log(this.name)
console.log(this.msg)
}
复制代码
@Watch('name',{ immediate: true, deep: true })
public onChildChanged(val: string, oldVal: string) {
console.log('watch new name=' + val);
}
复制代码
public get allname() {
return 'computed ' + this.name;
}
复制代码
allname是计算后的值,name是被监听的值
@Emit()
addToCount(n: number) {
this.count += n
}
@Emit('reset')
resetCount() {
this.count = 0
}
复制代码
第一个的事件名称为 add-to-count
,n
为传过去的参数;第二个事件名为reset-count
,参数为空
----------小小更新一下---------------
有小伙伴问指令和过滤器在ts下的写法,以前给遗漏了,如今补充一下。
我尝试了下,发现以前在入口文件直接引入指令或者过滤器的方式无论用了,由于用了ts后,组件的做用域跟以前的不同了,而后我找了官方的issue,截图以下
在做者在4月11号回复里,认可了这个问题,但具体何时将指令和过滤器的声明加上就未知了,不过做者在issue中给出了解决方案。我写个简单的小栗子
一个自定义指令
// ./directive/index.ts
export const focus = {
// 当被绑定的元素插入到 DOM 中时……
inserted: function (el:HTMLElement) {
// 聚焦元素
el.focus()
}
}
复制代码
一个过滤器
// ./filter/index.ts
export const capitalize = function (value:string) {
if (!value) return ''
value = value.toString()
return value.charAt(0).toUpperCase() + value.slice(1)
}
复制代码
组件中使用
import { capitalize }from '@/filter/index'
import { focus } from '@/directive/index'
@Component({
filters:{capitalize},
directives:{focus}
})
export default class Test extends Vue {}
复制代码
<div>
<input v-focus v-model="modelData">
<div>{{modelData | capitalize}}</div>
</div>
复制代码
可看出这是局部引用,全局引用目前还没找到办法,欢迎有解决办法的小伙伴指教~
由于vuex是个可选的,因此单独列出来。首先须要引用 vuex-class
库,该库有以下几个模块
import {
namespace,
Action,
Getter,
Mutation,
State
} from 'vuex-class';
复制代码
分别对应vuex中的 action、getter、mutation等,使用ts对vuex的影响主要在组件对vuex的调用上,vuex的定义仍是按照以前的写法便可
@State('foo') stateFoo
@State(state => state.bar) stateBar
@Getter('foo') getterFoo
@Action('foo') actionFoo
@Mutation('foo') mutationFoo
@someModule.Getter('foo') moduleGetterFoo
// If the argument is omitted, use the property name
// for each state/getter/action/mutation type
@State foo
@Getter bar
@Action baz
@Mutation qux
复制代码
若不想使用vuex定义的方法名,能够自定义属性名,由于都是定义在当前this上,因此直接使用this调用便可
this.getterFoo // -> store.getters.foo
this.actionFoo({ value: true }) // -> store.dispatch('foo', { value: true })
复制代码
搭建完成后,就能够像后端同样撸前端啦~