這次帶給大家使用Vue Mixin功能步驟詳解,使用Vue Mixin功能的注意事項有哪些,以下就是實戰案例,一起來看一下。
轉到用 Typescript 寫 Vue 應用以後,經過一輪工具鍊和依賴的洗禮,總算蹣跚地能走起來了,不過有一個很常用的功能 mixin,似乎還沒有官方的解決方案。
既想享受 mixin 的靈活和方便,又想收穫 ts 的類型系統帶來的安全保障和開發時使用 IntelliSense 的順滑體驗。
vuejs 官方組織裡有一個 'vue-class-component'
以及連帶推薦的 'vue-property-decorator'
,都沒有相應實作。翻了下前者的 issue,有一條掛了好些時間的待做 feature 是 mixin 的支持。
也不是什麼複雜的事,自己寫一個吧。
後註:vue-class-component 6.2.0 開始提供 mixins 方法,和本文的實作思路相似。
實作
import Vue, { VueConstructor } from 'vue' export type VClass<T> = { new(): T } & Pick<VueConstructor, keyof VueConstructor> /** * mixins for class style vue component */ function Mixins<A>(c: VClass<A>): VClass<A> function Mixins<A, B>(c: VClass<A>, c1: VClass<B>): VClass<A&B> function Mixins<A, B, C>(c: VClass<A>, c1: VClass<B>, c2: VClass<C>): VClass<A&B&C> function Mixins<T>(c: VClass<T>, ...traits: Array<VClass<T>>): VClass<T> { return c.extend({ mixins: traits }) }
宣告 VClass
function Mixins<T>(c: VClass<T>, ...traits: Array<VClass<T>>): VClass<T> { return c.extend({ mixins: traits }) }
至於 ABC 這個純粹是類型宣告的體力活了。
使用
實際使用時:
import { Component, Vue } from 'vue-property-decorator' import { Mixins } from '../../util/mixins' @Component class PageMixin extends Vue { title = 'Test Page' redirectTo(path: string) { console.log('calling reidrectTo', path) this.$router.push({ path }) } } interface IDisposable { dispose(...args: any[]): any } class DisposableMixin extends Vue { _disposables: IDisposable[] created() { console.log('disposable mixin created'); this._disposables = [] } beforeDestroy() { console.log('about to clear disposables') this._disposables.map((d) => { d.dispose() }) delete this._disposables } registerDisposable(d: IDisposable) { this._disposables.push(d) } } @Component({ template: ` <p> <h1>{{ title }}</h1> <p>Counted: {{ counter }}</p> </p> ` }) export default class TimerPage extends Mixins(PageMixin, DisposableMixin) { counter = 0 mounted() { const timer = setInterval(() => { if (this.counter++ >= 3) { return this.redirectTo('/otherpage') } console.log('count to', this.counter); }, 1000) this.registerDisposable({ dispose() { clearInterval(timer) } }) } } count to 1 count to 2 count to 3 calling reidrectTo /otherpage about to clear disposables
#注意到直接extends Vue 的DisposableMixin 並不是一個有效的Vue 元件,也不可以直接在mixins 選項會使用,如果要以Vue.extend 方式擴充的自定義元件使用,請記住使用Component 包裝一層。
const ExtendedComponent = Vue.extend({ name: 'ExtendedComponent', mixins: [Component(DisposableMixin)], })
Abstract class
在業務系統中會使用到的Mixin 其實多數情況下會更複雜,提供一些基礎功能,但有些部分需要留給繼承者自行實現,這個時候使用抽象類別就很合適。
abstract class AbstractMusicPlayer extends Vue { abstract audioSrc: string playing = false togglePlay() { this.playing = !this.playing } } class MusicPlayerA extends AbstractMusicPlayer { audioSrc = '/audio-a.mp3' } class MusicPlayerB extends AbstractMusicPlayer { staticBase = '/statics' get audioSrc() { return `${this.staticBase}/audio-b.mp3` } }
但抽象類別是無法被實例化的,並不滿足{ new(): T }
這個要求,因此只能被繼承,而不能被混入,由於同樣的原因,抽象類別也無法被'vue-class-component'
的Component 函式裝飾。
這時候只好將實作了的功能寫入 Mixin 中,待實現的功能放到介面裡,讓具體類別來實現。
interface IMusicSourceProvider { audioSrc: string } /** * @implements IPlayerImplementation */ class PlayerMixin extends Vue { /** @abstract */ audioSrc: string logSrc() { console.log(this.audioSrc) } } interface IPlayerImplementation extends IMusicSourceProvider {} class RealPlayer extends Mixins(PlayerMixin) implements IPlayerImplementation { audioSrc = '/audio-c.mp3' }
這種欺騙編譯器的方式其實還是比較拙劣的,如果一個具體類別繼承了 PlayerMixin,卻沒有顯示聲明實現 IPlayerImplementation ,編譯器無法告訴你這個錯誤。我們只能在程式碼裡小心翼翼地寫上註釋,期待使用者不要忘了這件事。
相信看了本文案例你已經掌握了方法,更多精彩請關注php中文網其它相關文章!
推薦閱讀:
以上是使用Vue Mixin功能步驟詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!