UNPKG

@nlabs/gothamjs

Version:
124 lines (123 loc) 17.4 kB
'use client'; import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; import { Label, Listbox, ListboxButton, ListboxOptions } from '@headlessui/react'; import { cn } from '@nlabs/utils'; import { ChevronsUpDown } from 'lucide-react'; import { useMemo, useState } from 'react'; import { Controller, useFormContext } from 'react-hook-form'; import { useIsMobile } from '../../hooks/useIsMobile.js'; import { getBackgroundClasses, getOutlineClasses, getTextClasses } from '../../utils/colorUtils.js'; import { getInputBorderClass } from '../InputField/InputField.js'; import { Svg } from '../Svg/Svg.js'; import { SelectOption } from './SelectOption.js'; export const SelectField = ({ backgroundColor = 'transparent', borderColor = 'black', borderType = 'solid', className = 'cursor-default grid outline-1 w-full grid-cols-1 rounded-md px-3.5 py-2 text-left sm:text-sm/6', color = 'primary', defaultValue, label, labelClass, labelColor = 'neutral', name, options })=>{ const isMobile = useIsMobile(); const { control, trigger } = useFormContext(); const [selected, setSelected] = useState(options?.find((option)=>option?.value === defaultValue)); // const selectClasses = useMemo(() => cn( // className, // borderType === 'underline' ? 'bg-transparent' : 'bg-white/30 dark:bg-black/30', // getOutlineClasses(color, {hasFocus: true, hasHover: true}) // ), [backgroundColor, className, color]); const selectClasses = useMemo(()=>cn('flex relative w-full', getInputBorderClass(borderType, borderColor, color, 'transparent'), className), [ borderType, borderColor, color, className ]); const labelClasses = useMemo(()=>cn(labelClass, 'block text-sm/6 font-medium', getTextClasses(labelColor)), [ labelClass, labelColor ]); const optionsClasses = useMemo(()=>cn('absolute z-10 max-h-56 w-full overflow-auto rounded-md py-1 text-base focus:outline-hidden data-leave:transition data-leave:duration-100 data-leave:ease-in data-closed:data-leave:opacity-0 sm:text-sm', getBackgroundClasses('white'), getOutlineClasses(color, { hasFocus: true, hasHover: true })), [ backgroundColor, color ]); const chevronClasses = useMemo(()=>cn('col-start-1 row-start-1 size-5 self-center justify-self-end sm:size-4', getTextClasses(color)), [ color ]); const onChange = (value)=>{ setSelected(options?.find((option)=>option?.value === value)); trigger(name); }; return /*#__PURE__*/ _jsx(Controller, { control: control, defaultValue: defaultValue, name: name, render: ({ field })=>{ return isMobile ? /*#__PURE__*/ _jsx("select", { ...field, value: selected?.value, children: options.map((option)=>/*#__PURE__*/ _jsx("option", { value: option.value, children: option.label }, option.id)) }) : /*#__PURE__*/ _jsx("div", { className: "flex flex-col w-full", children: /*#__PURE__*/ _jsxs(Listbox, { value: selected, onChange: (value)=>onChange(value), children: [ /*#__PURE__*/ _jsx(Label, { className: labelClasses, children: label }), /*#__PURE__*/ _jsx("select", { ...field, hidden: true, value: selected?.value }), /*#__PURE__*/ _jsxs("div", { className: cn('flex flex-col relative w-full', { 'mt-2': label }), children: [ /*#__PURE__*/ _jsxs(ListboxButton, { className: selectClasses, children: [ /*#__PURE__*/ _jsxs("span", { className: "col-start-1 row-start-1 flex items-center gap-3 pr-6", children: [ selected?.image && /*#__PURE__*/ _jsx("img", { alt: "", src: selected.image, className: "size-5 shrink-0 rounded-full" }), selected?.icon && /*#__PURE__*/ _jsx(Svg, { className: "size-5 shrink-0 rounded-full", name: selected.icon }), /*#__PURE__*/ _jsxs("span", { className: "block truncate", children: [ selected?.label, " " ] }) ] }), /*#__PURE__*/ _jsx(ChevronsUpDown, { "aria-hidden": "true", className: chevronClasses }) ] }), /*#__PURE__*/ _jsx(ListboxOptions, { transition: true, className: optionsClasses, children: options.map((option)=>option && /*#__PURE__*/ _jsx(SelectOption, { option: option }, option?.id || option?.label)) }) ] }) ] }) }); } }); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/gothamjs/src/components/SelectField/SelectField.tsx"],"sourcesContent":["'use client';\n\nimport {Label, Listbox, ListboxButton, ListboxOptions} from '@headlessui/react';\nimport {cn} from '@nlabs/utils';\nimport {ChevronsUpDown} from 'lucide-react';\nimport {useMemo, useState, type FC} from 'react';\nimport {Controller, useFormContext} from 'react-hook-form';\n\nimport {useIsMobile} from '../../hooks/useIsMobile.js';\nimport {getBackgroundClasses, getOutlineClasses, getTextClasses} from '../../utils/colorUtils.js';\nimport {InputBorderType, getInputBorderClass} from '../InputField/InputField.js';\nimport {Svg} from '../Svg/Svg.js';\nimport {SelectOption, SelectOptionProps} from './SelectOption.js';\n\nimport type {GothamColor} from '../../utils/colorUtils.js';\n\nexport type SelectFieldProps = {\n  readonly backgroundColor?: GothamColor;\n  readonly borderColor?: GothamColor;\n  readonly borderType?: InputBorderType;\n  readonly className?: string;\n  readonly color?: GothamColor\n  readonly defaultValue?: string;\n  readonly label?: string;\n  readonly labelColor?: GothamColor;\n  readonly labelClass?: string;\n  readonly name: string;\n  readonly options: SelectOptionProps['option'][];\n};\n\nexport const SelectField: FC<SelectFieldProps> = ({\n  backgroundColor = 'transparent',\n  borderColor = 'black',\n  borderType = 'solid',\n  className = 'cursor-default grid outline-1 w-full grid-cols-1 rounded-md px-3.5 py-2 text-left sm:text-sm/6',\n  color = 'primary',\n  defaultValue,\n  label,\n  labelClass,\n  labelColor = 'neutral',\n  name,\n  options\n}) => {\n  const isMobile = useIsMobile();\n  const {control, trigger} = useFormContext();\n  const [selected, setSelected] = useState<SelectOptionProps['option']>(options?.find((option) => option?.value === defaultValue) as SelectOptionProps['option']);\n  // const selectClasses = useMemo(() => cn(\n  //   className,\n  //   borderType === 'underline' ? 'bg-transparent' : 'bg-white/30 dark:bg-black/30',\n  //   getOutlineClasses(color, {hasFocus: true, hasHover: true})\n  // ), [backgroundColor, className, color]);\n  const selectClasses = useMemo(() => cn(\n    'flex relative w-full',\n    getInputBorderClass(borderType, borderColor, color, 'transparent'), className), [borderType, borderColor, color, className]\n  );\n  const labelClasses = useMemo(() => cn(\n    labelClass,\n    'block text-sm/6 font-medium',\n    getTextClasses(labelColor)\n  ), [labelClass, labelColor]);\n  const optionsClasses = useMemo(() => cn(\n    'absolute z-10 max-h-56 w-full overflow-auto rounded-md py-1 text-base focus:outline-hidden data-leave:transition data-leave:duration-100 data-leave:ease-in data-closed:data-leave:opacity-0 sm:text-sm',\n    getBackgroundClasses('white'),\n    getOutlineClasses(color, {hasFocus: true, hasHover: true})\n  ), [backgroundColor, color]);\n  const chevronClasses = useMemo(() => cn(\n    'col-start-1 row-start-1 size-5 self-center justify-self-end sm:size-4',\n    getTextClasses(color)\n  ), [color]);\n  const onChange = (value: string) => {\n    setSelected(options?.find((option) => option?.value === value) as SelectOptionProps['option']);\n    trigger(name);\n  };\n\n  return (\n    <Controller\n      control={control}\n      defaultValue={defaultValue}\n      name={name}\n      render={({field}) => {\n        return isMobile ? (\n          <select {...field} value={selected?.value}>\n            {options.map((option) => (\n              <option key={option.id} value={option.value}>{option.label}</option>\n            ))}\n          </select>\n        ) : (\n          <div className=\"flex flex-col w-full\">\n            <Listbox value={selected} onChange={(value) => onChange(value as unknown as string)}>\n              <Label className={labelClasses}>\n                {label}\n              </Label>\n              <select {...field} hidden value={selected?.value} />\n              <div className={cn('flex flex-col relative w-full', {'mt-2': label})}>\n                <ListboxButton className={selectClasses}>\n                  <span className=\"col-start-1 row-start-1 flex items-center gap-3 pr-6\">\n                    {selected?.image && <img alt=\"\" src={selected.image} className=\"size-5 shrink-0 rounded-full\" />}\n                    {selected?.icon && <Svg className=\"size-5 shrink-0 rounded-full\" name={selected.icon} />}\n                    <span className=\"block truncate\">{selected?.label}&nbsp;</span>\n                  </span>\n                  <ChevronsUpDown\n                    aria-hidden=\"true\"\n                    className={chevronClasses}\n                  />\n                </ListboxButton>\n\n                <ListboxOptions\n                  transition\n                  className={optionsClasses}\n                >\n                  {options.map((option) => option && (\n                    <SelectOption key={option?.id || option?.label} option={option} />\n                  ))}\n                </ListboxOptions>\n              </div>\n            </Listbox>\n          </div>\n        );\n      }}\n    />\n  );\n};\n"],"names":["Label","Listbox","ListboxButton","ListboxOptions","cn","ChevronsUpDown","useMemo","useState","Controller","useFormContext","useIsMobile","getBackgroundClasses","getOutlineClasses","getTextClasses","getInputBorderClass","Svg","SelectOption","SelectField","backgroundColor","borderColor","borderType","className","color","defaultValue","label","labelClass","labelColor","name","options","isMobile","control","trigger","selected","setSelected","find","option","value","selectClasses","labelClasses","optionsClasses","hasFocus","hasHover","chevronClasses","onChange","render","field","select","map","id","div","hidden","span","image","img","alt","src","icon","aria-hidden","transition"],"mappings":"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"}