引言
近幾年前端對(duì) TypeScript的呼聲越來越高,Typescript也成為了前端必備的技能董栽。TypeScript 是 JS類型的超集恼策,并支持了泛型鸦致、類型、命名空間涣楷、枚舉等特性分唾,彌補(bǔ)了 JS 在大型應(yīng)用開發(fā)中的不足。
在單獨(dú)學(xué)習(xí) TypeScript時(shí)狮斗,你會(huì)感覺很多概念還是比較好理解的绽乔,但是和一些框架結(jié)合使用的話坑還是比較多的,例如使用 React情龄、Vue 這些框架的時(shí)候與 TypeScript 的結(jié)合會(huì)成為一大障礙迄汛,需要去查看框架提供的.d.ts的聲明文件中一些復(fù)雜類型的定義、組件的書寫方式等都要做出不小的調(diào)整骤视。
本篇文章主要是結(jié)合我的經(jīng)驗(yàn)和大家聊一下如何在Vue
中平滑的從js
過渡到ts
鞍爱,閱讀本文建議對(duì) TypeScript 有一定了解,因?yàn)槲闹袑?duì)于一些 TypeScript 的基礎(chǔ)的知識(shí)不會(huì)有太過于詳細(xì)的講解专酗。(具體可以參考官方文檔https://www.w3cschool.cn/typescript/typescript-tutorial.html
睹逃,官方文檔就是最好的入門手冊(cè))
構(gòu)建
通過官方腳手架構(gòu)建安裝
# 1. 如果沒有安裝 Vue CLI 就先安裝
npm install --global @vue/cli
最新的Vue CLI
工具允許開發(fā)者 使用 TypeScript 集成環(huán)境 創(chuàng)建新項(xiàng)目。
只需運(yùn)行vue create my-app
祷肯。
然后沉填,命令行會(huì)要求選擇預(yù)設(shè)。使用箭頭鍵選擇 Manually select features佑笋。
接下來翼闹,只需確保選擇了 TypeScript 和 Babel 選項(xiàng),如下圖:
然后配置其余設(shè)置蒋纬,如下圖:
設(shè)置完成 vue cli 就會(huì)開始安裝依賴并設(shè)置項(xiàng)目猎荠。
目錄解析
安裝完成打開項(xiàng)目,你會(huì)發(fā)現(xiàn)集成 ts 后的項(xiàng)目目錄結(jié)構(gòu)是這樣子的:
|-- ts-vue
|-- .browserslistrc # browserslistrc 配置文件 (用于支持 Autoprefixer)
|-- .eslintrc.js # eslint 配置
|-- .gitignore
|-- babel.config.js # babel-loader 配置
|-- package-lock.json
|-- package.json # package.json 依賴
|-- postcss.config.js # postcss 配置
|-- README.md
|-- tsconfig.json # typescript 配置
|-- vue.config.js # vue-cli 配置
|-- public # 靜態(tài)資源 (會(huì)被直接復(fù)制)
| |-- favicon.ico # favicon圖標(biāo)
| |-- index.html # html模板
|-- src
| |-- App.vue # 入口頁面
| |-- main.ts # 入口文件 加載組件 初始化等
| |-- shims-tsx.d.ts
| |-- shims-vue.d.ts
| |-- assets # 主題 字體等靜態(tài)資源 (由 webpack 處理加載)
| |-- components # 全局組件
| |-- router # 路由
| |-- store # 全局 vuex store
| |-- styles # 全局樣式
| |-- views # 所有頁面
|-- tests # 測(cè)試
其實(shí)大致看下來,與之前用js
構(gòu)建的項(xiàng)目目錄沒有什么太大的不同蜀备,區(qū)別主要是之前 js 后綴的現(xiàn)在改為了ts后綴关摇,還多了tsconfig.json
、shims-tsx.d.ts
碾阁、shims-vue.d.ts
這幾個(gè)文件输虱,那這幾個(gè)文件是干嘛的呢:
-
tsconfig.json
: typescript配置文件,主要用于指定待編譯的文件和定義編譯選項(xiàng) -
shims-tsx.d.ts
: 允許.tsx 結(jié)尾的文件,在 Vue 項(xiàng)目中編寫 jsx 代碼 -
shims-vue.d.ts
: 主要用于 TypeScript 識(shí)別.vue 文件脂凶,Ts 默認(rèn)并不支持導(dǎo)入 vue 文件
使用
開始前我們先來了解一下在 vue 中使用 typescript 非常好用的幾個(gè)庫
-
vue-class-component
:vue-class-component
是一個(gè) Class Decorator,也就是類的裝飾器 -
vue-property-decorator
:vue-property-decorator
是基于 vue 組織里 vue-class-component 所做的拓展import { Vue, Component, Inject, Provide, Prop, Model, Watch, Emit, Mixins } from 'vue-property-decorator'
-
vuex-module-decorators
: 用 typescript 寫 vuex 很好用的一個(gè)庫import { Module, VuexModule, Mutation, Action, MutationAction, getModule } from 'vuex-module-decorators'
組件聲明
創(chuàng)建組件的方式變成如下
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
@Component
export default class Test extends Vue {
}
data 對(duì)象
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
@Component
export default class Test extends Vue {
private name: string;
}
Prop 聲明
@Prop({ default: false }) private isCollapse!: boolean;
@Prop({ default: true }) private isFirstLevel!: boolean;
@Prop({ default: "" }) private basePath!: string;
- !: 表示一定存在宪睹,?: 表示可能不存在愁茁。這兩種在語法上叫賦值斷言
- @Prop(options: (PropOptions | Constructor[] | Constructor) = {})
- PropOptions,可以使用以下選項(xiàng):type横堡,default埋市,required,validator
- Constructor[]命贴,指定 prop 的可選類型
- Constructor道宅,例如 String,Number胸蛛,Boolean 等污茵,指定 prop 的類型
method
js 下是需要在 method 對(duì)象中聲明方法,現(xiàn)變成如下
public clickFunc(): void {
console.log(this.name)
console.log(this.msg)
}
Watch 監(jiān)聽屬性
@Watch("$route", { immediate: true })
private onRouteChange(route: Route) {
const query = route.query as Dictionary<string>;
if (query) {
this.redirect = query.redirect;
this.otherQuery = this.getOtherQuery(query);
}
}
- @Watch(path: string, options: WatchOptions = {})
- options 包含兩個(gè)屬性 immediate?:boolean 偵聽開始之后是否立即調(diào)用該回調(diào)函數(shù) / deep?:boolean 被偵聽的對(duì)象的屬性被改變時(shí)葬项,是否調(diào)用該回調(diào)函數(shù)
- @Watch('arr', { immediate: true, deep: true })
onArrChanged(newValue: number[], oldValue: number[]) {}
computed 計(jì)算屬性
public get allname() {
return 'computed ' + this.name;
}
allname 是計(jì)算后的值泞当,name 是被監(jiān)聽的值
生命周期函數(shù)
public created(): void {
console.log('created');
}
public mounted():void{
console.log('mounted')
}
emit 事件
import { Vue, Component, Emit } from 'vue-property-decorator'
@Component
export default class MyComponent extends Vue {
count = 0
@Emit()
addToCount(n: number) {
this.count += n
}
@Emit('reset')
resetCount() {
this.count = 0
}
@Emit()
returnValue() {
return 10
}
@Emit()
onInputChange(e) {
return e.target.value
}
@Emit()
promise() {
return new Promise(resolve => {
setTimeout(() => {
resolve(20)
}, 0)
})
}
}
使用 js 寫法
export default {
data() {
return {
count: 0
}
},
methods: {
addToCount(n) {
this.count += n
this.$emit('add-to-count', n)
},
resetCount() {
this.count = 0
this.$emit('reset')
},
returnValue() {
this.$emit('return-value', 10)
},
onInputChange(e) {
this.$emit('on-input-change', e.target.value, e)
},
promise() {
const promise = new Promise(resolve => {
setTimeout(() => {
resolve(20)
}, 0)
})
promise.then(value => {
this.$emit('promise', value)
})
}
}
}
- @Emit(event?: string)
- @Emit 裝飾器接收一個(gè)可選參數(shù),該參數(shù)是$Emit 的第一個(gè)參數(shù)民珍,充當(dāng)事件名襟士。如果沒有提供這個(gè)參數(shù),$Emit 會(huì)將回調(diào)函數(shù)名的 camelCase 轉(zhuǎn)為 kebab-case嚷量,并將其作為事件名
- @Emit 會(huì)將回調(diào)函數(shù)的返回值作為第二個(gè)參數(shù)陋桂,如果返回值是一個(gè) Promise 對(duì)象,$emit 會(huì)在 Promise 對(duì)象被標(biāo)記為 resolved 之后觸發(fā)
- @Emit 的回調(diào)函數(shù)的參數(shù)蝶溶,會(huì)放在其返回值之后嗜历,一起被$emit 當(dāng)做參數(shù)使用
vuex
在使用 store 裝飾器之前,先過一下傳統(tǒng)的 store 用法吧
export default {
namespaced:true,
state:{
foo:""
},
getters:{
getFoo(state){ return state.foo}
},
mutations:{
setFooSync(state,payload){
state.foo = payload
}
},
actions:{
setFoo({commit},payload){
commot("getFoo",payload)
}
}
}
然后開始使用vuex-module-decorators
import {
VuexModule,
Mutation,
Action,
getModule,
Module
} from "vuex-module-decorators";
-
VuexModule
用于基本屬性export default class TestModule extends VuexModule { }
VuexModule 提供了一些基本屬性抖所,包括 namespaced,state,getters,modules,mutations,actions,context
-
@Module
標(biāo)記當(dāng)前為 module@Module({ dynamic: true, store, name: "settings" }) class Settings extends VuexModule implements ISettingsState { }
module 本身有幾種可以配置的屬性:
- namespaced:boolean 啟/停用 分模塊
- stateFactory:boolean 狀態(tài)工廠
- dynamic:boolean 在 store 創(chuàng)建之后梨州,再添加到 store 中。開啟 dynamic 之后必須提供下面的屬性
- name:string 指定模塊名稱
- store:Vuex.Store 實(shí)體 提供初始的 store
-
@Mutation
標(biāo)注為 mutation@Mutation private SET_NAME(name: string) { // 設(shè)置用戶名 this.name = name; }
-
@Action
標(biāo)注為 action@Action public async Login(userInfo: { username: string; password: string }) { // 登錄接口田轧,拿到token let { username, password } = userInfo; username = username.trim(); const { data } = await login({ username, password }); setToken(data.accessToken); this.SET_TOKEN(data.accessToken); }
-
getModule
得到一個(gè)類型安全的 store暴匠,module 必須提供 name 屬性export const UserModule = getModule(User);
示例
我之前基于 ts+vue+element 構(gòu)建了一個(gè)簡(jiǎn)單的中后臺(tái)通用模板。
涵蓋的功能如下:
- 登錄 / 注銷
- 權(quán)限驗(yàn)證
- 頁面權(quán)限
- 權(quán)限配置
- 多環(huán)境發(fā)布
- Dev / Stage / Prod
- 全局功能
- 動(dòng)態(tài)換膚
- 動(dòng)態(tài)側(cè)邊欄(支持多級(jí)路由嵌套)
- Svg 圖標(biāo)
- 全屏
- 設(shè)置
- Mock 數(shù)據(jù) / Mock 服務(wù)器
- 組件
- ECharts 圖表
- 表格
- 復(fù)雜表格
- 控制臺(tái)
- 引導(dǎo)頁
- 錯(cuò)誤頁面
- 404
里面對(duì)于在 vue 中使用 typescript 的各種場(chǎng)景都有很好的實(shí)踐傻粘,大家感興趣的可以參考一下巷查,https://github.com/FSFED/ts-vue
,當(dāng)然不要吝惜你的 starDㄍ取!旭寿!