SoFunction
Updated on 2025-04-03

vue-class-setup Write class style combination API

Preface

Our company is basedvue-class-componentThere are hundreds of projects developed, of which nearly 100 were deployed. At the beginning of such a huge scale, the project was fantasized about whether to upgrade Vue3. After a survey, I found out thatvue-class-componentIt has been two years since the last version of Vue3's support, and the official version has not been released yet. Currently, it is basically in a state of unmaintained, and there are a lot of destructive updates in the upgrade. I still have reservations about whether to continue using Vue3 in the future, but it does not prevent us from making the component library common to Vue2 and Vue3 first, so this article comes into being.

In the past three years,vue-class-componentThe biggest problem is that the component's parameters and event type cannot be correctly verified, which has brought me a huge shadow. After some research, I was surprised to find it.defineComponentThe defined components can correctly identify the types in Vue2.7 and both, so first plan the internal component library to support Vue2 and Vue3 at the same time. If you continue to use Vue3 later, it will be much easier.

So, back to the beginning, investigatedvue-class-componentIn Vue3 support, the latest version is8.0.0-rc.1, and was disappointed. At present, it is basically in a state of unmaintained. There is no one in the community that can meet my needs, and it supports Vue2 and Vue3 at the same time.

Birth idea

In view ofvue-class-componentComponents cannot currently perform the correct component type verification. When I was surprised to find that the code written by the combination API can be correctly identified, an idea of ​​using the class style to write a combination API was born. So I spent a month of practice, stepped on all the pitfalls, and finally it was born.vue-class-setup, a library that uses class style to write code, it gzip compressed, 1kb size.

Start quickly

npm install vue-class-setup
<script lang="ts">
import { defineComponent } from 'vue';
import { Setup, Context } from 'vue-class-setup';
// Setup and Context must work together@Setup
class App extends Context {
    private _value = 0;
    public get text() {
        return String(this._value);
    }
    public set text(text: string) {
        this._value = Number(text);
    }
    public onClick() {
        this._value++;
    }
}
export default defineComponent({
    // Logic for injecting class instances    ...(),
});
</script>
<template>
    <div>
        <p>{{ text }}</p>
        <button @click="onClick()"></button>
    </div>
</template>

Try many more solutions, and finally adopt the above form as best practice, it cannot do itexport defaultExport a class directly, you must usedefineComponentLet's wrap a layer, because it's just oneCombination Class (API), not a component.

Best Practices

<script lang="ts">
import { defineComponent } from 'vue';
import { Setup, Define } from 'vue-class-setup';
// Pass in the component's Props and Emit to get the correct `Props` and `Emit` types in combination@Setup
class App extends Define<Props, Emit> {
    // ✨ You can directly define the default value of Props here, without using a Prop decorator like vue-property-decorator    public readonly dest = '--';
    // Automatically convert to Vue's 'compute'    public get text() {
        return String();
    }
    public click(evt: MouseEvent) {
        // Launch event, can correctly identify the type        this.$emit('click', evt);
    }
}
/**
  * Here is another example of using in the setup function. It is recommended to use `defineComponent` by default.
  * If there are multiple class instances, you can also instantiate the class in the setup
  * <script lang="ts" setup>
  * const app = new App();
  * <\/script>
  * <template>
  * <div>{{ }}</div>
  * </template>
  */
export default defineComponent({
    ...(),
});
&lt;/script&gt;
&lt;script lang="ts" setup&gt;
// If you define a type in the setup, you need to export itexport interface Props {
    value: number;
    dest?: string;
}
export interface Emit {
    (event: 'click', evt: MouseEvent): void;
}
// There is no need to use variables to receive here. You can use Vue's compiled macros to generate the correct Props and Emits for the component.// ❌ const props = defineProps&lt;Props&gt;();
// ❌ const emit = defineEmits&lt;Emit&gt;();
defineProps&lt;Props&gt;(); //  ✅
defineEmits&lt;Emit&gt;(); //  ✅
// This default value definition is no longer recommended, but is declared directly on the class// ❌ withDefaults(defineProps&lt;Props&gt;(), { dest: '--' });
// ✅ @Setup
// ✅ class App extends Define&lt;Props, Emit&gt; {
// ✅     public readonly dest = '--'
// ✅ }
// Setup decorator will automatically use reactive to wrap the class when the class is instantiated.// If you instantiate it manually in setup, you don't need to perform reactive again// const app = reactive(new App()); // ❌
// const app = new App();           // ✅
&lt;/script&gt;
&lt;template&gt;
    &lt;button class="btn" @click="click($event)"&gt;
        &lt;span class="text"&gt;{{ text }}&lt;/span&gt;
        &lt;span class="props-dest"&gt;{{ dest }}&lt;/span&gt;
        &lt;span class="props-value"&gt;{{ $ }}&lt;/span&gt;
    &lt;/button&gt;
