Plugins
WARNING
The plugin system is still under development and the API is subject to change. If you want to develop a plugin, please open a discussion to share your progress and issues you might encounter.
Plugins will likely have to interact with the Query Cache, more specifically, listen to actions with $onAction()
:
It is recommended to create plugins as functions to accept any options and return the plugin itself. This way, you can pass options to the plugin when creating it.
import type { PiniaColadaPlugin } from '@pinia/colada'
interface MyOptions {
foo?: string
}
export function PiniaColadaDebugPlugin(options: MyOptions = {}): PiniaColadaPlugin {
return ({ queryCache, pinia }) => {
queryCache.$onAction(({ name, args }) => {
if (name === 'setQueryData') {
// args type gets narrowed down to the correct type
const [queryKey, data] = args
} else {
// ...
}
})
}
}
Cache Keys
The cache keys are used to identify queries in the cache. The key
passed to queries get serialized deterministically with the toCacheKey()
function. You can use this function in plugins if you needed:
import { toCacheKey } from '@pinia/colada'
const key = toCacheKey(['users', 1, { type: 'friends' }])
TypeScript
As plugins can add properties to the types, you need to augment the types to add the new properties. This is done with module augmentation.
Adding Options to Queries
When adding options to queries, you can augment the UseQueryOptions
and UseQueryOptionsGlobal
interfaces. The first one gives is generic and exposes its type params for more precise type inference, the second one is global and doesn't expose the type params.
/**
* Options for the auto-refetch plugin.
*/
export interface PiniaColadaAutoRefetchOptions {
/**
* Whether to enable auto refresh by default.
* @default false
*/
autoRefetch?: MaybeRefOrGetter<boolean>
}
// Add types for the new option
declare module '@pinia/colada' {
interface UseQueryOptions<TResult, TError, TDataInitial> extends PiniaColadaAutoRefetchOptions {}
interface UseQueryOptionsGlobal extends PiniaColadaAutoRefetchOptions {}
}
To avoid repetition, define the options in a separate interface and augment both UseQueryOptions
and UseQueryOptionsGlobal
in a module augmentation. Note you need to write the three type params, TResult
, TError
, and TDataInitial
, even if you don't use them and they must be named exactly like that.
Examples
Here are some practical examples you can learn from.
Adding a dataUpdatedAt
property to queries
This plugin adds a dataUpdatedAt
property to queries that represents the last time the data was updated. Most of the time this can be achieved at the component level where the query is used with a watcher:
const { data: contact } = useQuery({
// ...
})
const dataUpdatedAt = ref<number>()
watch(
() => contact.value,
() => {
dataUpdatedAt.value = Date.now()
},
)
If you need to use this very often, you might as well create a plugin:
import type { PiniaColadaPlugin } from '@pinia/colada'
import { shallowRef } from 'vue'
import type { ShallowRef } from 'vue'
/**
* Adds a `dataUpdatedAt` property to queries that represents the last time the
* data was updated.
*/
export function PiniaColadaDataUpdatedAtPlugin(): PiniaColadaPlugin {
return ({ queryCache, scope }) => {
queryCache.$onAction(({ name, after, args }) => {
if (name === 'create') {
after((entry) => {
// all effects must be created within the scope
scope.run(() => {
entry.ext.dataUpdatedAt = shallowRef<number>(entry.when)
})
})
} else if (name === 'setEntryState') {
const [entry] = args
after(() => {
entry.ext.dataUpdatedAt.value = entry.when
})
} else {
// ...
}
})
}
}
// Add the property to the types
declare module '@pinia/colada' {
interface UseQueryEntryExtensions<TResult, TError> {
/**
* Time stamp of the last time the data was updated.
*/
dataUpdatedAt: ShallowRef<number>
}
}