e10f54bc12
This PR bumps the internal React playground to use Tailwind CSS v4
134 lines
4.4 KiB
Vue
134 lines
4.4 KiB
Vue
<script>
|
|
import { countries as allCountries } from '../../data'
|
|
import { ref, defineComponent, computed, onMounted, watch } from 'vue'
|
|
import {
|
|
Combobox,
|
|
ComboboxButton,
|
|
ComboboxInput,
|
|
ComboboxLabel,
|
|
ComboboxOption,
|
|
ComboboxOptions,
|
|
} from '@headlessui/vue'
|
|
|
|
export default defineComponent({
|
|
components: {
|
|
Combobox,
|
|
ComboboxButton,
|
|
ComboboxInput,
|
|
ComboboxLabel,
|
|
ComboboxOption,
|
|
ComboboxOptions,
|
|
},
|
|
setup() {
|
|
let query = ref('')
|
|
let activeCountry = ref(allCountries[2]) // allCountries[Math.floor(Math.random() * allCountries.length)]
|
|
let filteredCountries = computed(() => {
|
|
return query.value === ''
|
|
? allCountries
|
|
: allCountries.filter((country) => {
|
|
return country.toLowerCase().includes(query.value.toLowerCase())
|
|
})
|
|
})
|
|
|
|
// Choose a random country on mount
|
|
onMounted(() => {
|
|
activeCountry.value = allCountries[Math.floor(Math.random() * allCountries.length)]
|
|
})
|
|
|
|
watch(activeCountry, () => {
|
|
query.value = ''
|
|
})
|
|
|
|
return {
|
|
query,
|
|
activeCountry,
|
|
filteredCountries,
|
|
}
|
|
},
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div class="flex h-full w-screen justify-center bg-gray-50 p-12">
|
|
<div class="mx-auto w-full max-w-xs">
|
|
<div class="py-8 font-mono text-xs">
|
|
Selected country: {{ activeCountry?.name ?? 'Nothing yet' }}
|
|
</div>
|
|
<div class="space-y-1">
|
|
<Combobox v-model="activeCountry" as="div">
|
|
<ComboboxLabel class="block text-sm font-medium leading-5 text-gray-700">
|
|
Assigned to
|
|
</ComboboxLabel>
|
|
|
|
<div class="relative">
|
|
<span class="shadow-xs relative inline-flex flex-row overflow-hidden rounded-md border">
|
|
<ComboboxInput
|
|
@change="query = $event.target.value"
|
|
class="outline-hidden border-none px-3 py-1"
|
|
/>
|
|
<ComboboxButton
|
|
class="focus:outline-hidden cursor-default border-l bg-gray-100 px-1 text-indigo-600"
|
|
>
|
|
<span class="pointer-events-none flex items-center px-2">
|
|
<svg
|
|
class="h-5 w-5 text-gray-400"
|
|
viewBox="0 0 20 20"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
d="M7 7l3-3 3 3m0 6l-3 3-3-3"
|
|
strokeWidth="1.5"
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
/>
|
|
</svg>
|
|
</span>
|
|
</ComboboxButton>
|
|
</span>
|
|
|
|
<div class="absolute mt-1 w-full rounded-md bg-white shadow-lg">
|
|
<ComboboxOptions
|
|
class="shadow-2xs focus:outline-hidden max-h-60 overflow-auto rounded-md py-1 text-base leading-6 sm:text-sm sm:leading-5"
|
|
>
|
|
<ComboboxOption
|
|
v-for="country in filteredCountries"
|
|
:key="country"
|
|
:value="country"
|
|
v-slot="{ active, selected }"
|
|
>
|
|
<div
|
|
:class="[
|
|
'focus:outline-hidden relative cursor-default select-none py-2 pl-3 pr-9',
|
|
active ? 'bg-indigo-600 text-white' : 'text-gray-900',
|
|
]"
|
|
>
|
|
<span :class="['block truncate', selected ? 'font-semibold' : 'font-normal']">
|
|
{{ country }}
|
|
</span>
|
|
<span
|
|
v-if="selected"
|
|
:class="[
|
|
'absolute inset-y-0 right-0 flex items-center pr-4',
|
|
active ? 'text-white' : 'text-indigo-600',
|
|
]"
|
|
>
|
|
<svg class="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path
|
|
fillRule="evenodd"
|
|
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
|
|
clipRule="evenodd"
|
|
/>
|
|
</svg>
|
|
</span>
|
|
</div>
|
|
</ComboboxOption>
|
|
</ComboboxOptions>
|
|
</div>
|
|
</div>
|
|
</Combobox>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|