&lt;/template&gt;

Multiple Class Instances

In some complex business, multiple instances are sometimes required

<script lang="ts">
import { onBeforeMount, onMounted } from 'vue';
import { Setup, Context, PassOnTo } from 'vue-class-setup';
@Setup
class Base extends Context {
    public value = 0;
    public get text() {
        return String();
    }
    @PassOnTo(onBeforeMount)
    public init() {
        ++;
    }
}
@Setup
class Left extends Base {
    public left = 0;
    public get text() {
        return String(`value:${}`);
    }
    public init() {
        ();
        ++;
    }
    @PassOnTo(onMounted)
    public initLeft() {
        ++;
    }
}
@Setup
class Right extends Base {
    public right = 0;
    public init() {
        ();
        ++;
    }
    @PassOnTo(onMounted)
    public initLeft() {
        ++;
    }
}
</script>
<script setup lang="ts">
const left = new Left();
const right = new Right();
</script>
<template>
    <p class="left">{{  }}</p>
    <p class="right">{{  }}</p>
</template>

PassOnTo

After the class instance is ready, the PassOnTo decorator will pass the corresponding function to the callback, so that we can smoothly sum itonMountedWait for the hook to work together

import { onMounted } from 'vue';
@Setup
class App extends Define {
    @PassOnTo(onMounted)
    public onMounted() {}
}

Watch

In usevue-property-decoratorofWatchWhen decorator, it will receive a string type, which cannot correctly identify whether the class instance exists, but nowvue-class-setupCan check whether your type is correct. If you pass in a field that does not exist in a class instance, the type will report an error.

<script lang="ts">
import { Setup, Watch, Context } from 'vue-class-setup';
@Setup
class App extends Context {
    public value = 0;
    public immediateValue = 0;
    public onClick() {
        ++;
    }
    @Watch('value')
    public watchValue(value: number, oldValue: number) {
        if (value > 100) {
             = 100;
        }
    }
    @Watch('value', { immediate: true })
    public watchImmediateValue(value: number, oldValue: number | undefined) {
        if (typeof oldValue === 'undefined') {
             = 10;
        } else {
            ++;
        }
    }
}
</script>
<script setup lang="ts">
const app = new App();
</script>
<template>
    <p class="value">{{  }}</p>
    <p class="immediate-value">{{  }}</p>
    <button @click="()">Add</button>
</template>

defineExpose

In some scenarios, we hope to expose some methods and properties of components, so we need to usedefineExposeCompile the macro to define the export, so a.useThe static class method helps you get the currently injected class instance

<script lang="ts">
import { defineComponent } from 'vue';
import { Setup, Context } from 'vue-class-setup';
@Setup
class App extends Context {
    private _value = 0;
    public get text() {
        return String(this._value);
    }
    public set text(text: string) {
        this._value = Number(text);
    }
    public addValue() {
        this._value++;
    }
}
export default defineComponent({
    ...(),
});
</script>
<script lang="ts" setup>
const app = ();
defineExpose({
    addValue: ,
});
</script>
<template>
    <div>
        <p class="text">{{ text }}</p>
        <p class="text-eq">{{  === text }}</p>
        <button @click="addValue"></button>
    </div>
</template>

Why use class?

Actually, I don’t really want to discuss this issue. I will naturally like those who like them, and I will naturally dislike those who don’t like them. There is no way in the world, and when there are more people walking, there will be a way.

at last

Whether it is an option API or a combination API, the code is written by people. Others say that Vue is not competent for large-scale projects, but it has withstood the practice in our company's practice and basically does not produce thousands of lines of component code.

If you like to write code using class style, you might as well pay attention to it

  • vue-class-setup

If your business is complex and needs to use SSR and microservice architecture, you might as well pay attention to it

  • vue-genesis

The above is the detailed content of vue-class-setup writing class-style combination API. For more information about vue-class-setup writing combination APIs, please pay attention to my other related articles!