This is a Sanity Studio v3 plugin.
A multi-tag input for sanity studio. Fully featured with autocomplete capabilities, live updates, predefined tag options, style and component customizability, and much more.
Sanity v3: npm install sanity-plugin-tags
Sanity v2: sanity install tags
Add it as a plugin in sanity.config.ts
(or .js):
import {defineConfig} from 'sanity'
import {tags} from 'sanity-plugin-tags'
export default defineConfig({
//...
plugins: [tags({})],
})
Simply use 'tag' or 'tags' as a type (single or multi select respectively) in your fields. If you want autocompletion, set the includeFromRelated
option to the name of your field.
That's it! It will even update the autocompletion list live as changes are made to other documents!
Dive into the Options Section for more advanced use cases like predefined tags and the onCreate
hook.
{
name: 'myTags',
title: 'Tags',
type: 'tags',
options: {
includeFromRelated: 'myTags'
...
}
}
{
name: string,
type: "tags" | "tag",
options: {
predefinedTags?: Tag | Tag[] | () => Tag[] | Tag | () => Promise<Tag[] | Tag>
includeFromReference?: false | string
includeFromRelated?: false | string
customLabel?: string
customValue?: string
allowCreate?: boolean
onCreate?: (inputValue: string) => Tag | Promise<Tag>
checkValid?: (inputValue: string, currentValues: string[]) => boolean
reactSelectOptions?: {
[key: string]: any
}
},
//... all other Sanity Properties
},
A tag is simply an object with a label and value. Example:
{
"label": "My Tag",
"value": "my-tag"
}
This can be used for all sorts of things: categorization, single select, and much more. Essentially, if you want to limit people to a single or multi list of strings, tags will fit your use case perfectly.
default: []
This option allows you to add any tags that you would like to the autocomplete list. This can take any form from a single tag to an array of tags, to a function that dynamically returns a tag or tags.
{
// ...
predefinedTags: { label: "My Tag", value: 'my-tag' }
// ...
}
{
// ...
predefinedTags: [
{label: 'My Tag 1', value: 'my-tag-1'},
{label: 'My Tag 2', value: 'my-tag-2'},
]
// ...
}
{
// ...
predefinedTags: async () => client.fetch(...)
// ...
}
default: false
If you already have a sanity schema that contains a tag-like structure and want to add them to the autocomplete list, set this option to the name of your sanity schema document. This option applies no filters. If you would like to filter, use the predefinedTags
option.
{
// ...
includeFromReference: 'category'
// ...
}
default: false
This option is similar to includeFromReference
, but it allows you to add to the autocomplete list from a field in the related document. Typically, you would set this option to the name of the current field to allow autocompletion for tags that were already selected previously.
{
// ...
includeFromRelated: 'category'
// ...
}
default: 'label'
If you want to change the label key for your tags, set this option. Useful when you want to use the default label key to store some other value.
Note: If you set this option, all tags specified by predefinedTags
and the structure returned by onCreate
must use this custom label
{
// ...
customLabel: 'myLabelKey'
// ...
}
default: 'value'
If you want to change the value key for your tags, set this option. Useful when you want to use the default value key to store some other value.
Note: If you set this option, all tags specified by predefinedTags
and the structure returned by onCreate
must use this custom value
{
// ...
customValue: 'myValueKey'
// ...
}
default: true
By default, new tags can be created inline from this input. If you implement the input with a reference, this does not work. See Parts for more information.
{
// ...
allowCreate: false
// ...
}
default: (value) => ({ [customLabel]: value, [customValue]: value})
If you want to edit the label or value of the tag when a new one is created before saving it, use this hook. You do not need to specify this property if you set customLabel
or customValue
and like the default value. If you do specify it, make sure it returns an object that contains the custom label key and the custom value key. This hook provides an easy solution for 'slugifying' the label.
{
// ...
onCreate: (value) => ({
label: value,
value: value.toLowerCase().replace(/\W/g, '-'),
})
// ...
}
default: (inputValue: string, currentValues: string[]) => !currentValues.includes(inputValue) && !!inputValue && inputValue.trim() === inputValue
This allows you to check the validity of a tag when creation is allowed. inputValue
contains the string of the input while currentValues
contains an array of strings that represent all of the values of any options available to select as well as any already-selected options.
{
// ...
checkValid: (input, values) => {
return (
!!input &&
input.trim() === input &&
!values.includes(input.trim().toLowerCase().replace(/\W/g, '-'))
)
}
// ...
}
default: {}
The input component uses React Select under the hood. If you want to change and override any of the options passed to the select component, specify this option. Specify this option at your own risk!
If you want to override React Select's components see Parts for more information.
{
// ...
reactSelectOptions: {
closeMenuOnSelect: false
}
// ...
}
I love feedback, and any help is appreciated! Feel free to install the plugin, submit an issue, or open a PR.
This plugin uses @sanity/plugin-kit with default configuration for build & watch scripts.
See Testing a plugin in Sanity Studio on how to run this plugin with hotreload in the studio.
This plugin is based off of sanity-plugin-autocomplete-tags, though it enhances it by adding a couple additional options while improving support for default sanity values like initialValues
and readOnly
.