JavaScript in Plain English

New JavaScript and Web Development content every day. Follow to join our 3.5M+ monthly readers.

Follow publication

Vue 3 Programmatic Component Design

Abhimanyu Chauhan
JavaScript in Plain English
4 min readJan 22, 2022

--

Vue 3 Programmatic Component Design
<template>
<h1 v-if="$props.as === 'h1'" :class="classes" :style="$props.style">
<slot></slot>
</h1>
<h2 v-if="$props.as === 'h2'" :class="classes" :style="$props.style">
<slot></slot>
</h2>
<p v-if="$props.as === 'p'" :class="classes" :style="$props.style">
<slot></slot>
</p>
</template>

Functional Component Pattern

<script lang="ts">
import { computed, CSSProperties, defineComponent, h, PropType } from "vue";
export default defineComponent({
name: "TypographyFunctional",
props: {
as: {type: String as PropType<"h1" | "h2" | "h3" | "h4" | "h5" | "h6" | "p">, default: "p"},
style: { type: String as PropType<CSSProperties>, default:()=>{})},class: {type: String,default: ""},fontWeight: {type: String as PropType<"bold" | "light">,default:"light"}},setup(props) {
const classes = computed(() => {
return `${props.fontWeight==="light"? "light" : "bold"} ${props.class}`;});
return { classes };
},
render() {
return h(this.as,
{ class: this.classes, style: this.style,},
this.$slots.default?.()
);
},
});
</script>
<template>
<TypographyFunctionalVue :as="'h1'" class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
<TypographyFunctionalVue :as="'h2'" class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
<TypographyFunctionalVue :as="'h3'" class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
<TypographyFunctionalVue :as="'h4'" class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
<TypographyFunctionalVue :as="'h5'" class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
<TypographyFunctionalVue class="border margin"
:font-weight="'bold'" :style="{ color: 'green' }">
This is a functional component
</TypographyFunctionalVue>
</template>
rendered HTML

Stateful Component Pattern

<component is="tag-name"></component>
<template>
<component :is="$props.as" :class="classes" :style="$props.style"
<slot></slot>
</component>
</template>
<script lang="ts">
import { computed, CSSProperties, defineComponent, h, PropType } from "vue";
export default defineComponent({
name: "Typography",
props: {
as: {type: String as PropType<"h1" | "h2" | "h3" | "h4" | "h5" | "h6" | "p">, default: "p"},
style: { type: String as PropType<CSSProperties>, default:()=>{})},class: {type: String,default: ""},fontWeight: {type: String as PropType<"bold" | "light">,default:"light"}},setup(props) {
const classes = computed(() => {
return `${props.fontWeight==="light"? "light" : "bold"} ${props.class}`;});
return { classes };
},
});</script>

--

--

Published in JavaScript in Plain English

New JavaScript and Web Development content every day. Follow to join our 3.5M+ monthly readers.

Written by Abhimanyu Chauhan

JS Enthusiast, with a passion for travelling and writing.

Responses (3)

Write a response