Created
October 9, 2024 07:42
-
-
Save Jamiewarb/fd21ac4098e738fde981afc7aa8a2d7f to your computer and use it in GitHub Desktop.
Sanity Character Count component for string and text
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
.textInput { | |
border: 1.5px solid #e1e8fa; | |
border-radius: 3px; | |
color: #6f7fb3; | |
font-size: 12px; | |
font-weight: bold; | |
padding: 1px 3px; | |
@media (prefers-color-scheme: dark) { | |
border-color: #575f73; | |
} | |
} | |
.textInputWarning { | |
border-color: #fbf4ce; | |
color: #8b8775; | |
@media (prefers-color-scheme: dark) { | |
border-color: #575134; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import styles from './CharacterCount.module.css'; | |
import type { StringCountInputProps, TextCountInputProps } from '../types'; | |
function getMinMax( | |
schemaType: StringCountInputProps['schemaType'] | TextCountInputProps['schemaType'], | |
) { | |
return { | |
min: schemaType.options?.minLength, | |
max: schemaType.options?.maxLength, | |
}; | |
} | |
export default function CharacterCount( | |
props: StringCountInputProps | TextCountInputProps, | |
) { | |
const { value = '', schemaType } = props; | |
const { min, max } = getMinMax(schemaType); | |
let characters: string | undefined = undefined; | |
const classes = [styles.textInput]; | |
if (min ?? max) { | |
characters = `${value.length}`; | |
if (max) { | |
characters = `${characters} / ${max}`; | |
} | |
} | |
if ((min && value.length < min) ?? (max && value.length > max)) { | |
classes.push(styles.textInputWarning); | |
} | |
return <span className={classes.join(' ')}>{characters ?? ''}</span>; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { type ChangeEvent, useCallback } from 'react'; | |
import { Stack, Text, TextInput } from '@sanity/ui'; | |
import { set, unset } from 'sanity'; | |
import CharacterCount from './CharacterCount'; | |
import type { StringCountInputProps } from '../types'; | |
export const StringInput = (props: StringCountInputProps) => { | |
const { elementProps, onChange, value = '' } = props; | |
const handleChange = useCallback( | |
(event: ChangeEvent<HTMLInputElement>) => { | |
const nextValue = event.currentTarget.value; | |
onChange(nextValue ? set(nextValue) : unset()); | |
}, | |
[onChange], | |
); | |
return ( | |
<Stack space={2}> | |
<TextInput {...elementProps} onChange={handleChange} value={value} /> | |
<Text style={{ marginLeft: 'auto' }}> | |
<CharacterCount {...props} /> | |
</Text> | |
</Stack> | |
); | |
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { type ChangeEvent, useCallback } from 'react'; | |
import { Stack, Text, TextArea } from '@sanity/ui'; | |
import { set, unset } from 'sanity'; | |
import CharacterCount from './CharacterCount'; | |
import type { TextCountInputProps } from '../types'; | |
export const TextInput = (props: TextCountInputProps) => { | |
const { elementProps, onChange, value = '' } = props; | |
const handleChange = useCallback( | |
(event: ChangeEvent<HTMLTextAreaElement>) => { | |
const nextValue = event.currentTarget.value; | |
onChange(nextValue ? set(nextValue) : unset()); | |
}, | |
[onChange], | |
); | |
return ( | |
<Stack space={2}> | |
<TextArea | |
{...elementProps} | |
onChange={handleChange} | |
value={value} | |
style={{ resize: 'vertical' }} | |
/> | |
<Text style={{ marginLeft: 'auto' }}> | |
<CharacterCount {...props} /> | |
</Text> | |
</Stack> | |
); | |
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import type { | |
StringDefinition, | |
TextDefinition, | |
StringInputProps, | |
StringSchemaType, | |
TextInputProps, | |
} from 'sanity'; | |
/** | |
* You can also pass any of the properties of Sanity object types described here: https://www.sanity.io/docs/string-type | |
*/ | |
export interface StringCountSchemaDefinition extends StringDefinition { | |
options?: StringSchemaType['options'] & CharCountOptions; | |
} | |
/** | |
* You can also pass any of the properties of Sanity object types described here: https://www.sanity.io/docs/text-type | |
*/ | |
export interface TextCountSchemaDefinition extends TextDefinition { | |
options?: TextDefinition['options'] & CharCountOptions; | |
} | |
export interface CharCountOptions { | |
minLength?: number; | |
maxLength?: number; | |
} | |
export interface StringCountInputProps extends StringInputProps { | |
schemaType: StringInputProps['schemaType'] & { | |
options?: StringInputProps['schemaType']['options'] & CharCountOptions; | |
}; | |
} | |
export interface TextCountInputProps extends TextInputProps { | |
schemaType: TextInputProps['schemaType'] & { | |
options?: TextInputProps['schemaType']['options'] & CharCountOptions; | |
}; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { StringInput } from '../components/StringInput'; | |
import { TextInput } from '../components/TextInput'; | |
import { StringCountSchemaDefinition, TextCountSchemaDefinition } from '../types'; | |
export function defineStringCountField( | |
schemaTypeDefinition: Omit<StringCountSchemaDefinition, 'type'>, | |
): StringCountSchemaDefinition { | |
return { | |
...schemaTypeDefinition, | |
type: 'string', | |
components: { input: StringInput }, | |
}; | |
} | |
export function defineTextCountField( | |
schemaTypeDefinition: Omit<TextCountSchemaDefinition, 'type'>, | |
): TextCountSchemaDefinition { | |
return { | |
...schemaTypeDefinition, | |
type: 'text', | |
components: { input: TextInput }, | |
}; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Can be used like