2023-02-16 22:39:57 +00:00
|
|
|
import type { ReactElement, ReactNode, Ref } from "react";
|
|
|
|
import React, { forwardRef, useCallback, useId, useState } from "react";
|
|
|
|
import type { FieldValues, SubmitHandler, UseFormReturn } from "react-hook-form";
|
|
|
|
import { FormProvider, useFormContext } from "react-hook-form";
|
2022-07-23 00:39:50 +00:00
|
|
|
|
|
|
|
import classNames from "@calcom/lib/classNames";
|
|
|
|
import { getErrorFromUnknown } from "@calcom/lib/errors";
|
|
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
|
|
|
|
2023-01-23 23:08:01 +00:00
|
|
|
import { Alert, showToast, Skeleton, Tooltip, UnstyledSelect } from "../../..";
|
2023-01-26 22:51:03 +00:00
|
|
|
import { FiEye, FiEyeOff, FiX } from "../../icon";
|
2022-11-04 15:40:46 +00:00
|
|
|
import { HintsOrErrors } from "./HintOrErrors";
|
|
|
|
import { Label } from "./Label";
|
2022-07-23 00:39:50 +00:00
|
|
|
|
2022-11-28 11:20:49 +00:00
|
|
|
type InputProps = JSX.IntrinsicElements["input"] & { isFullWidth?: boolean };
|
2022-07-23 00:39:50 +00:00
|
|
|
|
2022-11-28 11:20:49 +00:00
|
|
|
export const Input = forwardRef<HTMLInputElement, InputProps>(function Input(
|
|
|
|
{ isFullWidth = true, ...props },
|
|
|
|
ref
|
|
|
|
) {
|
2022-07-23 00:39:50 +00:00
|
|
|
return (
|
|
|
|
<input
|
|
|
|
{...props}
|
|
|
|
ref={ref}
|
|
|
|
className={classNames(
|
2022-11-28 11:20:49 +00:00
|
|
|
"mb-2 block h-9 rounded-md border border-gray-300 py-2 px-3 text-sm placeholder:text-gray-400 hover:border-gray-400 focus:border-neutral-300 focus:outline-none focus:ring-2 focus:ring-neutral-800 focus:ring-offset-1",
|
|
|
|
isFullWidth && "w-full",
|
2022-07-23 00:39:50 +00:00
|
|
|
props.className
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
export function InputLeading(props: JSX.IntrinsicElements["div"]) {
|
|
|
|
return (
|
2023-01-04 07:38:45 +00:00
|
|
|
<span className="inline-flex flex-shrink-0 items-center rounded-l-sm border border-gray-300 bg-gray-50 px-3 text-gray-500 ltr:border-r-0 rtl:border-l-0 sm:text-sm">
|
2022-07-23 00:39:50 +00:00
|
|
|
{props.children}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
type InputFieldProps = {
|
|
|
|
label?: ReactNode;
|
|
|
|
hint?: ReactNode;
|
2022-07-29 13:23:54 +00:00
|
|
|
hintErrors?: string[];
|
2022-07-23 00:39:50 +00:00
|
|
|
addOnLeading?: ReactNode;
|
|
|
|
addOnSuffix?: ReactNode;
|
2022-11-28 11:20:49 +00:00
|
|
|
inputIsFullWidth?: boolean;
|
2022-07-23 00:39:50 +00:00
|
|
|
addOnFilled?: boolean;
|
2022-11-30 20:51:44 +00:00
|
|
|
addOnClassname?: string;
|
2022-07-23 00:39:50 +00:00
|
|
|
error?: string;
|
|
|
|
labelSrOnly?: boolean;
|
|
|
|
containerClassName?: string;
|
2022-08-03 16:59:40 +00:00
|
|
|
t?: (key: string) => string;
|
2022-07-23 00:39:50 +00:00
|
|
|
} & React.ComponentProps<typeof Input> & {
|
|
|
|
labelProps?: React.ComponentProps<typeof Label>;
|
2022-09-12 19:07:52 +00:00
|
|
|
labelClassName?: string;
|
2022-07-23 00:39:50 +00:00
|
|
|
};
|
|
|
|
|
2022-10-05 14:22:06 +00:00
|
|
|
type AddonProps = {
|
|
|
|
children: React.ReactNode;
|
|
|
|
isFilled?: boolean;
|
|
|
|
className?: string;
|
|
|
|
error?: boolean;
|
|
|
|
};
|
|
|
|
|
2022-11-30 20:51:44 +00:00
|
|
|
const Addon = ({ isFilled, children, className, error }: AddonProps) => (
|
2022-10-05 14:22:06 +00:00
|
|
|
<div
|
2022-11-30 20:51:44 +00:00
|
|
|
className={classNames(
|
|
|
|
"addon-wrapper h-9 border border-gray-300 px-3",
|
|
|
|
isFilled && "bg-gray-100",
|
|
|
|
className
|
|
|
|
)}>
|
2022-11-21 19:20:33 +00:00
|
|
|
<div className={classNames("flex h-full flex-col justify-center text-sm", error && "text-red-900")}>
|
2022-10-05 14:22:06 +00:00
|
|
|
<span className="whitespace-nowrap py-2.5">{children}</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
2022-11-04 15:40:46 +00:00
|
|
|
export const InputField = forwardRef<HTMLInputElement, InputFieldProps>(function InputField(props, ref) {
|
2022-07-23 00:39:50 +00:00
|
|
|
const id = useId();
|
2022-09-29 16:19:03 +00:00
|
|
|
const { t: _t, isLocaleReady, i18n } = useLocale();
|
2022-08-03 16:59:40 +00:00
|
|
|
const t = props.t || _t;
|
2022-09-02 19:00:41 +00:00
|
|
|
const name = props.name || "";
|
2022-07-23 00:39:50 +00:00
|
|
|
const {
|
2022-09-02 19:00:41 +00:00
|
|
|
label = t(name),
|
2022-07-23 00:39:50 +00:00
|
|
|
labelProps,
|
2022-09-12 19:07:52 +00:00
|
|
|
labelClassName,
|
2022-09-29 16:19:03 +00:00
|
|
|
placeholder = isLocaleReady && i18n.exists(name + "_placeholder") ? t(name + "_placeholder") : "",
|
2022-07-23 00:39:50 +00:00
|
|
|
className,
|
|
|
|
addOnLeading,
|
|
|
|
addOnSuffix,
|
|
|
|
addOnFilled = true,
|
2022-11-30 20:51:44 +00:00
|
|
|
addOnClassname,
|
2022-11-28 11:20:49 +00:00
|
|
|
inputIsFullWidth,
|
2022-07-23 00:39:50 +00:00
|
|
|
hint,
|
2022-11-30 20:51:44 +00:00
|
|
|
type,
|
2022-07-29 13:23:54 +00:00
|
|
|
hintErrors,
|
2022-07-23 00:39:50 +00:00
|
|
|
labelSrOnly,
|
|
|
|
containerClassName,
|
2022-12-07 21:47:02 +00:00
|
|
|
readOnly,
|
2022-08-03 16:59:40 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
|
|
t: __t,
|
2022-07-23 00:39:50 +00:00
|
|
|
...passThrough
|
|
|
|
} = props;
|
|
|
|
|
2022-11-30 20:51:44 +00:00
|
|
|
const [inputValue, setInputValue] = useState<string>("");
|
|
|
|
|
2022-07-23 00:39:50 +00:00
|
|
|
return (
|
|
|
|
<div className={classNames(containerClassName)}>
|
2022-09-02 19:00:41 +00:00
|
|
|
{!!name && (
|
2022-09-21 16:01:47 +00:00
|
|
|
<Skeleton
|
|
|
|
as={Label}
|
2022-07-23 00:39:50 +00:00
|
|
|
htmlFor={id}
|
2022-09-21 16:01:47 +00:00
|
|
|
loadingClassName="w-16"
|
2022-07-23 00:39:50 +00:00
|
|
|
{...labelProps}
|
2022-09-12 19:07:52 +00:00
|
|
|
className={classNames(labelClassName, labelSrOnly && "sr-only", props.error && "text-red-900")}>
|
2022-07-23 00:39:50 +00:00
|
|
|
{label}
|
2022-09-21 16:01:47 +00:00
|
|
|
</Skeleton>
|
2022-07-23 00:39:50 +00:00
|
|
|
)}
|
|
|
|
{addOnLeading || addOnSuffix ? (
|
Admin Wizard Choose License (#6574)
* Implementation
* i18n
* More i18n
* extracted i18n, needs api to get most recent price, added hint: update later
* Fixing i18n var
* Fix booking filters not working for admin (#6576)
* fix: react-select overflow issue in some modals. (#6587)
* feat: add a disable overflow prop
* feat: use the disable overflow prop
* Tailwind Merge (#6596)
* Tailwind Merge
* Fix merge classNames
* [CAL-808] /availability/single - UI issue on buttons beside time inputs (#6561)
* [CAL-808] /availability/single - UI issue on buttons beside time inputs
* Update apps/web/public/static/locales/en/common.json
* Update packages/features/schedules/components/Schedule.tsx
* create new translation for tooltip
Co-authored-by: gitstart-calcom <gitstart@users.noreply.github.com>
Co-authored-by: Peer Richelsen <peeroke@gmail.com>
Co-authored-by: CarinaWolli <wollencarina@gmail.com>
* Bye bye submodules (#6585)
* WIP
* Uses ssh instead
* Update .gitignore
* Update .gitignore
* Update Makefile
* Update git-setup.sh
* Update git-setup.sh
* Replaced Makefile with bash script
* Update package.json
* fix: show button on empty string (#6601)
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
* fix: add delete in dropdown (#6599)
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
* Update README.md
* Update README.md
* Changed a neutral- classes to gray (#6603)
* Changed a neutral- classes to gray
* Changed all border-1 to border
* Update package.json
* Test fixes
* Yarn lock fixes
* Fix string equality check in git-setup.sh
* [CAL-811] Avatar icon not redirecting user back to the main page (#6586)
* Remove cursor-pointer, remove old Avatar* files
* Fixed styling for checkedSelect + some cleanup
Co-authored-by: gitstart-calcom <gitstart@users.noreply.github.com>
Co-authored-by: Alex van Andel <me@alexvanandel.com>
* Harsh/add member invite (#6598)
Co-authored-by: Guest <guest@pop-os.localdomain>
Co-authored-by: root <harsh.singh@gocomet.com>
* Regenerated lockfile without upgrade (#6610)
* fix: remove annoying outline when <Button /> clicked (#6537)
* fix: remove annoying outline when <Button /> clicked
* Delete yarn.lock
* remove 1 on 1 icon (#6609)
* removed 1-on-1 badge
* changed user to users for group events
* fix: case-sensitivity in apps path (#6552)
* fix: lowercase slug
* fix: make fallback blocking
* Fix FAB (#6611)
* feat: add LocationSelect component (#6571)
* feat: add LocationSelect component
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
* fix: type error
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
* chore: type error
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
* Update booking filters design (#6543)
* Update booking filters
* Add filter on YOUR bookings
* Fix pending members showing up in list
* Reduce the avatar size to 'sm' for now
* Bugfix/dropdown menu trigger as child remove class names (#6614)
* Fix UsernameTextfield to take right height
* Remove className side-effect
* Incorrect resolution version fixed
* Converted mobile DropdownMenuTrigger styles into Button
* v2.5.3
* fix: use items-center (#6618)
* fix tooltip and modal stacking issues (#6491)
* fix tooltip and modal stacking issues
* use z-index in larger screens and less
Co-authored-by: Alex van Andel <me@alexvanandel.com>
* Temporary fix (#6626)
* Fix Ga4 tracking (#6630)
* generic <UpgradeScreen> component (#6594)
* first attempt of <UpgradeScreen>
* changes to icons
* reverted changes back to initial state, needs fix: teams not showing
* WIP
* Fix weird reactnode error
* Fix loading text
* added upgradeTip to routing forms
* icon colors
* create and use hook to check if user has team plan
* use useTeamPlan for upgradeTeamsBadge
* replace huge svg with compressed jpeg
* responsive fixes
* Update packages/ui/components/badge/UpgradeTeamsBadge.tsx
Co-authored-by: sean-brydon <55134778+sean-brydon@users.noreply.github.com>
* Give team plan features to E2E tests
* Allow option to make a user part of team int ests
* Remove flash of paywall for team user
* Add team user for typeform tests as well
Co-authored-by: Peer Richelsen <peer@cal.com>
Co-authored-by: CarinaWolli <wollencarina@gmail.com>
Co-authored-by: Carina Wollendorfer <30310907+CarinaWolli@users.noreply.github.com>
Co-authored-by: Alex van Andel <me@alexvanandel.com>
Co-authored-by: Hariom Balhara <hariombalhara@gmail.com>
* Removing env var to rely on db
* Restoring i18n keys, set loading moved
* Fixing tailwind-preset glob
* Wizard width fix for md+ screens
* Converting licenses options to radix radio
* Applying feedback + other tweaks
* Reverting this, not this PR related
* Unneeded code removal
* Reverting unneeded style change
* Applying feedback
* Removing licenseType
* Upgrades typescript
* Update yarn lock
* Typings
* Hotfix: ping,riverside,whereby and around not showing up in list (#6712)
* Hotfix: ping,riverside,whereby and around not showing up in list (#6712) (#6713)
* Adds deployment settings to DB (#6706)
* WIP
* Adds DeploymentTheme
* Add missing migrations
* Adds client extensions for deployment
* Cleanup
* Delete migration.sql
* Relying on both, env var and new model
* Restoring env example doc for backward compat
* Maximum call stack size exceeded fix?
* Revert upgrade
* Update index.ts
* Delete index.ts
* Not exposing license key, fixed radio behavior
* Covering undefined env var
* Self contained checkLicense
* Feedback
* Moar feedback
* Feedback
* Feedback
* Feedback
* Cleanup
---------
Signed-off-by: Udit Takkar <udit.07814802719@cse.mait.ac.in>
Co-authored-by: Peer Richelsen <peer@cal.com>
Co-authored-by: sean-brydon <55134778+sean-brydon@users.noreply.github.com>
Co-authored-by: Nafees Nazik <84864519+G3root@users.noreply.github.com>
Co-authored-by: GitStart-Cal.com <121884634+gitstart-calcom@users.noreply.github.com>
Co-authored-by: gitstart-calcom <gitstart@users.noreply.github.com>
Co-authored-by: Peer Richelsen <peeroke@gmail.com>
Co-authored-by: CarinaWolli <wollencarina@gmail.com>
Co-authored-by: Omar López <zomars@me.com>
Co-authored-by: Udit Takkar <53316345+Udit-takkar@users.noreply.github.com>
Co-authored-by: Alex van Andel <me@alexvanandel.com>
Co-authored-by: Harsh Singh <51085015+harshsinghatz@users.noreply.github.com>
Co-authored-by: Guest <guest@pop-os.localdomain>
Co-authored-by: root <harsh.singh@gocomet.com>
Co-authored-by: Luis Cadillo <luiscaf3r@gmail.com>
Co-authored-by: Mohammed Cherfaoui <hi@cherfaoui.dev>
Co-authored-by: Hariom Balhara <hariombalhara@gmail.com>
Co-authored-by: Carina Wollendorfer <30310907+CarinaWolli@users.noreply.github.com>
2023-02-08 00:23:42 +00:00
|
|
|
<div className="group relative mb-1 flex items-center rounded-md focus-within:outline-none focus-within:ring-2 focus-within:ring-neutral-800 focus-within:ring-offset-1">
|
2022-10-05 14:22:06 +00:00
|
|
|
{addOnLeading && (
|
2023-01-04 07:38:45 +00:00
|
|
|
<Addon
|
|
|
|
isFilled={addOnFilled}
|
|
|
|
className={classNames(
|
|
|
|
"ltr:rounded-l-md ltr:border-r-0 rtl:rounded-r-md rtl:border-l-0",
|
|
|
|
addOnClassname
|
|
|
|
)}>
|
2022-10-05 14:22:06 +00:00
|
|
|
{addOnLeading}
|
|
|
|
</Addon>
|
|
|
|
)}
|
2022-07-23 00:39:50 +00:00
|
|
|
<Input
|
|
|
|
id={id}
|
2022-12-05 13:28:21 +00:00
|
|
|
type={type}
|
2022-09-29 16:19:03 +00:00
|
|
|
placeholder={placeholder}
|
2022-11-28 11:20:49 +00:00
|
|
|
isFullWidth={inputIsFullWidth}
|
2022-07-23 00:39:50 +00:00
|
|
|
className={classNames(
|
|
|
|
className,
|
2023-01-04 07:38:45 +00:00
|
|
|
addOnLeading && "ltr:rounded-l-none rtl:rounded-r-none",
|
|
|
|
addOnSuffix && "ltr:rounded-r-none rtl:rounded-l-none",
|
2022-11-30 20:51:44 +00:00
|
|
|
type === "search" && "pr-8",
|
2022-07-23 00:39:50 +00:00
|
|
|
"!my-0 !ring-0"
|
|
|
|
)}
|
2022-12-07 21:47:02 +00:00
|
|
|
{...passThrough}
|
2022-12-09 17:10:54 +00:00
|
|
|
{...(type == "search" && {
|
|
|
|
onChange: (e) => {
|
|
|
|
setInputValue(e.target.value);
|
|
|
|
props.onChange && props.onChange(e);
|
|
|
|
},
|
|
|
|
value: inputValue,
|
|
|
|
})}
|
2022-12-07 21:47:02 +00:00
|
|
|
readOnly={readOnly}
|
2022-07-23 00:39:50 +00:00
|
|
|
ref={ref}
|
|
|
|
/>
|
2022-10-05 14:22:06 +00:00
|
|
|
{addOnSuffix && (
|
2023-01-04 07:38:45 +00:00
|
|
|
<Addon
|
|
|
|
isFilled={addOnFilled}
|
|
|
|
className={classNames(
|
|
|
|
"ltr:rounded-r-md ltr:border-l-0 rtl:rounded-l-md rtl:border-r-0",
|
|
|
|
addOnClassname
|
|
|
|
)}>
|
2022-10-05 14:22:06 +00:00
|
|
|
{addOnSuffix}
|
|
|
|
</Addon>
|
|
|
|
)}
|
2022-11-30 20:51:44 +00:00
|
|
|
{type === "search" && inputValue?.toString().length > 0 && (
|
2023-01-23 23:08:01 +00:00
|
|
|
<FiX
|
2023-01-04 07:38:45 +00:00
|
|
|
className="absolute top-2.5 h-4 w-4 cursor-pointer text-gray-500 ltr:right-2 rtl:left-2"
|
2022-11-30 20:51:44 +00:00
|
|
|
onClick={(e) => {
|
|
|
|
setInputValue("");
|
|
|
|
props.onChange && props.onChange(e as unknown as React.ChangeEvent<HTMLInputElement>);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
)}
|
2022-07-23 00:39:50 +00:00
|
|
|
</div>
|
|
|
|
) : (
|
2022-11-28 11:20:49 +00:00
|
|
|
<Input
|
|
|
|
id={id}
|
2022-12-05 13:28:21 +00:00
|
|
|
type={type}
|
2022-11-28 11:20:49 +00:00
|
|
|
placeholder={placeholder}
|
|
|
|
className={className}
|
|
|
|
{...passThrough}
|
2023-03-02 18:15:28 +00:00
|
|
|
readOnly={readOnly}
|
2022-11-28 11:20:49 +00:00
|
|
|
ref={ref}
|
|
|
|
isFullWidth={inputIsFullWidth}
|
|
|
|
/>
|
2022-07-23 00:39:50 +00:00
|
|
|
)}
|
2022-09-02 19:00:41 +00:00
|
|
|
<HintsOrErrors hintErrors={hintErrors} fieldName={name} t={t} />
|
2022-07-29 13:23:54 +00:00
|
|
|
{hint && <div className="text-gray mt-2 flex items-center text-sm text-gray-700">{hint}</div>}
|
2022-07-23 00:39:50 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
export const TextField = forwardRef<HTMLInputElement, InputFieldProps>(function TextField(props, ref) {
|
|
|
|
return <InputField ref={ref} {...props} />;
|
|
|
|
});
|
|
|
|
|
|
|
|
export const PasswordField = forwardRef<HTMLInputElement, InputFieldProps>(function PasswordField(
|
|
|
|
props,
|
|
|
|
ref
|
|
|
|
) {
|
2022-10-03 19:52:57 +00:00
|
|
|
const { t } = useLocale();
|
2022-09-13 16:35:14 +00:00
|
|
|
const [isPasswordVisible, setIsPasswordVisible] = useState(false);
|
|
|
|
const toggleIsPasswordVisible = useCallback(
|
|
|
|
() => setIsPasswordVisible(!isPasswordVisible),
|
|
|
|
[isPasswordVisible, setIsPasswordVisible]
|
|
|
|
);
|
|
|
|
const textLabel = isPasswordVisible ? t("hide_password") : t("show_password");
|
2022-10-03 19:52:57 +00:00
|
|
|
|
2022-09-13 16:35:14 +00:00
|
|
|
return (
|
2022-10-03 19:52:57 +00:00
|
|
|
<div className="relative [&_.group:hover_.addon-wrapper]:border-gray-400 [&_.group:focus-within_.addon-wrapper]:border-neutral-300">
|
2022-09-13 16:35:14 +00:00
|
|
|
<InputField
|
2022-10-03 19:52:57 +00:00
|
|
|
type={isPasswordVisible ? "text" : "password"}
|
2022-10-04 09:20:21 +00:00
|
|
|
placeholder={props.placeholder || "•••••••••••••"}
|
2022-09-13 16:35:14 +00:00
|
|
|
ref={ref}
|
|
|
|
{...props}
|
2023-01-04 07:38:45 +00:00
|
|
|
className={classNames("mb-0 ltr:border-r-0 ltr:pr-10 rtl:border-l-0 rtl:pl-10", props.className)}
|
2022-10-03 19:52:57 +00:00
|
|
|
addOnFilled={false}
|
|
|
|
addOnSuffix={
|
|
|
|
<Tooltip content={textLabel}>
|
|
|
|
<button
|
2023-01-04 07:38:45 +00:00
|
|
|
className="absolute bottom-0 h-9 text-gray-900 ltr:right-3 rtl:left-3"
|
2022-10-03 19:52:57 +00:00
|
|
|
type="button"
|
|
|
|
onClick={() => toggleIsPasswordVisible()}>
|
|
|
|
{isPasswordVisible ? (
|
2023-01-23 23:08:01 +00:00
|
|
|
<FiEyeOff className="h-4 stroke-[2.5px]" />
|
2022-10-03 19:52:57 +00:00
|
|
|
) : (
|
2023-01-23 23:08:01 +00:00
|
|
|
<FiEye className="h-4 stroke-[2.5px]" />
|
2022-10-03 19:52:57 +00:00
|
|
|
)}
|
|
|
|
<span className="sr-only">{textLabel}</span>
|
|
|
|
</button>
|
|
|
|
</Tooltip>
|
|
|
|
}
|
2022-09-13 16:35:14 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
2022-07-23 00:39:50 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
export const EmailInput = forwardRef<HTMLInputElement, InputFieldProps>(function EmailInput(props, ref) {
|
|
|
|
return (
|
|
|
|
<Input
|
|
|
|
ref={ref}
|
|
|
|
type="email"
|
|
|
|
autoCapitalize="none"
|
|
|
|
autoComplete="email"
|
|
|
|
autoCorrect="off"
|
|
|
|
inputMode="email"
|
|
|
|
{...props}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
export const EmailField = forwardRef<HTMLInputElement, InputFieldProps>(function EmailField(props, ref) {
|
|
|
|
return (
|
|
|
|
<InputField
|
|
|
|
ref={ref}
|
|
|
|
type="email"
|
|
|
|
autoCapitalize="none"
|
|
|
|
autoComplete="email"
|
|
|
|
autoCorrect="off"
|
|
|
|
inputMode="email"
|
|
|
|
{...props}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2022-09-02 19:00:41 +00:00
|
|
|
type TextAreaProps = JSX.IntrinsicElements["textarea"];
|
2022-07-23 00:39:50 +00:00
|
|
|
|
|
|
|
export const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(function TextAreaInput(props, ref) {
|
|
|
|
return (
|
|
|
|
<textarea
|
|
|
|
ref={ref}
|
|
|
|
{...props}
|
|
|
|
className={classNames(
|
2022-11-26 12:50:55 +00:00
|
|
|
"block w-full rounded-md border border-gray-300 py-2 px-3 text-sm hover:border-gray-400 focus:border-neutral-300 focus:outline-none focus:ring-2 focus:ring-neutral-800 focus:ring-offset-1",
|
2022-07-23 00:39:50 +00:00
|
|
|
props.className
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
type TextAreaFieldProps = {
|
|
|
|
label?: ReactNode;
|
2022-08-03 16:59:40 +00:00
|
|
|
t?: (key: string) => string;
|
2022-07-23 00:39:50 +00:00
|
|
|
} & React.ComponentProps<typeof TextArea> & {
|
2022-09-02 19:00:41 +00:00
|
|
|
name: string;
|
2022-07-23 00:39:50 +00:00
|
|
|
labelProps?: React.ComponentProps<typeof Label>;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const TextAreaField = forwardRef<HTMLTextAreaElement, TextAreaFieldProps>(function TextField(
|
|
|
|
props,
|
|
|
|
ref
|
|
|
|
) {
|
|
|
|
const id = useId();
|
2022-08-03 16:59:40 +00:00
|
|
|
const { t: _t } = useLocale();
|
|
|
|
const t = props.t || _t;
|
2022-07-23 00:39:50 +00:00
|
|
|
const methods = useFormContext();
|
|
|
|
const {
|
|
|
|
label = t(props.name as string),
|
|
|
|
labelProps,
|
|
|
|
/** Prevents displaying untranslated placeholder keys */
|
|
|
|
placeholder = t(props.name + "_placeholder") !== props.name + "_placeholder"
|
|
|
|
? t(props.name + "_placeholder")
|
|
|
|
: "",
|
|
|
|
...passThrough
|
|
|
|
} = props;
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
{!!props.name && (
|
|
|
|
<Label htmlFor={id} {...labelProps}>
|
|
|
|
{label}
|
|
|
|
</Label>
|
|
|
|
)}
|
|
|
|
<TextArea ref={ref} placeholder={placeholder} {...passThrough} />
|
2022-08-17 17:38:21 +00:00
|
|
|
{methods?.formState?.errors[props.name]?.message && (
|
|
|
|
<Alert
|
|
|
|
className="mt-1"
|
|
|
|
severity="error"
|
2022-11-04 15:40:46 +00:00
|
|
|
message={<>{methods.formState.errors[props.name]?.message}</>}
|
2022-08-17 17:38:21 +00:00
|
|
|
/>
|
2022-07-23 00:39:50 +00:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
type FormProps<T extends object> = { form: UseFormReturn<T>; handleSubmit: SubmitHandler<T> } & Omit<
|
|
|
|
JSX.IntrinsicElements["form"],
|
|
|
|
"onSubmit"
|
|
|
|
>;
|
|
|
|
|
|
|
|
const PlainForm = <T extends FieldValues>(props: FormProps<T>, ref: Ref<HTMLFormElement>) => {
|
|
|
|
const { form, handleSubmit, ...passThrough } = props;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<FormProvider {...form}>
|
|
|
|
<form
|
|
|
|
ref={ref}
|
|
|
|
onSubmit={(event) => {
|
|
|
|
event.preventDefault();
|
2022-08-24 20:18:42 +00:00
|
|
|
event.stopPropagation();
|
2022-07-23 00:39:50 +00:00
|
|
|
form
|
|
|
|
.handleSubmit(handleSubmit)(event)
|
|
|
|
.catch((err) => {
|
2023-03-02 18:15:28 +00:00
|
|
|
// FIXME: Booking Pages don't have toast, so this error is never shown
|
2022-07-23 00:39:50 +00:00
|
|
|
showToast(`${getErrorFromUnknown(err).message}`, "error");
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
{...passThrough}>
|
2022-11-16 13:41:40 +00:00
|
|
|
{props.children}
|
2022-07-23 00:39:50 +00:00
|
|
|
</form>
|
|
|
|
</FormProvider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const Form = forwardRef(PlainForm) as <T extends FieldValues>(
|
|
|
|
p: FormProps<T> & { ref?: Ref<HTMLFormElement> }
|
|
|
|
) => ReactElement;
|
|
|
|
|
|
|
|
export function FieldsetLegend(props: JSX.IntrinsicElements["legend"]) {
|
|
|
|
return (
|
|
|
|
<legend {...props} className={classNames("text-sm font-medium text-gray-700", props.className)}>
|
|
|
|
{props.children}
|
|
|
|
</legend>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function InputGroupBox(props: JSX.IntrinsicElements["div"]) {
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
{...props}
|
|
|
|
className={classNames("space-y-2 rounded-sm border border-gray-300 bg-white p-2", props.className)}>
|
|
|
|
{props.children}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2022-11-28 11:20:49 +00:00
|
|
|
|
|
|
|
export const InputFieldWithSelect = forwardRef<
|
|
|
|
HTMLInputElement,
|
|
|
|
InputFieldProps & { selectProps: typeof UnstyledSelect }
|
|
|
|
>(function EmailField(props, ref) {
|
|
|
|
return (
|
|
|
|
<InputField
|
|
|
|
ref={ref}
|
|
|
|
{...props}
|
|
|
|
inputIsFullWidth={false}
|
2022-11-30 20:51:44 +00:00
|
|
|
addOnClassname="!px-0"
|
2022-11-28 11:20:49 +00:00
|
|
|
addOnSuffix={<UnstyledSelect {...props.selectProps} />}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|