TLDR
Improve your Vue component performance and readability by applying the Computed Inlining pattern - a technique inspired by Martin Fowler’s Inline Function pattern. By consolidating helper functions directly into computed properties, you can reduce unnecessary abstractions and function calls, making your code more straightforward and efficient.
Introduction
Vue 3’s reactivity system is powered by computed properties that efficiently update only when their dependencies change. But sometimes we overcomplicate our components by creating too many small helper functions that only serve a single computed property. This creates unnecessary indirection and can make code harder to follow.
The Computed Inlining pattern addresses this problem by consolidating these helper functions directly into the computed properties that use them. This pattern is the inverse of Martin Fowler’s Extract Function pattern and is particularly powerful in the context of Vue’s reactive system.
Understanding Inline Function
This pattern comes from Martin Fowler’s Refactoring catalog, where he describes it as a way to simplify code by removing unnecessary function calls when the function body is just as clear as its name. You can see his original pattern here: refactoring.com/catalog/inlineFunction.html
Here’s his example:
function getRating(driver) {
return moreThanFiveLateDeliveries(driver) ? 2 : 1;
}
function moreThanFiveLateDeliveries(driver) {
return driver.numberOfLateDeliveries > 5;
}
After applying the Inline Function pattern:
function getRating(driver) {
return (driver.numberOfLateDeliveries > 5) ? 2 : 1;
}
The code becomes more direct and eliminates an unnecessary function call, while maintaining readability.
Bringing Inline Function to Vue Computed Properties
In Vue components, we often create helper functions that are only used once inside a computed property. While these can improve readability in complex cases, they can also add unnecessary layers of abstraction when the logic is simple.
Let’s look at how this pattern applies specifically to computed properties in Vue.
Before Refactoring
Here’s how a Vue component might look before applying Computed Inlining:
// src/components/OrderSummary.vue
<script setup lang="ts">
import { const computed: {
<T>(getter: ComputedGetter<T>, debugOptions?: DebuggerOptions): ComputedRef<T>;
<T, S = T>(options: WritableComputedOptions<T, S>, debugOptions?: DebuggerOptions): WritableComputedRef<T, S>;
}
computed, function ref<T>(value: T): [T] extends [Ref] ? IfAny<T, Ref<T>, T> : Ref<UnwrapRef<T>, UnwrapRef<T> | T> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref, function watch<T, Immediate extends Readonly<boolean> = false>(source: WatchSource<T>, cb: WatchCallback<T, MaybeUndefined<T, Immediate>>, options?: WatchOptions<Immediate>): WatchHandle (+3 overloads)
watch } from 'vue'
interface OrderItem {
OrderItem.id: number
id: number
OrderItem.quantity: number
quantity: number
OrderItem.unitPrice: number
unitPrice: number
OrderItem.isDiscounted: boolean
isDiscounted: boolean
}
const const orderItems: Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]>
orderItems = ref<OrderItem[]>(value: OrderItem[]): Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref<OrderItem[]>([
{ OrderItem.id: number
id: 1, OrderItem.quantity: number
quantity: 2, OrderItem.unitPrice: number
unitPrice: 100, OrderItem.isDiscounted: boolean
isDiscounted: true },
{ OrderItem.id: number
id: 2, OrderItem.quantity: number
quantity: 1, OrderItem.unitPrice: number
unitPrice: 50, OrderItem.isDiscounted: boolean
isDiscounted: false }
])
const const taxRate: Ref<number, number>
taxRate = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(0.1)
const const discountRate: Ref<number, number>
discountRate = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(0.15)
const const shippingCost: Ref<number, number>
shippingCost = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(15)
const const freeShippingThreshold: Ref<number, number>
freeShippingThreshold = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(200)
// Helper function to calculate item total
function function calculateItemTotal(item: OrderItem): number
calculateItemTotal(item: OrderItem
item: OrderItem): number {
if (item: OrderItem
item.OrderItem.isDiscounted: boolean
isDiscounted) {
return item: OrderItem
item.OrderItem.quantity: number
quantity * item: OrderItem
item.OrderItem.unitPrice: number
unitPrice * (1 - const discountRate: Ref<number, number>
discountRate.Ref<number, number>.value: number
value)
}
return item: OrderItem
item.OrderItem.quantity: number
quantity * item: OrderItem
item.OrderItem.unitPrice: number
unitPrice
}
// Helper function to sum all items
function function calculateSubtotal(): number
calculateSubtotal(): number {
return const orderItems: Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]>
orderItems.Ref<{ id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }[], OrderItem[] | { id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }[]>.value: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]
value.Array<{ id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }>.reduce<number>(callbackfn: (previousValue: number, currentValue: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}, currentIndex: number, array: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]) => number, initialValue: number): number (+2 overloads)
Calls the specified callback function for all the elements in an array. The return value of the callback function is the accumulated result, and is provided as an argument in the next call to the callback function.reduce((sum: number
sum, item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item) => {
return sum: number
sum + function calculateItemTotal(item: OrderItem): number
calculateItemTotal(item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item)
}, 0)
}
// Helper function to determine shipping
function function getShippingCost(subtotal: number): number
getShippingCost(subtotal: number
subtotal: number): number {
return subtotal: number
subtotal > const freeShippingThreshold: Ref<number, number>
freeShippingThreshold.Ref<number, number>.value: number
value ? 0 : const shippingCost: Ref<number, number>
shippingCost.Ref<number, number>.value: number
value
}
// Computed property for subtotal
const const subtotal: ComputedRef<number>
subtotal = computed<number>(getter: ComputedGetter<number>, debugOptions?: DebuggerOptions): ComputedRef<number> (+1 overload)
Takes a getter function and returns a readonly reactive ref object for the
returned value from the getter. It can also take an object with get and set
functions to create a writable ref object.computed(() => {
return function calculateSubtotal(): number
calculateSubtotal()
})
// Computed property for tax
const const tax: ComputedRef<number>
tax = computed<number>(getter: ComputedGetter<number>, debugOptions?: DebuggerOptions): ComputedRef<number> (+1 overload)
Takes a getter function and returns a readonly reactive ref object for the
returned value from the getter. It can also take an object with get and set
functions to create a writable ref object.computed(() => {
return const subtotal: ComputedRef<number>
subtotal.ComputedRef<number>.value: number
value * const taxRate: Ref<number, number>
taxRate.Ref<number, number>.value: number
value
})
// Watch for changes to update final total
const const finalTotal: Ref<number, number>
finalTotal = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(0)
watch<[ComputedRef<number>, ComputedRef<number>], true>(sources: [ComputedRef<number>, ComputedRef<number>] | readonly [ComputedRef<number>, ComputedRef<...>], cb: WatchCallback<...>, options?: WatchOptions<...> | undefined): WatchHandle (+3 overloads)
watch(
[const subtotal: ComputedRef<number>
subtotal, const tax: ComputedRef<number>
tax],
([newSubtotal: number
newSubtotal, newTax: number
newTax]) => {
const const shipping: number
shipping = function getShippingCost(subtotal: number): number
getShippingCost(newSubtotal: number
newSubtotal)
const finalTotal: Ref<number, number>
finalTotal.Ref<number, number>.value: number
value = newSubtotal: number
newSubtotal + newTax: number
newTax + const shipping: number
shipping
},
{ WatchOptions<true>.immediate?: true | undefined
immediate: true }
)
</script>
The component works but has several issues:
- Uses a watch when a computed would be more appropriate
- Has multiple helper functions that are only used once
- Splits related logic across different properties and functions
- Creates unnecessary intermediate values
After Refactoring with Computed Inlining
Now let’s apply Computed Inlining to simplify the code:
// src/components/OrderSummary.vue
<script setup lang="ts">
import { const computed: {
<T>(getter: ComputedGetter<T>, debugOptions?: DebuggerOptions): ComputedRef<T>;
<T, S = T>(options: WritableComputedOptions<T, S>, debugOptions?: DebuggerOptions): WritableComputedRef<T, S>;
}
computed, function ref<T>(value: T): [T] extends [Ref] ? IfAny<T, Ref<T>, T> : Ref<UnwrapRef<T>, UnwrapRef<T> | T> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref } from 'vue'
interface OrderItem {
OrderItem.id: number
id: number
OrderItem.quantity: number
quantity: number
OrderItem.unitPrice: number
unitPrice: number
OrderItem.isDiscounted: boolean
isDiscounted: boolean
}
const const orderItems: Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]>
orderItems = ref<OrderItem[]>(value: OrderItem[]): Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref<OrderItem[]>([
{ OrderItem.id: number
id: 1, OrderItem.quantity: number
quantity: 2, OrderItem.unitPrice: number
unitPrice: 100, OrderItem.isDiscounted: boolean
isDiscounted: true },
{ OrderItem.id: number
id: 2, OrderItem.quantity: number
quantity: 1, OrderItem.unitPrice: number
unitPrice: 50, OrderItem.isDiscounted: boolean
isDiscounted: false }
])
const const taxRate: Ref<number, number>
taxRate = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(0.1)
const const discountRate: Ref<number, number>
discountRate = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(0.15)
const const shippingCost: Ref<number, number>
shippingCost = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(15)
const const freeShippingThreshold: Ref<number, number>
freeShippingThreshold = ref<number>(value: number): Ref<number, number> (+1 overload)
Takes an inner value and returns a reactive and mutable ref object, which
has a single property `.value` that points to the inner value.ref(200)
const const orderTotal: ComputedRef<number>
orderTotal = computed<number>(getter: ComputedGetter<number>, debugOptions?: DebuggerOptions): ComputedRef<number> (+1 overload)
Takes a getter function and returns a readonly reactive ref object for the
returned value from the getter. It can also take an object with get and set
functions to create a writable ref object.computed(() => {
// Calculate subtotal with inline discount logic
const const subtotal: number
subtotal = const orderItems: Ref<{
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[], OrderItem[] | {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]>
orderItems.Ref<{ id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }[], OrderItem[] | { id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }[]>.value: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]
value.Array<{ id: number; quantity: number; unitPrice: number; isDiscounted: boolean; }>.reduce<number>(callbackfn: (previousValue: number, currentValue: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}, currentIndex: number, array: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}[]) => number, initialValue: number): number (+2 overloads)
Calls the specified callback function for all the elements in an array. The return value of the callback function is the accumulated result, and is provided as an argument in the next call to the callback function.reduce((sum: number
sum, item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item) => {
const const itemTotal: number
itemTotal = item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item.isDiscounted: boolean
isDiscounted
? item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item.quantity: number
quantity * item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item.unitPrice: number
unitPrice * (1 - const discountRate: Ref<number, number>
discountRate.Ref<number, number>.value: number
value)
: item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item.quantity: number
quantity * item: {
id: number;
quantity: number;
unitPrice: number;
isDiscounted: boolean;
}
item.unitPrice: number
unitPrice
return sum: number
sum + const itemTotal: number
itemTotal
}, 0)
// Calculate tax
const const tax: number
tax = const subtotal: number
subtotal * const taxRate: Ref<number, number>
taxRate.Ref<number, number>.value: number
value
// Determine shipping cost inline
const const shipping: number
shipping = const subtotal: number
subtotal > const freeShippingThreshold: Ref<number, number>
freeShippingThreshold.Ref<number, number>.value: number
value ? 0 : const shippingCost: Ref<number, number>
shippingCost.Ref<number, number>.value: number
value
return const subtotal: number
subtotal + const tax: number
tax + const shipping: number
shipping
})
</script>
The refactored version:
- Consolidates all pricing logic into a single computed property
- Eliminates the need for a watch by using Vue’s reactive system properly
- Removes unnecessary helper functions and intermediate values
- Makes the data flow more clear and direct
- Reduces the number of reactive dependencies being tracked
Best Practices
- Apply Computed Inlining when the helper function is only used once
- Use this pattern when the logic is simple enough to be understood inline
- Add comments to clarify steps if the inline logic is non-trivial
- Keep computed properties focused on a single responsibility, even after inlining
- Consider keeping functions separate if they’re reused or complex
When to Use Computed Inlining
- When the helper functions are only used by a single computed property
- When performance is critical (eliminates function call overhead)
- When the helper functions don’t significantly improve readability
- When you want to reduce the cognitive load of jumping between functions
- When debugging and following the execution flow is important
When to Avoid Computed Inlining
- When the helper function is used in multiple places
- When the logic is complex and the function name significantly improves clarity
- When the function might need to be reused in the future
- When testing the helper function independently is important
Conclusion
The Computed Inlining pattern in Vue is a practical application of Martin Fowler’s Inline Function refactoring technique. It helps streamline your reactive code by:
- Reducing unnecessary abstractions
- Eliminating function call overhead
- Making execution flow more direct and easier to follow
- Keeping related logic together in one place
While not appropriate for every situation, Computed Inlining is a valuable tool in your Vue refactoring toolkit, especially when optimizing components with many small helper functions.
Try applying Computed Inlining in your next Vue component refactoring, and see how it can make your code both simpler and more efficient.