268 lines
7.2 KiB
TypeScript
268 lines
7.2 KiB
TypeScript
import { CheckIcon } from '@modrinth/assets'
|
||
import type { Meta, StoryObj } from '@storybook/vue3-vite'
|
||
import { computed, ref } from 'vue'
|
||
|
||
import MultiSelect from '../../components/base/MultiSelect.vue'
|
||
|
||
const meta = {
|
||
title: 'Base/MultiSelect',
|
||
// @ts-ignore - error comes from generically typed component
|
||
component: MultiSelect,
|
||
render: (args) => ({
|
||
components: { MultiSelect },
|
||
setup() {
|
||
const selected = ref(args.modelValue)
|
||
return { args, selected }
|
||
},
|
||
template: /*html*/ `
|
||
<div style="width: 400px;">
|
||
<MultiSelect v-bind="args" v-model="selected" />
|
||
</div>
|
||
`,
|
||
}),
|
||
} satisfies Meta<typeof MultiSelect>
|
||
|
||
export default meta
|
||
type Story = StoryObj<typeof meta>
|
||
|
||
export const Default: Story = {
|
||
args: {
|
||
options: [
|
||
{ value: 'en', label: 'English' },
|
||
{ value: 'es', label: 'Spanish' },
|
||
{ value: 'fr', label: 'French' },
|
||
{ value: 'de', label: 'German' },
|
||
{ value: 'zh-CN', label: 'Chinese (Simplified)' },
|
||
{ value: 'ko', label: 'Korean' },
|
||
{ value: 'ja', label: 'Japanese' },
|
||
{ value: 'pt', label: 'Portuguese' },
|
||
{ value: 'ru', label: 'Russian' },
|
||
{ value: 'it', label: 'Italian' },
|
||
{ value: 'ar', label: 'Arabic' },
|
||
],
|
||
modelValue: ['en', 'es', 'fr', 'zh-CN'],
|
||
placeholder: 'Select languages',
|
||
},
|
||
}
|
||
|
||
export const WithSearch: Story = {
|
||
args: {
|
||
...Default.args,
|
||
searchable: true,
|
||
searchPlaceholder: 'Search versions',
|
||
},
|
||
}
|
||
|
||
export const WithSelectAll: Story = {
|
||
args: {
|
||
...Default.args,
|
||
searchable: true,
|
||
includeSelectAllOption: true,
|
||
searchPlaceholder: 'Search versions',
|
||
},
|
||
}
|
||
|
||
export const WithSelectionActions: Story = {
|
||
args: {
|
||
...Default.args,
|
||
searchable: true,
|
||
showSelectionActions: true,
|
||
searchPlaceholder: 'Search versions',
|
||
},
|
||
}
|
||
|
||
export const WithTopSlot: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: [],
|
||
searchable: true,
|
||
showSelectionActions: true,
|
||
searchPlaceholder: 'Search languages',
|
||
placeholder: 'All languages',
|
||
},
|
||
render: (args) => ({
|
||
components: { CheckIcon, MultiSelect },
|
||
setup() {
|
||
const selected = ref(args.modelValue)
|
||
const isAllLanguagesSelected = computed(() => selected.value.length === 0)
|
||
const selectAllLanguages = () => {
|
||
selected.value = []
|
||
}
|
||
return { args, isAllLanguagesSelected, selectAllLanguages, selected }
|
||
},
|
||
template: /*html*/ `
|
||
<div style="width: 400px;">
|
||
<MultiSelect v-bind="args" v-model="selected">
|
||
<template #top>
|
||
<div class="px-3">
|
||
<button
|
||
type="button"
|
||
class="flex w-full cursor-pointer items-center gap-2.5 rounded-xl border-0 bg-transparent p-3 text-left text-contrast shadow-none transition-colors duration-150 hover:bg-surface-5 focus:bg-surface-5"
|
||
:aria-selected="isAllLanguagesSelected"
|
||
role="option"
|
||
@click="selectAllLanguages"
|
||
@keydown.enter.stop
|
||
@keydown.space.stop
|
||
>
|
||
<span
|
||
class="flex h-5 w-5 shrink-0 items-center justify-center rounded-md border-[1px] border-solid"
|
||
:class="
|
||
isAllLanguagesSelected
|
||
? 'border-button-border bg-brand text-brand-inverted'
|
||
: 'border-surface-5 bg-surface-2'
|
||
"
|
||
>
|
||
<CheckIcon v-if="isAllLanguagesSelected" aria-hidden="true" stroke-width="3" />
|
||
</span>
|
||
<span class="font-semibold leading-tight text-primary">All languages</span>
|
||
</button>
|
||
</div>
|
||
</template>
|
||
</MultiSelect>
|
||
</div>
|
||
`,
|
||
}),
|
||
}
|
||
|
||
export const DropdownMinWidth: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: ['en'],
|
||
dropdownMinWidth: 320,
|
||
placeholder: 'Languages',
|
||
},
|
||
render: (args) => ({
|
||
components: { MultiSelect },
|
||
setup() {
|
||
const selected = ref(args.modelValue)
|
||
return { args, selected }
|
||
},
|
||
template: /*html*/ `
|
||
<div style="width: 11rem;">
|
||
<MultiSelect v-bind="args" v-model="selected" />
|
||
</div>
|
||
`,
|
||
}),
|
||
}
|
||
|
||
export const ManySelected: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: ['en', 'es', 'fr', 'zh-CN', 'ko', 'ja', 'pt', 'ru', 'it', 'ar', 'de'],
|
||
searchable: true,
|
||
includeSelectAllOption: true,
|
||
},
|
||
}
|
||
|
||
export const TwoTagRows: Story = {
|
||
args: {
|
||
...ManySelected.args,
|
||
maxTagRows: 2,
|
||
},
|
||
}
|
||
|
||
export const CustomInputContent: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: ['en', 'es'],
|
||
fitContent: true,
|
||
showChevron: false,
|
||
clearable: false,
|
||
triggerClass:
|
||
'h-10 max-w-[16rem] border border-solid border-surface-5 bg-surface-4 px-3 py-1.5 hover:bg-surface-5 hover:brightness-100 active:brightness-100',
|
||
},
|
||
render: (args) => ({
|
||
components: { MultiSelect },
|
||
setup() {
|
||
const selected = ref(args.modelValue)
|
||
const selectedLabel = () => {
|
||
if (selected.value.length === 0) return 'All languages'
|
||
if (selected.value.length === 1) {
|
||
const option = args.options.find((item) => item.value === selected.value[0])
|
||
return option?.label ?? '1 selected'
|
||
}
|
||
return `${selected.value.length} selected`
|
||
}
|
||
return { args, selected, selectedLabel }
|
||
},
|
||
template: /*html*/ `
|
||
<div style="width: 400px;">
|
||
<MultiSelect v-bind="args" v-model="selected">
|
||
<template #input-content="{ isOpen }">
|
||
<div class="flex min-w-0 items-center gap-2">
|
||
<span class="truncate">
|
||
<span class="font-medium">Languages:</span>
|
||
<span class="ml-1 font-semibold text-contrast">{{ selectedLabel() }}</span>
|
||
</span>
|
||
<span
|
||
class="text-secondary transition-transform duration-150"
|
||
:style="{ transform: isOpen ? 'rotate(180deg)' : 'rotate(0deg)' }"
|
||
>
|
||
⌄
|
||
</span>
|
||
<button
|
||
type="button"
|
||
class="-mr-1 inline-flex size-5 shrink-0 items-center justify-center rounded-full border-0 bg-transparent text-secondary shadow-none transition-colors hover:bg-transparent hover:text-contrast"
|
||
@click.stop="selected = []"
|
||
>
|
||
×
|
||
</button>
|
||
</div>
|
||
</template>
|
||
</MultiSelect>
|
||
</div>
|
||
`,
|
||
}),
|
||
}
|
||
|
||
export const WithBottomSlot: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: ['en', 'es'],
|
||
searchable: true,
|
||
includeSelectAllOption: true,
|
||
},
|
||
render: (args) => ({
|
||
components: { MultiSelect },
|
||
setup() {
|
||
const selected = ref(args.modelValue)
|
||
const minimum = ref('')
|
||
return { args, selected, minimum }
|
||
},
|
||
template: /*html*/ `
|
||
<div style="width: 400px;">
|
||
<MultiSelect v-bind="args" v-model="selected">
|
||
<template #bottom>
|
||
<div style="display: flex; align-items: center; gap: 0.75rem; border-top: 1px solid var(--color-surface-5); padding: 0.75rem;">
|
||
<span style="font-size: 0.875rem; font-weight: 600; color: var(--color-text-primary);">Projects above</span>
|
||
<input
|
||
v-model="minimum"
|
||
type="text"
|
||
style="height: 2rem; width: 5rem; border: 1px solid var(--color-surface-5); border-radius: 0.5rem; background: var(--color-surface-3); color: var(--color-text-primary); text-align: center; font-weight: 600;"
|
||
/>
|
||
<span style="font-size: 0.875rem; font-weight: 600; color: var(--color-text-primary);">downloads</span>
|
||
</div>
|
||
</template>
|
||
</MultiSelect>
|
||
</div>
|
||
`,
|
||
}),
|
||
}
|
||
|
||
export const NoOptions: Story = {
|
||
args: {
|
||
...Default.args,
|
||
options: [],
|
||
modelValue: [],
|
||
searchable: true,
|
||
noOptionsMessage: 'No options available',
|
||
},
|
||
}
|
||
|
||
export const Empty: Story = {
|
||
args: {
|
||
...Default.args,
|
||
modelValue: [],
|
||
},
|
||
}
|