Guidelines
Dos and Don'ts
import { defineProps, defineEmits, onMounted, watch } from 'vue';
import { type Use, ivue, iref, iuse } from 'ivue';
import { useMouse } from '@vueuse/core';
type UseMouse = Use<typeof useMouse>;
interface CounterProps {
initialCount: number;
}
interface CounterEmits {
(e: 'increment', count: number): void;
}
const props = defineProps<CounterProps>();
const emit = defineEmits<CounterEmits>();
/**
* Example of a properly defined ivue class.
*/
class Counter {
/** ✓ Properly declared unwrapped composable. */
mouse: UseMouse;
/**
* ✓ Properly declared DOM Ref.
* ✓ Custom declared type of HTMLElement or null
*/
spanElementRef = iref<HTMLElement | null>(null);
constructor(public props: CounterProps, public emit: CounterEmits) {
/** ✓ Properly declared auto-unwrapped composable. */
this.mouse = iuse(useMouse);
}
/** ✓ Properly declared init function. */
init() {
/** ✓ Properly set lifecycle hook. */
onMounted(() => {
this.count = 4;
});
/** ✓ Properly set watch function */
watch(
() => this.count,
(newCount: number) => {
if (newCount === 5) {
alert('You reached the count of ' + newCount + '!');
}
}
);
}
/**
* Use iref() for the property to be auto cast to number
* because refs auto-unwrap inside reactive().
*/
count = iref(0); // ✓ Properly inferred type -> number
/**
* Use iref() for the property to be auto cast to number
* because refs auto-unwrap inside reactive().
*/
timesClicked = iref(0); // ✓ Properly inferred type -> number
/** ✓ Properly declared function (not arrow function). */
increment() {
this.count++;
}
/** ✓ Properly declared function (not arrow function). */
click() {
this.increment();
this.timesClicked++;
}
/** ✓ Properly declared computed getter. */
get doubleCount() {
return this.count * 2;
}
}
/** ✓ Properly initialized IVUE class runner. */
const counter = ivue(Counter, props, emit);
counter.count;
// ttest
Use ref() for properties
ivue
recommends all class properties to be defined as ref()
to be able to interoperate with defineExpose()
, if you simply pass reactive props which are not Refs through defineExpose()
, they will lose reactivity. ref()
refs just like computed refs get flattened into the reactive()
object, so there is no need to worry about using .value
. The ref()
refs are necessary just internally for Vue 3 to know which refs to keep reactive.
Unwrap (de-Ref) the types
Next, we convert the types back to their normal types as if they have no reactivity at all, so Ref<number>
is number
in ivue
, so rather than going in the direction of complexifying the types, we are going in the opposite direction towards simplification.
Use standard declaration syntax for functions
ivue
recommends all class functions to be defined in plain full function style (not arrow functions), this allows all ivue
classes to be extensible at any point. By using plain standard functions, it allows the developer to be able to override them at any time by simply extending the class.
Do not use arrow functions in class declarations
WARNING
Arrow functions break full extensibility of classes because they carry their own context at the point of declaration, so avoid using them inside of ivue
classes.
constructor()
vs .init()
Use constructor()
to assign properties of the class and cast Refs to Unwrapped bare types.
Use .init()
to declare reactive state functions like watch
, watchEffect
, and lifecycle hooks like onMounted
, onBeforeMount
etc, do assignments of reactive properties, since .init()
already has access to reactive()
state through this
.
Inside the constructor()
method you still have access to non-reactive state, because when constructor()
is initialized, it does NOT yet have access to the reactive properties of the class, since it was not yet converted to reactive()
by ivue
, so if you use the properties like Refs or ComputedRefs inside constructor()
you would have to use them with the .value
.
As a general rule, there is no need to manipulate the values in the constructor()
, use constructor only for assigning the properties and casting the types of those assigned properties to the unwrapped (de-Refed) final state of the resulting reactive()
object.
Let's look at a constructor()
vs .init()
example:
<script setup lang="ts">
import { onMounted, ref } from 'vue';
import { ivue, iref, iuse } from 'ivue';
class Counter {
constructor(public span?: HTMLElement) {
// ❌ Do not do this in the constructor() because this.span
// still refers to this.span.value here:
// onMounted(() => {
// (this.span as HTMLElement /*❌*/).innerHTML = 'Initial span text!';
// });
}
init() {
// ✅ Do this inside init() because this.span (.value) is now
// flattened & (this) is now reactive():
onMounted(() => {
(this.span as HTMLElement/*✅*/).innerHTML = 'Initial span text!';
});
}
count = iref(0);
increment() {
this.count++;
(this.span as HTMLElement).innerHTML = String(this.count + 1);
}
}
const span = ref<HTMLElement>();
const counter = ivue(
Counter,
iuse(span)
);
defineExpose<Counter>(counter);
</script>
<template>
<a href="javascript:void(0)" @click="() => counter.increment()">Increment</a><br />
Count: {{ counter.count }} <br />
<span ref="span"></span>
</template>
For this example we initialize the component like this:
<template>
<CounterExternalRefsDetailed />
</template>
Count: 0
Unwrapping Refs
The key to working with ivue
is understanding correctly the way Vue 3 does automatic Unwrapping of Refs when they are passed into the reactive()
object. In that regard we are not relying on some magic ivue
behavior but rather the default behavior of reactive()
Vue 3 function.
To match that unwrapping behavior, our class needs to Unwrap (or de-Ref) the types of Composables, Refs, ComputedRefs, if they are being passed into the constructor, to get their raw basic types those Refs are pointing to, so that you can start operating in the ivue
environment, where there is no need to worry about .value
. This unwrapping should be mainly done inside the constructor()
method.
Naming Conventions
To benefit from the full power of ivue
, it is recommended to extract the classes into separate files. What has been an effective pattern is to name the classes and put them right beside components in the same folder that these classes are being used with. So if you have CounterComponent.vue
component, it can have a class inside CounterComponentClass.ts
, and you can store props, emits, and other runtime definitions inside CounterComponentProps.ts
.