UNPKG

@nlabs/gothamjs

Version:
101 lines (100 loc) 14.2 kB
"use client"; 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"; import { jsx, jsxs } from "react/jsx-runtime"; 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( "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, defaultValue, 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, "\xA0" ] }) ] }), /* @__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?.id || option?.label)) } ) ] }) ] }) }); } } ); }; export { SelectField }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../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"],
  "mappings": "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",
  "names": []
}
