@mirohq/design-system-base-form 1.2.4 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/main.js +17 -123
- package/dist/main.js.map +1 -1
- package/dist/module.js +2 -107
- package/dist/module.js.map +1 -1
- package/dist/types.d.ts +17 -58
- package/package.json +3 -6
package/dist/main.js
CHANGED
|
@@ -1,12 +1,9 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
3
|
var jsxRuntime = require('react/jsx-runtime');
|
|
4
|
-
var
|
|
4
|
+
var react = require('react');
|
|
5
5
|
var designSystemUseId = require('@mirohq/design-system-use-id');
|
|
6
6
|
var designSystemUtils = require('@mirohq/design-system-utils');
|
|
7
|
-
var designSystemStitches = require('@mirohq/design-system-stitches');
|
|
8
|
-
var designSystemPrimitive = require('@mirohq/design-system-primitive');
|
|
9
|
-
var designSystemBaseTextField = require('@mirohq/design-system-base-text-field');
|
|
10
7
|
|
|
11
8
|
const FORM_FIELD_STATUS = Object.freeze({
|
|
12
9
|
// the field value is valid
|
|
@@ -21,35 +18,33 @@ const FORM_FIELD_STATUS = Object.freeze({
|
|
|
21
18
|
touched: "touched"
|
|
22
19
|
});
|
|
23
20
|
|
|
24
|
-
const FormFieldContext =
|
|
21
|
+
const FormFieldContext = react.createContext({});
|
|
25
22
|
const FormFieldProvider = ({
|
|
26
23
|
children,
|
|
27
24
|
status: customStatus,
|
|
28
25
|
id: customId,
|
|
29
26
|
...restProps
|
|
30
27
|
}) => {
|
|
31
|
-
const [status, setStatus] =
|
|
28
|
+
const [status, setStatus] = react.useState(
|
|
32
29
|
customStatus != null ? customStatus : "pristine"
|
|
33
30
|
);
|
|
34
|
-
const [focused, setFocused] =
|
|
35
|
-
const [
|
|
36
|
-
const [
|
|
37
|
-
const [label, setLabel] = React.useState(null);
|
|
38
|
-
const shouldUseFloatingLabel = label != null && isFloatingLabel;
|
|
31
|
+
const [focused, setFocused] = react.useState(false);
|
|
32
|
+
const [isHiddenLabel, setIsHiddenLabel] = react.useState(false);
|
|
33
|
+
const [label, setLabel] = react.useState(null);
|
|
39
34
|
const autoId = designSystemUseId.useId();
|
|
40
35
|
const id = customId != null ? customId : autoId;
|
|
41
|
-
const [messageId, setMessageId] =
|
|
42
|
-
const [helperId, setHelperId] =
|
|
43
|
-
const [descriptionId, setDescriptionId] =
|
|
44
|
-
const formElementRef =
|
|
45
|
-
const [required, setRequired] =
|
|
46
|
-
const [readOnly, setReadOnly] =
|
|
47
|
-
const [disabled, setDisabled] =
|
|
48
|
-
const [ariaDisabled, setAriaDisabled] =
|
|
49
|
-
const [ariaDescribedBy, setAriaDescribedBy] =
|
|
36
|
+
const [messageId, setMessageId] = react.useState("".concat(id, "-message"));
|
|
37
|
+
const [helperId, setHelperId] = react.useState("".concat(id, "-helper"));
|
|
38
|
+
const [descriptionId, setDescriptionId] = react.useState("".concat(id, "-description"));
|
|
39
|
+
const formElementRef = react.useRef(null);
|
|
40
|
+
const [required, setRequired] = react.useState();
|
|
41
|
+
const [readOnly, setReadOnly] = react.useState();
|
|
42
|
+
const [disabled, setDisabled] = react.useState();
|
|
43
|
+
const [ariaDisabled, setAriaDisabled] = react.useState();
|
|
44
|
+
const [ariaDescribedBy, setAriaDescribedBy] = react.useState("");
|
|
50
45
|
const valid = status === FORM_FIELD_STATUS.valid ? true : status === FORM_FIELD_STATUS.invalid ? false : void 0;
|
|
51
46
|
const ariaInvalid = status === FORM_FIELD_STATUS.invalid ? true : void 0;
|
|
52
|
-
|
|
47
|
+
react.useEffect(() => {
|
|
53
48
|
const message = document.getElementById(messageId);
|
|
54
49
|
const helper = document.getElementById(helperId);
|
|
55
50
|
const description = document.getElementById(descriptionId);
|
|
@@ -78,14 +73,11 @@ const FormFieldProvider = ({
|
|
|
78
73
|
setStatus,
|
|
79
74
|
focused,
|
|
80
75
|
setFocused,
|
|
81
|
-
isFloatingLabel,
|
|
82
|
-
setIsFloatingLabel,
|
|
83
76
|
isHiddenLabel,
|
|
84
77
|
setIsHiddenLabel,
|
|
85
78
|
label,
|
|
86
79
|
setLabel,
|
|
87
80
|
ariaDescribedBy,
|
|
88
|
-
shouldUseFloatingLabel,
|
|
89
81
|
required,
|
|
90
82
|
setRequired,
|
|
91
83
|
readOnly,
|
|
@@ -101,107 +93,9 @@ const FormFieldProvider = ({
|
|
|
101
93
|
}
|
|
102
94
|
);
|
|
103
95
|
};
|
|
104
|
-
const useFormFieldContext = () =>
|
|
105
|
-
|
|
106
|
-
const StyledFloatingLabel = designSystemStitches.styled(designSystemPrimitive.Primitive.label, {
|
|
107
|
-
transitionProperty: "transform, margin, top, left, font-size",
|
|
108
|
-
transitionDuration: "200ms",
|
|
109
|
-
transitionTimingFunction: "cubic-bezier(0, 0, 0.2, 1)",
|
|
110
|
-
position: "absolute",
|
|
111
|
-
top: 0,
|
|
112
|
-
left: 0,
|
|
113
|
-
zIndex: 1,
|
|
114
|
-
pointerEvents: "none",
|
|
115
|
-
whiteSpace: "nowrap",
|
|
116
|
-
fontStyle: "normal",
|
|
117
|
-
lineHeight: 1.5,
|
|
118
|
-
height: "0.75em",
|
|
119
|
-
...designSystemBaseTextField.textFieldStyles.base.placeholder,
|
|
120
|
-
variants: {
|
|
121
|
-
floating: {
|
|
122
|
-
true: {
|
|
123
|
-
top: "0",
|
|
124
|
-
transform: "translateY(-50%)",
|
|
125
|
-
left: "$150",
|
|
126
|
-
fontSize: "$150",
|
|
127
|
-
padding: "0 2px",
|
|
128
|
-
height: "auto",
|
|
129
|
-
lineHeight: 1,
|
|
130
|
-
backgroundColor: "$background-neutrals-container"
|
|
131
|
-
},
|
|
132
|
-
false: {
|
|
133
|
-
fontSize: "$200",
|
|
134
|
-
marginTop: "-1px"
|
|
135
|
-
// adjust for the input's border
|
|
136
|
-
}
|
|
137
|
-
},
|
|
138
|
-
visuallyHidden: {
|
|
139
|
-
true: {
|
|
140
|
-
// todo MDS-1003: replace with shared styles
|
|
141
|
-
border: "0",
|
|
142
|
-
clip: "rect(0 0 0 0)",
|
|
143
|
-
height: "1px",
|
|
144
|
-
margin: "-1px",
|
|
145
|
-
overflow: "hidden",
|
|
146
|
-
padding: "0",
|
|
147
|
-
position: "absolute",
|
|
148
|
-
width: "1px"
|
|
149
|
-
}
|
|
150
|
-
},
|
|
151
|
-
size: {
|
|
152
|
-
medium: {},
|
|
153
|
-
large: {},
|
|
154
|
-
"x-large": {}
|
|
155
|
-
}
|
|
156
|
-
},
|
|
157
|
-
compoundVariants: [
|
|
158
|
-
{
|
|
159
|
-
size: "medium",
|
|
160
|
-
floating: false,
|
|
161
|
-
css: {
|
|
162
|
-
top: "$50",
|
|
163
|
-
left: "$100"
|
|
164
|
-
}
|
|
165
|
-
},
|
|
166
|
-
{
|
|
167
|
-
size: "large",
|
|
168
|
-
floating: false,
|
|
169
|
-
css: {
|
|
170
|
-
top: "$100",
|
|
171
|
-
left: "$150"
|
|
172
|
-
}
|
|
173
|
-
},
|
|
174
|
-
{
|
|
175
|
-
size: "x-large",
|
|
176
|
-
floating: false,
|
|
177
|
-
css: {
|
|
178
|
-
top: "$150",
|
|
179
|
-
// $space$250 does not exist by design but we will force it here
|
|
180
|
-
left: "calc($200 + $space$50)"
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
],
|
|
184
|
-
defaultVariants: {
|
|
185
|
-
size: "large"
|
|
186
|
-
}
|
|
187
|
-
});
|
|
188
|
-
|
|
189
|
-
const FloatingLabel = React.forwardRef((props, forwardRef) => {
|
|
190
|
-
const { isFloatingLabel, isHiddenLabel, formElementId } = useFormFieldContext();
|
|
191
|
-
return /* @__PURE__ */ jsxRuntime.jsx(
|
|
192
|
-
StyledFloatingLabel,
|
|
193
|
-
{
|
|
194
|
-
htmlFor: formElementId,
|
|
195
|
-
floating: isFloatingLabel,
|
|
196
|
-
visuallyHidden: isHiddenLabel,
|
|
197
|
-
...props,
|
|
198
|
-
ref: forwardRef
|
|
199
|
-
}
|
|
200
|
-
);
|
|
201
|
-
});
|
|
96
|
+
const useFormFieldContext = () => react.useContext(FormFieldContext);
|
|
202
97
|
|
|
203
98
|
exports.FORM_FIELD_STATUS = FORM_FIELD_STATUS;
|
|
204
|
-
exports.FloatingLabel = FloatingLabel;
|
|
205
99
|
exports.FormFieldProvider = FormFieldProvider;
|
|
206
100
|
exports.useFormFieldContext = useFormFieldContext;
|
|
207
101
|
//# sourceMappingURL=main.js.map
|
package/dist/main.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"main.js","sources":["../src/constants.ts","../src/hooks/use-form-field-context.tsx","../src/floating-label.styled.ts","../src/floating-label.tsx"],"sourcesContent":["export const FORM_FIELD_STATUS = Object.freeze({\n // the field value is valid\n valid: 'valid',\n // the field value is invalid\n invalid: 'invalid',\n // the field value has been changed\n dirty: 'dirty',\n // the field value has not been changed\n pristine: 'pristine',\n // the field has been focused and blurred\n touched: 'touched',\n})\n\nexport type FormFieldStatus = keyof typeof FORM_FIELD_STATUS\n","import React, {\n createContext,\n useContext,\n useState,\n useRef,\n useEffect,\n} from 'react'\nimport type { AriaAttributes } from 'react'\nimport type { Booleanish } from '@mirohq/design-system-types'\nimport { useId } from '@mirohq/design-system-use-id'\nimport { stringAttrValue } from '@mirohq/design-system-utils'\n\nimport type { FormFieldStatus } from '../constants'\nimport { FORM_FIELD_STATUS } from '../constants'\n\ntype FormElements = HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement\n\ninterface FormFieldProps {\n /**\n * Field status\n * @default \"pristine\"\n */\n status?: FormFieldStatus\n\n id?: string\n}\n\ninterface FormFieldContextProps<FormElementRef extends FormElements>\n extends FormFieldProps {\n setStatus: React.Dispatch<React.SetStateAction<FormFieldStatus>>\n isFloatingLabel: boolean\n setIsFloatingLabel: React.Dispatch<React.SetStateAction<boolean>>\n isHiddenLabel: boolean\n setIsHiddenLabel: React.Dispatch<React.SetStateAction<boolean>>\n label: React.ReactNode\n setLabel: React.Dispatch<React.SetStateAction<React.ReactNode>>\n focused: boolean\n setFocused: React.Dispatch<React.SetStateAction<boolean>>\n helperId: string\n setHelperId: React.Dispatch<React.SetStateAction<string>>\n messageId: string\n setMessageId: React.Dispatch<React.SetStateAction<string>>\n descriptionId: string\n setDescriptionId: React.Dispatch<React.SetStateAction<string>>\n formElementId: string\n formElementRef: React.MutableRefObject<FormElementRef>\n ariaDescribedBy: string | undefined\n shouldUseFloatingLabel: boolean\n required: boolean | undefined\n setRequired: React.Dispatch<React.SetStateAction<boolean | undefined>>\n readOnly: boolean | undefined\n setReadOnly: React.Dispatch<React.SetStateAction<boolean | undefined>>\n disabled: boolean | undefined\n setDisabled: React.Dispatch<React.SetStateAction<boolean | undefined>>\n ariaDisabled: Booleanish | undefined\n setAriaDisabled: React.Dispatch<React.SetStateAction<Booleanish | undefined>>\n ariaInvalid: AriaAttributes['aria-invalid']\n valid: boolean | undefined\n}\n\nexport interface FormFieldProviderProps extends FormFieldProps {\n children?: React.ReactNode\n}\n\nconst FormFieldContext = createContext<FormFieldContextProps<any>>({} as any)\n\nexport const FormFieldProvider = ({\n children,\n status: customStatus,\n id: customId,\n ...restProps\n}: FormFieldProviderProps): JSX.Element => {\n // form field state\n const [status, setStatus] = useState<FormFieldStatus>(\n customStatus ?? 'pristine'\n )\n const [focused, setFocused] = useState(false)\n const [isFloatingLabel, setIsFloatingLabel] = useState(true)\n const [isHiddenLabel, setIsHiddenLabel] = useState(false)\n const [label, setLabel] = useState<React.ReactNode>(null)\n const shouldUseFloatingLabel = label != null && isFloatingLabel\n\n // partials\n const autoId = useId()\n const id = customId ?? autoId\n\n const [messageId, setMessageId] = useState(`${id}-message`)\n const [helperId, setHelperId] = useState(`${id}-helper`)\n const [descriptionId, setDescriptionId] = useState(`${id}-description`)\n\n // form element\n const formElementRef = useRef<HTMLInputElement>(null)\n const [required, setRequired] = useState<boolean>()\n const [readOnly, setReadOnly] = useState<boolean>()\n const [disabled, setDisabled] = useState<boolean>()\n const [ariaDisabled, setAriaDisabled] = useState<Booleanish>()\n const [ariaDescribedBy, setAriaDescribedBy] = useState<string | undefined>('')\n\n const valid =\n status === FORM_FIELD_STATUS.valid\n ? true\n : status === FORM_FIELD_STATUS.invalid\n ? false\n : undefined\n\n const ariaInvalid: AriaAttributes['aria-invalid'] =\n status === FORM_FIELD_STATUS.invalid ? true : undefined\n\n useEffect(() => {\n const message = document.getElementById(messageId)\n const helper = document.getElementById(helperId)\n const description = document.getElementById(descriptionId)\n\n const ariaDescribedBy = stringAttrValue(\n // message or description\n message != null ? messageId : description != null && descriptionId,\n helper != null && helperId\n )\n\n /**\n * message + helper text\n * description + helper text\n */\n setAriaDescribedBy(ariaDescribedBy === '' ? undefined : ariaDescribedBy)\n }, [messageId, helperId, descriptionId])\n\n return (\n <FormFieldContext.Provider\n value={{\n ...restProps,\n id,\n formElementRef,\n formElementId: `${id}-form-element`,\n messageId,\n setMessageId,\n helperId,\n setHelperId,\n descriptionId,\n setDescriptionId,\n status,\n setStatus,\n focused,\n setFocused,\n isFloatingLabel,\n setIsFloatingLabel,\n isHiddenLabel,\n setIsHiddenLabel,\n label,\n setLabel,\n ariaDescribedBy,\n shouldUseFloatingLabel,\n required,\n setRequired,\n readOnly,\n setReadOnly,\n disabled,\n setDisabled,\n ariaDisabled,\n setAriaDisabled,\n ariaInvalid,\n valid,\n }}\n >\n {children}\n </FormFieldContext.Provider>\n )\n}\n\nexport const useFormFieldContext = <\n T extends FormElements\n>(): FormFieldContextProps<T> => useContext(FormFieldContext)\n","import type { ComponentPropsWithRef } from 'react'\nimport { styled } from '@mirohq/design-system-stitches'\nimport { Primitive } from '@mirohq/design-system-primitive'\nimport { textFieldStyles } from '@mirohq/design-system-base-text-field'\n\nexport const StyledFloatingLabel = styled(Primitive.label, {\n transitionProperty: 'transform, margin, top, left, font-size',\n transitionDuration: '200ms',\n transitionTimingFunction: 'cubic-bezier(0, 0, 0.2, 1)',\n position: 'absolute',\n top: 0,\n left: 0,\n zIndex: 1,\n pointerEvents: 'none',\n whiteSpace: 'nowrap',\n fontStyle: 'normal',\n lineHeight: 1.5,\n height: '0.75em',\n ...textFieldStyles.base.placeholder,\n\n variants: {\n floating: {\n true: {\n top: '0',\n transform: 'translateY(-50%)',\n left: '$150',\n fontSize: '$150',\n padding: '0 2px',\n height: 'auto',\n lineHeight: 1,\n backgroundColor: '$background-neutrals-container',\n },\n false: {\n fontSize: '$200',\n marginTop: '-1px', // adjust for the input's border\n },\n },\n visuallyHidden: {\n true: {\n // todo MDS-1003: replace with shared styles\n border: '0',\n clip: 'rect(0 0 0 0)',\n height: '1px',\n margin: '-1px',\n overflow: 'hidden',\n padding: '0',\n position: 'absolute',\n width: '1px',\n },\n },\n size: {\n medium: {},\n large: {},\n 'x-large': {},\n },\n },\n compoundVariants: [\n {\n size: 'medium',\n floating: false,\n css: {\n top: '$50',\n left: '$100',\n },\n },\n {\n size: 'large',\n floating: false,\n css: {\n top: '$100',\n left: '$150',\n },\n },\n {\n size: 'x-large',\n floating: false,\n css: {\n top: '$150',\n // $space$250 does not exist by design but we will force it here\n left: 'calc($200 + $space$50)',\n },\n },\n ],\n defaultVariants: {\n size: 'large',\n },\n})\n\nexport type StyledFloatingLabelProps = ComponentPropsWithRef<\n typeof StyledFloatingLabel\n>\n","import React from 'react'\nimport type { ElementRef } from 'react'\n\nimport { useFormFieldContext } from './hooks/use-form-field-context'\nimport { StyledFloatingLabel } from './floating-label.styled'\nimport type { StyledFloatingLabelProps } from './floating-label.styled'\n\nexport interface FloatingLabelProps\n extends Omit<StyledFloatingLabelProps, 'visuallyHidden'> {}\n\nexport const FloatingLabel = React.forwardRef<\n ElementRef<typeof StyledFloatingLabel>,\n FloatingLabelProps\n>((props, forwardRef) => {\n const { isFloatingLabel, isHiddenLabel, formElementId } =\n useFormFieldContext()\n\n return (\n <StyledFloatingLabel\n htmlFor={formElementId}\n floating={isFloatingLabel}\n visuallyHidden={isHiddenLabel}\n {...props}\n ref={forwardRef}\n />\n )\n})\n"],"names":["createContext","useState","useId","useRef","useEffect","ariaDescribedBy","stringAttrValue","jsx","useContext","styled","Primitive","textFieldStyles"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"main.js","sources":["../src/constants.ts","../src/hooks/use-form-field-context.tsx"],"sourcesContent":["export const FORM_FIELD_STATUS = Object.freeze({\n // the field value is valid\n valid: 'valid',\n // the field value is invalid\n invalid: 'invalid',\n // the field value has been changed\n dirty: 'dirty',\n // the field value has not been changed\n pristine: 'pristine',\n // the field has been focused and blurred\n touched: 'touched',\n})\n\nexport type FormFieldStatus = keyof typeof FORM_FIELD_STATUS\n","import React, {\n createContext,\n useContext,\n useState,\n useRef,\n useEffect,\n} from 'react'\nimport type { AriaAttributes } from 'react'\nimport type { Booleanish } from '@mirohq/design-system-types'\nimport { useId } from '@mirohq/design-system-use-id'\nimport { stringAttrValue } from '@mirohq/design-system-utils'\n\nimport type { FormFieldStatus } from '../constants'\nimport { FORM_FIELD_STATUS } from '../constants'\n\ntype FormElements = HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement\n\ninterface FormFieldProps {\n /**\n * Field status\n * @default \"pristine\"\n */\n status?: FormFieldStatus\n\n id?: string\n}\n\ninterface FormFieldContextProps<FormElementRef extends FormElements>\n extends FormFieldProps {\n setStatus: React.Dispatch<React.SetStateAction<FormFieldStatus>>\n isHiddenLabel: boolean\n setIsHiddenLabel: React.Dispatch<React.SetStateAction<boolean>>\n label: React.ReactNode\n setLabel: React.Dispatch<React.SetStateAction<React.ReactNode>>\n focused: boolean\n setFocused: React.Dispatch<React.SetStateAction<boolean>>\n helperId: string\n setHelperId: React.Dispatch<React.SetStateAction<string>>\n messageId: string\n setMessageId: React.Dispatch<React.SetStateAction<string>>\n descriptionId: string\n setDescriptionId: React.Dispatch<React.SetStateAction<string>>\n formElementId: string\n formElementRef: React.MutableRefObject<FormElementRef>\n ariaDescribedBy: string | undefined\n required: boolean | undefined\n setRequired: React.Dispatch<React.SetStateAction<boolean | undefined>>\n readOnly: boolean | undefined\n setReadOnly: React.Dispatch<React.SetStateAction<boolean | undefined>>\n disabled: boolean | undefined\n setDisabled: React.Dispatch<React.SetStateAction<boolean | undefined>>\n ariaDisabled: Booleanish | undefined\n setAriaDisabled: React.Dispatch<React.SetStateAction<Booleanish | undefined>>\n ariaInvalid: AriaAttributes['aria-invalid']\n valid: boolean | undefined\n}\n\nexport interface FormFieldProviderProps extends FormFieldProps {\n children?: React.ReactNode\n}\n\nconst FormFieldContext = createContext<FormFieldContextProps<any>>({} as any)\n\nexport const FormFieldProvider = ({\n children,\n status: customStatus,\n id: customId,\n ...restProps\n}: FormFieldProviderProps): JSX.Element => {\n // form field state\n const [status, setStatus] = useState<FormFieldStatus>(\n customStatus ?? 'pristine'\n )\n const [focused, setFocused] = useState(false)\n const [isHiddenLabel, setIsHiddenLabel] = useState(false)\n const [label, setLabel] = useState<React.ReactNode>(null)\n\n // partials\n const autoId = useId()\n const id = customId ?? autoId\n\n const [messageId, setMessageId] = useState(`${id}-message`)\n const [helperId, setHelperId] = useState(`${id}-helper`)\n const [descriptionId, setDescriptionId] = useState(`${id}-description`)\n\n // form element\n const formElementRef = useRef<HTMLInputElement>(null)\n const [required, setRequired] = useState<boolean>()\n const [readOnly, setReadOnly] = useState<boolean>()\n const [disabled, setDisabled] = useState<boolean>()\n const [ariaDisabled, setAriaDisabled] = useState<Booleanish>()\n const [ariaDescribedBy, setAriaDescribedBy] = useState<string | undefined>('')\n\n const valid =\n status === FORM_FIELD_STATUS.valid\n ? true\n : status === FORM_FIELD_STATUS.invalid\n ? false\n : undefined\n\n const ariaInvalid: AriaAttributes['aria-invalid'] =\n status === FORM_FIELD_STATUS.invalid ? true : undefined\n\n useEffect(() => {\n const message = document.getElementById(messageId)\n const helper = document.getElementById(helperId)\n const description = document.getElementById(descriptionId)\n\n const ariaDescribedBy = stringAttrValue(\n // message or description\n message != null ? messageId : description != null && descriptionId,\n helper != null && helperId\n )\n\n /**\n * message + helper text\n * description + helper text\n */\n setAriaDescribedBy(ariaDescribedBy === '' ? undefined : ariaDescribedBy)\n }, [messageId, helperId, descriptionId])\n\n return (\n <FormFieldContext.Provider\n value={{\n ...restProps,\n id,\n formElementRef,\n formElementId: `${id}-form-element`,\n messageId,\n setMessageId,\n helperId,\n setHelperId,\n descriptionId,\n setDescriptionId,\n status,\n setStatus,\n focused,\n setFocused,\n isHiddenLabel,\n setIsHiddenLabel,\n label,\n setLabel,\n ariaDescribedBy,\n required,\n setRequired,\n readOnly,\n setReadOnly,\n disabled,\n setDisabled,\n ariaDisabled,\n setAriaDisabled,\n ariaInvalid,\n valid,\n }}\n >\n {children}\n </FormFieldContext.Provider>\n )\n}\n\nexport const useFormFieldContext = <\n T extends FormElements\n>(): FormFieldContextProps<T> => useContext(FormFieldContext)\n"],"names":["createContext","useState","useId","useRef","useEffect","ariaDescribedBy","stringAttrValue","jsx","useContext"],"mappings":"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"}
|
package/dist/module.js
CHANGED
|
@@ -1,10 +1,7 @@
|
|
|
1
1
|
import { jsx } from 'react/jsx-runtime';
|
|
2
|
-
import
|
|
2
|
+
import { createContext, useContext, useState, useRef, useEffect } from 'react';
|
|
3
3
|
import { useId } from '@mirohq/design-system-use-id';
|
|
4
4
|
import { stringAttrValue } from '@mirohq/design-system-utils';
|
|
5
|
-
import { styled } from '@mirohq/design-system-stitches';
|
|
6
|
-
import { Primitive } from '@mirohq/design-system-primitive';
|
|
7
|
-
import { textFieldStyles } from '@mirohq/design-system-base-text-field';
|
|
8
5
|
|
|
9
6
|
const FORM_FIELD_STATUS = Object.freeze({
|
|
10
7
|
// the field value is valid
|
|
@@ -30,10 +27,8 @@ const FormFieldProvider = ({
|
|
|
30
27
|
customStatus != null ? customStatus : "pristine"
|
|
31
28
|
);
|
|
32
29
|
const [focused, setFocused] = useState(false);
|
|
33
|
-
const [isFloatingLabel, setIsFloatingLabel] = useState(true);
|
|
34
30
|
const [isHiddenLabel, setIsHiddenLabel] = useState(false);
|
|
35
31
|
const [label, setLabel] = useState(null);
|
|
36
|
-
const shouldUseFloatingLabel = label != null && isFloatingLabel;
|
|
37
32
|
const autoId = useId();
|
|
38
33
|
const id = customId != null ? customId : autoId;
|
|
39
34
|
const [messageId, setMessageId] = useState("".concat(id, "-message"));
|
|
@@ -76,14 +71,11 @@ const FormFieldProvider = ({
|
|
|
76
71
|
setStatus,
|
|
77
72
|
focused,
|
|
78
73
|
setFocused,
|
|
79
|
-
isFloatingLabel,
|
|
80
|
-
setIsFloatingLabel,
|
|
81
74
|
isHiddenLabel,
|
|
82
75
|
setIsHiddenLabel,
|
|
83
76
|
label,
|
|
84
77
|
setLabel,
|
|
85
78
|
ariaDescribedBy,
|
|
86
|
-
shouldUseFloatingLabel,
|
|
87
79
|
required,
|
|
88
80
|
setRequired,
|
|
89
81
|
readOnly,
|
|
@@ -101,102 +93,5 @@ const FormFieldProvider = ({
|
|
|
101
93
|
};
|
|
102
94
|
const useFormFieldContext = () => useContext(FormFieldContext);
|
|
103
95
|
|
|
104
|
-
|
|
105
|
-
transitionProperty: "transform, margin, top, left, font-size",
|
|
106
|
-
transitionDuration: "200ms",
|
|
107
|
-
transitionTimingFunction: "cubic-bezier(0, 0, 0.2, 1)",
|
|
108
|
-
position: "absolute",
|
|
109
|
-
top: 0,
|
|
110
|
-
left: 0,
|
|
111
|
-
zIndex: 1,
|
|
112
|
-
pointerEvents: "none",
|
|
113
|
-
whiteSpace: "nowrap",
|
|
114
|
-
fontStyle: "normal",
|
|
115
|
-
lineHeight: 1.5,
|
|
116
|
-
height: "0.75em",
|
|
117
|
-
...textFieldStyles.base.placeholder,
|
|
118
|
-
variants: {
|
|
119
|
-
floating: {
|
|
120
|
-
true: {
|
|
121
|
-
top: "0",
|
|
122
|
-
transform: "translateY(-50%)",
|
|
123
|
-
left: "$150",
|
|
124
|
-
fontSize: "$150",
|
|
125
|
-
padding: "0 2px",
|
|
126
|
-
height: "auto",
|
|
127
|
-
lineHeight: 1,
|
|
128
|
-
backgroundColor: "$background-neutrals-container"
|
|
129
|
-
},
|
|
130
|
-
false: {
|
|
131
|
-
fontSize: "$200",
|
|
132
|
-
marginTop: "-1px"
|
|
133
|
-
// adjust for the input's border
|
|
134
|
-
}
|
|
135
|
-
},
|
|
136
|
-
visuallyHidden: {
|
|
137
|
-
true: {
|
|
138
|
-
// todo MDS-1003: replace with shared styles
|
|
139
|
-
border: "0",
|
|
140
|
-
clip: "rect(0 0 0 0)",
|
|
141
|
-
height: "1px",
|
|
142
|
-
margin: "-1px",
|
|
143
|
-
overflow: "hidden",
|
|
144
|
-
padding: "0",
|
|
145
|
-
position: "absolute",
|
|
146
|
-
width: "1px"
|
|
147
|
-
}
|
|
148
|
-
},
|
|
149
|
-
size: {
|
|
150
|
-
medium: {},
|
|
151
|
-
large: {},
|
|
152
|
-
"x-large": {}
|
|
153
|
-
}
|
|
154
|
-
},
|
|
155
|
-
compoundVariants: [
|
|
156
|
-
{
|
|
157
|
-
size: "medium",
|
|
158
|
-
floating: false,
|
|
159
|
-
css: {
|
|
160
|
-
top: "$50",
|
|
161
|
-
left: "$100"
|
|
162
|
-
}
|
|
163
|
-
},
|
|
164
|
-
{
|
|
165
|
-
size: "large",
|
|
166
|
-
floating: false,
|
|
167
|
-
css: {
|
|
168
|
-
top: "$100",
|
|
169
|
-
left: "$150"
|
|
170
|
-
}
|
|
171
|
-
},
|
|
172
|
-
{
|
|
173
|
-
size: "x-large",
|
|
174
|
-
floating: false,
|
|
175
|
-
css: {
|
|
176
|
-
top: "$150",
|
|
177
|
-
// $space$250 does not exist by design but we will force it here
|
|
178
|
-
left: "calc($200 + $space$50)"
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
],
|
|
182
|
-
defaultVariants: {
|
|
183
|
-
size: "large"
|
|
184
|
-
}
|
|
185
|
-
});
|
|
186
|
-
|
|
187
|
-
const FloatingLabel = React.forwardRef((props, forwardRef) => {
|
|
188
|
-
const { isFloatingLabel, isHiddenLabel, formElementId } = useFormFieldContext();
|
|
189
|
-
return /* @__PURE__ */ jsx(
|
|
190
|
-
StyledFloatingLabel,
|
|
191
|
-
{
|
|
192
|
-
htmlFor: formElementId,
|
|
193
|
-
floating: isFloatingLabel,
|
|
194
|
-
visuallyHidden: isHiddenLabel,
|
|
195
|
-
...props,
|
|
196
|
-
ref: forwardRef
|
|
197
|
-
}
|
|
198
|
-
);
|
|
199
|
-
});
|
|
200
|
-
|
|
201
|
-
export { FORM_FIELD_STATUS, FloatingLabel, FormFieldProvider, useFormFieldContext };
|
|
96
|
+
export { FORM_FIELD_STATUS, FormFieldProvider, useFormFieldContext };
|
|
202
97
|
//# sourceMappingURL=module.js.map
|
package/dist/module.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"module.js","sources":["../src/constants.ts","../src/hooks/use-form-field-context.tsx","../src/floating-label.styled.ts","../src/floating-label.tsx"],"sourcesContent":["export const FORM_FIELD_STATUS = Object.freeze({\n // the field value is valid\n valid: 'valid',\n // the field value is invalid\n invalid: 'invalid',\n // the field value has been changed\n dirty: 'dirty',\n // the field value has not been changed\n pristine: 'pristine',\n // the field has been focused and blurred\n touched: 'touched',\n})\n\nexport type FormFieldStatus = keyof typeof FORM_FIELD_STATUS\n","import React, {\n createContext,\n useContext,\n useState,\n useRef,\n useEffect,\n} from 'react'\nimport type { AriaAttributes } from 'react'\nimport type { Booleanish } from '@mirohq/design-system-types'\nimport { useId } from '@mirohq/design-system-use-id'\nimport { stringAttrValue } from '@mirohq/design-system-utils'\n\nimport type { FormFieldStatus } from '../constants'\nimport { FORM_FIELD_STATUS } from '../constants'\n\ntype FormElements = HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement\n\ninterface FormFieldProps {\n /**\n * Field status\n * @default \"pristine\"\n */\n status?: FormFieldStatus\n\n id?: string\n}\n\ninterface FormFieldContextProps<FormElementRef extends FormElements>\n extends FormFieldProps {\n setStatus: React.Dispatch<React.SetStateAction<FormFieldStatus>>\n isFloatingLabel: boolean\n setIsFloatingLabel: React.Dispatch<React.SetStateAction<boolean>>\n isHiddenLabel: boolean\n setIsHiddenLabel: React.Dispatch<React.SetStateAction<boolean>>\n label: React.ReactNode\n setLabel: React.Dispatch<React.SetStateAction<React.ReactNode>>\n focused: boolean\n setFocused: React.Dispatch<React.SetStateAction<boolean>>\n helperId: string\n setHelperId: React.Dispatch<React.SetStateAction<string>>\n messageId: string\n setMessageId: React.Dispatch<React.SetStateAction<string>>\n descriptionId: string\n setDescriptionId: React.Dispatch<React.SetStateAction<string>>\n formElementId: string\n formElementRef: React.MutableRefObject<FormElementRef>\n ariaDescribedBy: string | undefined\n shouldUseFloatingLabel: boolean\n required: boolean | undefined\n setRequired: React.Dispatch<React.SetStateAction<boolean | undefined>>\n readOnly: boolean | undefined\n setReadOnly: React.Dispatch<React.SetStateAction<boolean | undefined>>\n disabled: boolean | undefined\n setDisabled: React.Dispatch<React.SetStateAction<boolean | undefined>>\n ariaDisabled: Booleanish | undefined\n setAriaDisabled: React.Dispatch<React.SetStateAction<Booleanish | undefined>>\n ariaInvalid: AriaAttributes['aria-invalid']\n valid: boolean | undefined\n}\n\nexport interface FormFieldProviderProps extends FormFieldProps {\n children?: React.ReactNode\n}\n\nconst FormFieldContext = createContext<FormFieldContextProps<any>>({} as any)\n\nexport const FormFieldProvider = ({\n children,\n status: customStatus,\n id: customId,\n ...restProps\n}: FormFieldProviderProps): JSX.Element => {\n // form field state\n const [status, setStatus] = useState<FormFieldStatus>(\n customStatus ?? 'pristine'\n )\n const [focused, setFocused] = useState(false)\n const [isFloatingLabel, setIsFloatingLabel] = useState(true)\n const [isHiddenLabel, setIsHiddenLabel] = useState(false)\n const [label, setLabel] = useState<React.ReactNode>(null)\n const shouldUseFloatingLabel = label != null && isFloatingLabel\n\n // partials\n const autoId = useId()\n const id = customId ?? autoId\n\n const [messageId, setMessageId] = useState(`${id}-message`)\n const [helperId, setHelperId] = useState(`${id}-helper`)\n const [descriptionId, setDescriptionId] = useState(`${id}-description`)\n\n // form element\n const formElementRef = useRef<HTMLInputElement>(null)\n const [required, setRequired] = useState<boolean>()\n const [readOnly, setReadOnly] = useState<boolean>()\n const [disabled, setDisabled] = useState<boolean>()\n const [ariaDisabled, setAriaDisabled] = useState<Booleanish>()\n const [ariaDescribedBy, setAriaDescribedBy] = useState<string | undefined>('')\n\n const valid =\n status === FORM_FIELD_STATUS.valid\n ? true\n : status === FORM_FIELD_STATUS.invalid\n ? false\n : undefined\n\n const ariaInvalid: AriaAttributes['aria-invalid'] =\n status === FORM_FIELD_STATUS.invalid ? true : undefined\n\n useEffect(() => {\n const message = document.getElementById(messageId)\n const helper = document.getElementById(helperId)\n const description = document.getElementById(descriptionId)\n\n const ariaDescribedBy = stringAttrValue(\n // message or description\n message != null ? messageId : description != null && descriptionId,\n helper != null && helperId\n )\n\n /**\n * message + helper text\n * description + helper text\n */\n setAriaDescribedBy(ariaDescribedBy === '' ? undefined : ariaDescribedBy)\n }, [messageId, helperId, descriptionId])\n\n return (\n <FormFieldContext.Provider\n value={{\n ...restProps,\n id,\n formElementRef,\n formElementId: `${id}-form-element`,\n messageId,\n setMessageId,\n helperId,\n setHelperId,\n descriptionId,\n setDescriptionId,\n status,\n setStatus,\n focused,\n setFocused,\n isFloatingLabel,\n setIsFloatingLabel,\n isHiddenLabel,\n setIsHiddenLabel,\n label,\n setLabel,\n ariaDescribedBy,\n shouldUseFloatingLabel,\n required,\n setRequired,\n readOnly,\n setReadOnly,\n disabled,\n setDisabled,\n ariaDisabled,\n setAriaDisabled,\n ariaInvalid,\n valid,\n }}\n >\n {children}\n </FormFieldContext.Provider>\n )\n}\n\nexport const useFormFieldContext = <\n T extends FormElements\n>(): FormFieldContextProps<T> => useContext(FormFieldContext)\n","import type { ComponentPropsWithRef } from 'react'\nimport { styled } from '@mirohq/design-system-stitches'\nimport { Primitive } from '@mirohq/design-system-primitive'\nimport { textFieldStyles } from '@mirohq/design-system-base-text-field'\n\nexport const StyledFloatingLabel = styled(Primitive.label, {\n transitionProperty: 'transform, margin, top, left, font-size',\n transitionDuration: '200ms',\n transitionTimingFunction: 'cubic-bezier(0, 0, 0.2, 1)',\n position: 'absolute',\n top: 0,\n left: 0,\n zIndex: 1,\n pointerEvents: 'none',\n whiteSpace: 'nowrap',\n fontStyle: 'normal',\n lineHeight: 1.5,\n height: '0.75em',\n ...textFieldStyles.base.placeholder,\n\n variants: {\n floating: {\n true: {\n top: '0',\n transform: 'translateY(-50%)',\n left: '$150',\n fontSize: '$150',\n padding: '0 2px',\n height: 'auto',\n lineHeight: 1,\n backgroundColor: '$background-neutrals-container',\n },\n false: {\n fontSize: '$200',\n marginTop: '-1px', // adjust for the input's border\n },\n },\n visuallyHidden: {\n true: {\n // todo MDS-1003: replace with shared styles\n border: '0',\n clip: 'rect(0 0 0 0)',\n height: '1px',\n margin: '-1px',\n overflow: 'hidden',\n padding: '0',\n position: 'absolute',\n width: '1px',\n },\n },\n size: {\n medium: {},\n large: {},\n 'x-large': {},\n },\n },\n compoundVariants: [\n {\n size: 'medium',\n floating: false,\n css: {\n top: '$50',\n left: '$100',\n },\n },\n {\n size: 'large',\n floating: false,\n css: {\n top: '$100',\n left: '$150',\n },\n },\n {\n size: 'x-large',\n floating: false,\n css: {\n top: '$150',\n // $space$250 does not exist by design but we will force it here\n left: 'calc($200 + $space$50)',\n },\n },\n ],\n defaultVariants: {\n size: 'large',\n },\n})\n\nexport type StyledFloatingLabelProps = ComponentPropsWithRef<\n typeof StyledFloatingLabel\n>\n","import React from 'react'\nimport type { ElementRef } from 'react'\n\nimport { useFormFieldContext } from './hooks/use-form-field-context'\nimport { StyledFloatingLabel } from './floating-label.styled'\nimport type { StyledFloatingLabelProps } from './floating-label.styled'\n\nexport interface FloatingLabelProps\n extends Omit<StyledFloatingLabelProps, 'visuallyHidden'> {}\n\nexport const FloatingLabel = React.forwardRef<\n ElementRef<typeof StyledFloatingLabel>,\n FloatingLabelProps\n>((props, forwardRef) => {\n const { isFloatingLabel, isHiddenLabel, formElementId } =\n useFormFieldContext()\n\n return (\n <StyledFloatingLabel\n htmlFor={formElementId}\n floating={isFloatingLabel}\n visuallyHidden={isHiddenLabel}\n {...props}\n ref={forwardRef}\n />\n )\n})\n"],"names":["ariaDescribedBy"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"module.js","sources":["../src/constants.ts","../src/hooks/use-form-field-context.tsx"],"sourcesContent":["export const FORM_FIELD_STATUS = Object.freeze({\n // the field value is valid\n valid: 'valid',\n // the field value is invalid\n invalid: 'invalid',\n // the field value has been changed\n dirty: 'dirty',\n // the field value has not been changed\n pristine: 'pristine',\n // the field has been focused and blurred\n touched: 'touched',\n})\n\nexport type FormFieldStatus = keyof typeof FORM_FIELD_STATUS\n","import React, {\n createContext,\n useContext,\n useState,\n useRef,\n useEffect,\n} from 'react'\nimport type { AriaAttributes } from 'react'\nimport type { Booleanish } from '@mirohq/design-system-types'\nimport { useId } from '@mirohq/design-system-use-id'\nimport { stringAttrValue } from '@mirohq/design-system-utils'\n\nimport type { FormFieldStatus } from '../constants'\nimport { FORM_FIELD_STATUS } from '../constants'\n\ntype FormElements = HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement\n\ninterface FormFieldProps {\n /**\n * Field status\n * @default \"pristine\"\n */\n status?: FormFieldStatus\n\n id?: string\n}\n\ninterface FormFieldContextProps<FormElementRef extends FormElements>\n extends FormFieldProps {\n setStatus: React.Dispatch<React.SetStateAction<FormFieldStatus>>\n isHiddenLabel: boolean\n setIsHiddenLabel: React.Dispatch<React.SetStateAction<boolean>>\n label: React.ReactNode\n setLabel: React.Dispatch<React.SetStateAction<React.ReactNode>>\n focused: boolean\n setFocused: React.Dispatch<React.SetStateAction<boolean>>\n helperId: string\n setHelperId: React.Dispatch<React.SetStateAction<string>>\n messageId: string\n setMessageId: React.Dispatch<React.SetStateAction<string>>\n descriptionId: string\n setDescriptionId: React.Dispatch<React.SetStateAction<string>>\n formElementId: string\n formElementRef: React.MutableRefObject<FormElementRef>\n ariaDescribedBy: string | undefined\n required: boolean | undefined\n setRequired: React.Dispatch<React.SetStateAction<boolean | undefined>>\n readOnly: boolean | undefined\n setReadOnly: React.Dispatch<React.SetStateAction<boolean | undefined>>\n disabled: boolean | undefined\n setDisabled: React.Dispatch<React.SetStateAction<boolean | undefined>>\n ariaDisabled: Booleanish | undefined\n setAriaDisabled: React.Dispatch<React.SetStateAction<Booleanish | undefined>>\n ariaInvalid: AriaAttributes['aria-invalid']\n valid: boolean | undefined\n}\n\nexport interface FormFieldProviderProps extends FormFieldProps {\n children?: React.ReactNode\n}\n\nconst FormFieldContext = createContext<FormFieldContextProps<any>>({} as any)\n\nexport const FormFieldProvider = ({\n children,\n status: customStatus,\n id: customId,\n ...restProps\n}: FormFieldProviderProps): JSX.Element => {\n // form field state\n const [status, setStatus] = useState<FormFieldStatus>(\n customStatus ?? 'pristine'\n )\n const [focused, setFocused] = useState(false)\n const [isHiddenLabel, setIsHiddenLabel] = useState(false)\n const [label, setLabel] = useState<React.ReactNode>(null)\n\n // partials\n const autoId = useId()\n const id = customId ?? autoId\n\n const [messageId, setMessageId] = useState(`${id}-message`)\n const [helperId, setHelperId] = useState(`${id}-helper`)\n const [descriptionId, setDescriptionId] = useState(`${id}-description`)\n\n // form element\n const formElementRef = useRef<HTMLInputElement>(null)\n const [required, setRequired] = useState<boolean>()\n const [readOnly, setReadOnly] = useState<boolean>()\n const [disabled, setDisabled] = useState<boolean>()\n const [ariaDisabled, setAriaDisabled] = useState<Booleanish>()\n const [ariaDescribedBy, setAriaDescribedBy] = useState<string | undefined>('')\n\n const valid =\n status === FORM_FIELD_STATUS.valid\n ? true\n : status === FORM_FIELD_STATUS.invalid\n ? false\n : undefined\n\n const ariaInvalid: AriaAttributes['aria-invalid'] =\n status === FORM_FIELD_STATUS.invalid ? true : undefined\n\n useEffect(() => {\n const message = document.getElementById(messageId)\n const helper = document.getElementById(helperId)\n const description = document.getElementById(descriptionId)\n\n const ariaDescribedBy = stringAttrValue(\n // message or description\n message != null ? messageId : description != null && descriptionId,\n helper != null && helperId\n )\n\n /**\n * message + helper text\n * description + helper text\n */\n setAriaDescribedBy(ariaDescribedBy === '' ? undefined : ariaDescribedBy)\n }, [messageId, helperId, descriptionId])\n\n return (\n <FormFieldContext.Provider\n value={{\n ...restProps,\n id,\n formElementRef,\n formElementId: `${id}-form-element`,\n messageId,\n setMessageId,\n helperId,\n setHelperId,\n descriptionId,\n setDescriptionId,\n status,\n setStatus,\n focused,\n setFocused,\n isHiddenLabel,\n setIsHiddenLabel,\n label,\n setLabel,\n ariaDescribedBy,\n required,\n setRequired,\n readOnly,\n setReadOnly,\n disabled,\n setDisabled,\n ariaDisabled,\n setAriaDisabled,\n ariaInvalid,\n valid,\n }}\n >\n {children}\n </FormFieldContext.Provider>\n )\n}\n\nexport const useFormFieldContext = <\n T extends FormElements\n>(): FormFieldContextProps<T> => useContext(FormFieldContext)\n"],"names":["ariaDescribedBy"],"mappings":"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"}
|
package/dist/types.d.ts
CHANGED
|
@@ -1,44 +1,6 @@
|
|
|
1
|
-
import
|
|
2
|
-
import React__default, { ComponentPropsWithRef, AriaAttributes } from 'react';
|
|
3
|
-
import * as _mirohq_design_system_stitches from '@mirohq/design-system-stitches';
|
|
4
|
-
import * as _mirohq_design_system_primitive from '@mirohq/design-system-primitive';
|
|
1
|
+
import React, { AriaAttributes } from 'react';
|
|
5
2
|
import { Booleanish } from '@mirohq/design-system-types';
|
|
6
3
|
|
|
7
|
-
/* Utilities */
|
|
8
|
-
/* ========================================================================== */
|
|
9
|
-
|
|
10
|
-
/** Returns a string with the given prefix followed by the given values. */
|
|
11
|
-
type Prefixed<K extends string, T> = `${K}${Extract<T, boolean | number | string>}`
|
|
12
|
-
|
|
13
|
-
type TransformProps<Props, Media> = {
|
|
14
|
-
[K in keyof Props]: (
|
|
15
|
-
| Props[K]
|
|
16
|
-
| (
|
|
17
|
-
& {
|
|
18
|
-
[KMedia in Prefixed<'@', 'initial' | keyof Media>]?: Props[K]
|
|
19
|
-
}
|
|
20
|
-
& {
|
|
21
|
-
[KMedia in string]: Props[K]
|
|
22
|
-
}
|
|
23
|
-
)
|
|
24
|
-
)
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
declare const StyledFloatingLabel: React.ForwardRefExoticComponent<Omit<Omit<_mirohq_design_system_stitches.StyledComponentProps<React.ForwardRefExoticComponent<_mirohq_design_system_primitive.PrimitiveProps<"label">>>, "size" | "floating" | "visuallyHidden"> & TransformProps<{
|
|
28
|
-
floating?: boolean | "true" | "false" | undefined;
|
|
29
|
-
visuallyHidden?: boolean | "true" | undefined;
|
|
30
|
-
size?: "medium" | "large" | "x-large" | undefined;
|
|
31
|
-
}, {}> & _mirohq_design_system_stitches.CustomStylesProps, "ref"> & React.RefAttributes<HTMLLabelElement>> & _mirohq_design_system_stitches.StitchesInternals<React.ForwardRefExoticComponent<_mirohq_design_system_primitive.PrimitiveProps<"label">>, {
|
|
32
|
-
floating?: boolean | "true" | "false" | undefined;
|
|
33
|
-
visuallyHidden?: boolean | "true" | undefined;
|
|
34
|
-
size?: "medium" | "large" | "x-large" | undefined;
|
|
35
|
-
}, {}>;
|
|
36
|
-
type StyledFloatingLabelProps = ComponentPropsWithRef<typeof StyledFloatingLabel>;
|
|
37
|
-
|
|
38
|
-
interface FloatingLabelProps extends Omit<StyledFloatingLabelProps, 'visuallyHidden'> {
|
|
39
|
-
}
|
|
40
|
-
declare const FloatingLabel: React__default.ForwardRefExoticComponent<Omit<FloatingLabelProps, "ref"> & React__default.RefAttributes<HTMLLabelElement>>;
|
|
41
|
-
|
|
42
4
|
declare const FORM_FIELD_STATUS: Readonly<{
|
|
43
5
|
valid: "valid";
|
|
44
6
|
invalid: "invalid";
|
|
@@ -58,38 +20,35 @@ interface FormFieldProps {
|
|
|
58
20
|
id?: string;
|
|
59
21
|
}
|
|
60
22
|
interface FormFieldContextProps<FormElementRef extends FormElements> extends FormFieldProps {
|
|
61
|
-
setStatus:
|
|
62
|
-
isFloatingLabel: boolean;
|
|
63
|
-
setIsFloatingLabel: React__default.Dispatch<React__default.SetStateAction<boolean>>;
|
|
23
|
+
setStatus: React.Dispatch<React.SetStateAction<FormFieldStatus>>;
|
|
64
24
|
isHiddenLabel: boolean;
|
|
65
|
-
setIsHiddenLabel:
|
|
66
|
-
label:
|
|
67
|
-
setLabel:
|
|
25
|
+
setIsHiddenLabel: React.Dispatch<React.SetStateAction<boolean>>;
|
|
26
|
+
label: React.ReactNode;
|
|
27
|
+
setLabel: React.Dispatch<React.SetStateAction<React.ReactNode>>;
|
|
68
28
|
focused: boolean;
|
|
69
|
-
setFocused:
|
|
29
|
+
setFocused: React.Dispatch<React.SetStateAction<boolean>>;
|
|
70
30
|
helperId: string;
|
|
71
|
-
setHelperId:
|
|
31
|
+
setHelperId: React.Dispatch<React.SetStateAction<string>>;
|
|
72
32
|
messageId: string;
|
|
73
|
-
setMessageId:
|
|
33
|
+
setMessageId: React.Dispatch<React.SetStateAction<string>>;
|
|
74
34
|
descriptionId: string;
|
|
75
|
-
setDescriptionId:
|
|
35
|
+
setDescriptionId: React.Dispatch<React.SetStateAction<string>>;
|
|
76
36
|
formElementId: string;
|
|
77
|
-
formElementRef:
|
|
37
|
+
formElementRef: React.MutableRefObject<FormElementRef>;
|
|
78
38
|
ariaDescribedBy: string | undefined;
|
|
79
|
-
shouldUseFloatingLabel: boolean;
|
|
80
39
|
required: boolean | undefined;
|
|
81
|
-
setRequired:
|
|
40
|
+
setRequired: React.Dispatch<React.SetStateAction<boolean | undefined>>;
|
|
82
41
|
readOnly: boolean | undefined;
|
|
83
|
-
setReadOnly:
|
|
42
|
+
setReadOnly: React.Dispatch<React.SetStateAction<boolean | undefined>>;
|
|
84
43
|
disabled: boolean | undefined;
|
|
85
|
-
setDisabled:
|
|
44
|
+
setDisabled: React.Dispatch<React.SetStateAction<boolean | undefined>>;
|
|
86
45
|
ariaDisabled: Booleanish | undefined;
|
|
87
|
-
setAriaDisabled:
|
|
46
|
+
setAriaDisabled: React.Dispatch<React.SetStateAction<Booleanish | undefined>>;
|
|
88
47
|
ariaInvalid: AriaAttributes['aria-invalid'];
|
|
89
48
|
valid: boolean | undefined;
|
|
90
49
|
}
|
|
91
50
|
interface FormFieldProviderProps extends FormFieldProps {
|
|
92
|
-
children?:
|
|
51
|
+
children?: React.ReactNode;
|
|
93
52
|
}
|
|
94
53
|
declare const FormFieldProvider: ({ children, status: customStatus, id: customId, ...restProps }: FormFieldProviderProps) => JSX.Element;
|
|
95
54
|
declare const useFormFieldContext: <T extends FormElements>() => FormFieldContextProps<T>;
|
|
@@ -118,5 +77,5 @@ interface FormElementProps {
|
|
|
118
77
|
required?: boolean;
|
|
119
78
|
}
|
|
120
79
|
|
|
121
|
-
export { FORM_FIELD_STATUS,
|
|
122
|
-
export type {
|
|
80
|
+
export { FORM_FIELD_STATUS, FormFieldProvider, useFormFieldContext };
|
|
81
|
+
export type { FormElementProps, FormFieldProviderProps, FormFieldStatus };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@mirohq/design-system-base-form",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "2.0.0",
|
|
4
4
|
"description": "",
|
|
5
5
|
"author": "Miro",
|
|
6
6
|
"source": "src/index.ts",
|
|
@@ -26,12 +26,9 @@
|
|
|
26
26
|
"react": "^16.14 || ^17 || ^18 || ^19"
|
|
27
27
|
},
|
|
28
28
|
"dependencies": {
|
|
29
|
-
"@mirohq/design-system-
|
|
30
|
-
"@mirohq/design-system-primitive": "^2.2.1",
|
|
31
|
-
"@mirohq/design-system-stitches": "^3.3.1",
|
|
29
|
+
"@mirohq/design-system-types": "^1.0.3",
|
|
32
30
|
"@mirohq/design-system-use-id": "^1.1.1",
|
|
33
|
-
"@mirohq/design-system-utils": "^1.
|
|
34
|
-
"@mirohq/design-system-types": "^1.0.3"
|
|
31
|
+
"@mirohq/design-system-utils": "^1.3.0"
|
|
35
32
|
},
|
|
36
33
|
"scripts": {
|
|
37
34
|
"build": "rollup -c ../../../../rollup.config.js",
|