@transferwise/components 46.52.2 → 46.53.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/build/card/Card.js +20 -13
- package/build/card/Card.js.map +1 -1
- package/build/card/Card.mjs +21 -14
- package/build/card/Card.mjs.map +1 -1
- package/build/common/Option/Option.js.map +1 -1
- package/build/common/Option/Option.mjs.map +1 -1
- package/build/dateLookup/dateTrigger/DateTrigger.js +8 -4
- package/build/dateLookup/dateTrigger/DateTrigger.js.map +1 -1
- package/build/dateLookup/dateTrigger/DateTrigger.mjs +8 -4
- package/build/dateLookup/dateTrigger/DateTrigger.mjs.map +1 -1
- package/build/field/Field.js +36 -8
- package/build/field/Field.js.map +1 -1
- package/build/field/Field.mjs +37 -9
- package/build/field/Field.mjs.map +1 -1
- package/build/i18n/en.json +1 -0
- package/build/i18n/en.json.js +1 -0
- package/build/i18n/en.json.js.map +1 -1
- package/build/i18n/en.json.mjs +1 -0
- package/build/i18n/en.json.mjs.map +1 -1
- package/build/index.js +2 -2
- package/build/index.mjs +1 -1
- package/build/inlineAlert/InlineAlert.js +13 -6
- package/build/inlineAlert/InlineAlert.js.map +1 -1
- package/build/inlineAlert/InlineAlert.mjs +13 -6
- package/build/inlineAlert/InlineAlert.mjs.map +1 -1
- package/build/label/Label.js +35 -4
- package/build/label/Label.js.map +1 -1
- package/build/label/Label.messages.js +12 -0
- package/build/label/Label.messages.js.map +1 -0
- package/build/label/Label.messages.mjs +10 -0
- package/build/label/Label.messages.mjs.map +1 -0
- package/build/label/Label.mjs +36 -5
- package/build/label/Label.mjs.map +1 -1
- package/build/main.css +4 -8
- package/build/styles/dateLookup/dateTrigger/DateTrigger.css +0 -8
- package/build/styles/field/Field.css +4 -0
- package/build/styles/main.css +4 -8
- package/build/tabs/Tab.js +13 -38
- package/build/tabs/Tab.js.map +1 -1
- package/build/tabs/Tab.mjs +13 -34
- package/build/tabs/Tab.mjs.map +1 -1
- package/build/tabs/TabList.js +3 -11
- package/build/tabs/TabList.js.map +1 -1
- package/build/tabs/TabList.mjs +3 -7
- package/build/tabs/TabList.mjs.map +1 -1
- package/build/tabs/TabPanel.js +3 -16
- package/build/tabs/TabPanel.js.map +1 -1
- package/build/tabs/TabPanel.mjs +3 -12
- package/build/tabs/TabPanel.mjs.map +1 -1
- package/build/tabs/Tabs.js +24 -48
- package/build/tabs/Tabs.js.map +1 -1
- package/build/tabs/Tabs.mjs +24 -47
- package/build/tabs/Tabs.mjs.map +1 -1
- package/build/tabs/utils.js +0 -1
- package/build/tabs/utils.js.map +1 -1
- package/build/tabs/utils.mjs +0 -1
- package/build/tabs/utils.mjs.map +1 -1
- package/build/types/card/Card.d.ts.map +1 -1
- package/build/types/common/Option/Option.d.ts +2 -0
- package/build/types/common/Option/Option.d.ts.map +1 -1
- package/build/types/dateLookup/dateTrigger/DateTrigger.d.ts.map +1 -1
- package/build/types/field/Field.d.ts +4 -2
- package/build/types/field/Field.d.ts.map +1 -1
- package/build/types/index.d.ts +2 -1
- package/build/types/index.d.ts.map +1 -1
- package/build/types/inlineAlert/InlineAlert.d.ts +9 -0
- package/build/types/inlineAlert/InlineAlert.d.ts.map +1 -1
- package/build/types/label/Label.d.ts +21 -1
- package/build/types/label/Label.d.ts.map +1 -1
- package/build/types/label/Label.messages.d.ts +8 -0
- package/build/types/label/Label.messages.d.ts.map +1 -0
- package/build/types/label/index.d.ts +3 -0
- package/build/types/label/index.d.ts.map +1 -0
- package/build/types/tabs/Tab.d.ts +12 -1
- package/build/types/tabs/Tab.d.ts.map +1 -1
- package/build/types/tabs/TabList.d.ts +3 -8
- package/build/types/tabs/TabList.d.ts.map +1 -1
- package/build/types/tabs/TabPanel.d.ts +6 -14
- package/build/types/tabs/TabPanel.d.ts.map +1 -1
- package/build/types/tabs/Tabs.d.ts +83 -30
- package/build/types/tabs/Tabs.d.ts.map +1 -1
- package/build/types/tabs/index.d.ts +2 -1
- package/build/types/tabs/index.d.ts.map +1 -1
- package/build/types/tabs/utils.d.ts +12 -7
- package/build/types/tabs/utils.d.ts.map +1 -1
- package/package.json +3 -3
- package/src/card/Card.spec.tsx +35 -3
- package/src/card/Card.story.tsx +56 -40
- package/src/card/Card.tsx +32 -14
- package/src/common/Option/Option.tsx +2 -0
- package/src/dateInput/DateInput.tests.story.tsx +6 -42
- package/src/dateLookup/DateLookup.rtl.spec.tsx +1 -1
- package/src/dateLookup/dateTrigger/DateTrigger.css +0 -8
- package/src/dateLookup/dateTrigger/DateTrigger.less +0 -8
- package/src/dateLookup/dateTrigger/DateTrigger.spec.js +1 -1
- package/src/dateLookup/dateTrigger/DateTrigger.tsx +9 -4
- package/src/field/Field.css +4 -0
- package/src/field/Field.less +5 -0
- package/src/field/Field.spec.tsx +41 -5
- package/src/field/Field.story.tsx +105 -7
- package/src/field/Field.tsx +34 -10
- package/src/i18n/en.json +1 -0
- package/src/index.ts +2 -1
- package/src/inlineAlert/InlineAlert.story.tsx +7 -72
- package/src/inlineAlert/InlineAlert.tsx +14 -3
- package/src/inputWithDisplayFormat/InputWithDisplayFormat.story.js +5 -10
- package/src/inputs/InputGroup.spec.tsx +1 -1
- package/src/inputs/SearchInput.spec.tsx +1 -1
- package/src/inputs/SelectInput.spec.tsx +1 -1
- package/src/label/Label.messages.tsx +8 -0
- package/src/label/Label.spec.tsx +53 -4
- package/src/label/Label.story.tsx +32 -26
- package/src/label/Label.tsx +47 -2
- package/src/label/index.ts +2 -0
- package/src/main.css +4 -8
- package/src/main.less +1 -0
- package/src/moneyInput/MoneyInput.story.tsx +11 -11
- package/src/radioGroup/RadioGroup.rtl.spec.tsx +1 -1
- package/src/select/Select.rtl.spec.tsx +1 -1
- package/src/switch/Switch.spec.tsx +1 -1
- package/src/switch/Switch.story.tsx +19 -21
- package/src/tabs/Tab.tsx +72 -0
- package/src/tabs/TabList.tsx +11 -0
- package/src/tabs/TabPanel.tsx +14 -0
- package/src/tabs/{Tabs.story.js → Tabs.story.tsx} +1 -1
- package/src/tabs/{Tabs.js → Tabs.tsx} +111 -74
- package/src/tabs/index.ts +2 -0
- package/src/tabs/{utils.spec.js → utils.spec.ts} +24 -21
- package/src/tabs/{utils.js → utils.ts} +15 -9
- package/src/field/Field.tests.story.tsx +0 -33
- package/src/tabs/Tab.js +0 -71
- package/src/tabs/TabList.js +0 -15
- package/src/tabs/TabPanel.js +0 -20
- package/src/tabs/index.js +0 -1
package/src/field/Field.tsx
CHANGED
|
@@ -9,16 +9,18 @@ import {
|
|
|
9
9
|
InputIdContextProvider,
|
|
10
10
|
InputInvalidProvider,
|
|
11
11
|
} from '../inputs/contexts';
|
|
12
|
-
import { Label } from '../label
|
|
12
|
+
import { Label } from '../label';
|
|
13
13
|
|
|
14
14
|
export type FieldProps = {
|
|
15
15
|
/** `null` disables auto-generating the `id` attribute, falling back to nesting-based label association over setting `htmlFor` explicitly. */
|
|
16
16
|
id?: string | null;
|
|
17
17
|
/** Should be specified unless the wrapped control has its own labeling mechanism, e.g. `Checkbox`. */
|
|
18
18
|
label?: React.ReactNode;
|
|
19
|
-
|
|
19
|
+
required?: boolean;
|
|
20
|
+
/** @deprecated use `description` prop instead */
|
|
20
21
|
hint?: React.ReactNode;
|
|
21
22
|
message?: React.ReactNode;
|
|
23
|
+
description?: React.ReactNode;
|
|
22
24
|
/** @deprecated use `message` and `type={Sentiment.NEGATIVE}` prop instead */
|
|
23
25
|
error?: React.ReactNode;
|
|
24
26
|
sentiment?: `${Sentiment.NEGATIVE | Sentiment.NEUTRAL | Sentiment.POSITIVE | Sentiment.WARNING}`;
|
|
@@ -29,14 +31,17 @@ export type FieldProps = {
|
|
|
29
31
|
export const Field = ({
|
|
30
32
|
id,
|
|
31
33
|
label,
|
|
34
|
+
required = false,
|
|
32
35
|
message: propMessage,
|
|
36
|
+
hint,
|
|
37
|
+
description = hint,
|
|
33
38
|
sentiment: propType = Sentiment.NEUTRAL,
|
|
34
39
|
className,
|
|
35
40
|
children,
|
|
36
41
|
...props
|
|
37
42
|
}: FieldProps) => {
|
|
38
43
|
const sentiment = props.error ? Sentiment.NEGATIVE : propType;
|
|
39
|
-
const message =
|
|
44
|
+
const message = propMessage || props.error;
|
|
40
45
|
const hasError = sentiment === Sentiment.NEGATIVE;
|
|
41
46
|
|
|
42
47
|
const labelId = useId();
|
|
@@ -44,16 +49,32 @@ export const Field = ({
|
|
|
44
49
|
const fallbackInputId = useId();
|
|
45
50
|
const inputId = id !== null ? (id ?? fallbackInputId) : undefined;
|
|
46
51
|
|
|
52
|
+
const messageId = useId();
|
|
47
53
|
const descriptionId = useId();
|
|
48
54
|
|
|
55
|
+
/**
|
|
56
|
+
* form control can have multiple messages to describe it,
|
|
57
|
+
* e.g the description underneath the label and inline alert
|
|
58
|
+
*/
|
|
59
|
+
function ariaDescribedbyByIds() {
|
|
60
|
+
const messageIds = [];
|
|
61
|
+
if (description) {
|
|
62
|
+
messageIds.push(descriptionId);
|
|
63
|
+
}
|
|
64
|
+
if (message) {
|
|
65
|
+
messageIds.push(messageId);
|
|
66
|
+
}
|
|
67
|
+
return messageIds.length > 0 ? messageIds.join(' ') : undefined;
|
|
68
|
+
}
|
|
69
|
+
|
|
49
70
|
return (
|
|
50
71
|
<FieldLabelIdContextProvider value={labelId}>
|
|
51
72
|
<InputIdContextProvider value={inputId}>
|
|
52
|
-
<InputDescribedByProvider value={
|
|
73
|
+
<InputDescribedByProvider value={ariaDescribedbyByIds()}>
|
|
53
74
|
<InputInvalidProvider value={hasError}>
|
|
54
75
|
<div
|
|
55
76
|
className={clsx(
|
|
56
|
-
'form-group d-block',
|
|
77
|
+
'np-field form-group d-block',
|
|
57
78
|
{
|
|
58
79
|
'has-success': sentiment === Sentiment.POSITIVE,
|
|
59
80
|
'has-warning': sentiment === Sentiment.WARNING,
|
|
@@ -64,16 +85,19 @@ export const Field = ({
|
|
|
64
85
|
)}
|
|
65
86
|
>
|
|
66
87
|
{label != null ? (
|
|
67
|
-
|
|
68
|
-
{
|
|
69
|
-
|
|
70
|
-
|
|
88
|
+
<>
|
|
89
|
+
<Label id={labelId} htmlFor={inputId}>
|
|
90
|
+
{required ? label : <Label.Optional>{label}</Label.Optional>}
|
|
91
|
+
</Label>
|
|
92
|
+
<Label.Description id={descriptionId}>{description}</Label.Description>
|
|
93
|
+
<div className="np-field-control">{children}</div>
|
|
94
|
+
</>
|
|
71
95
|
) : (
|
|
72
96
|
children
|
|
73
97
|
)}
|
|
74
98
|
|
|
75
99
|
{message && (
|
|
76
|
-
<InlineAlert type={sentiment} id={
|
|
100
|
+
<InlineAlert type={sentiment} id={messageId}>
|
|
77
101
|
{message}
|
|
78
102
|
</InlineAlert>
|
|
79
103
|
)}
|
package/src/i18n/en.json
CHANGED
|
@@ -18,6 +18,7 @@
|
|
|
18
18
|
"neptune.DateLookup.year": "year",
|
|
19
19
|
"neptune.FlowNavigation.back": "back to previous step",
|
|
20
20
|
"neptune.Info.ariaLabel": "More information",
|
|
21
|
+
"neptune.Label.optional": "(Optional)",
|
|
21
22
|
"neptune.Link.opensInNewTab": "(opens in new tab)",
|
|
22
23
|
"neptune.MoneyInput.Select.placeholder": "Select an option...",
|
|
23
24
|
"neptune.PhoneNumberInput.SelectInput.placeholder": "Select an option...",
|
package/src/index.ts
CHANGED
|
@@ -44,7 +44,7 @@ export type {
|
|
|
44
44
|
} from './inputs/SelectInput';
|
|
45
45
|
export type { TextAreaProps } from './inputs/TextArea';
|
|
46
46
|
export type { InstructionsListProps } from './instructionsList';
|
|
47
|
-
export type { LabelProps } from './label/Label';
|
|
47
|
+
export type { LabelProps, LabelOptionalProps, LabelDescriptionProps } from './label/Label';
|
|
48
48
|
export type { LoaderProps } from './loader';
|
|
49
49
|
export type { MarkdownProps } from './markdown';
|
|
50
50
|
export type { ModalProps } from './modal';
|
|
@@ -75,6 +75,7 @@ export type { StickyProps } from './sticky';
|
|
|
75
75
|
export type { SummaryProps } from './summary';
|
|
76
76
|
export type { SwitchProps } from './switch';
|
|
77
77
|
export type { SwitchOptionProps } from './switchOption';
|
|
78
|
+
export type { TabItem, TabsProps } from './tabs';
|
|
78
79
|
export type { TextareaWithDisplayFormatProps } from './textareaWithDisplayFormat';
|
|
79
80
|
export type { TooltipProps } from './tooltip';
|
|
80
81
|
export type { TypeaheadOption, TypeaheadProps } from './typeahead';
|
|
@@ -2,9 +2,9 @@ import { select, text } from '@storybook/addon-knobs';
|
|
|
2
2
|
import { Meta } from '@storybook/react';
|
|
3
3
|
|
|
4
4
|
import { Sentiment } from '../common';
|
|
5
|
-
import { Input } from '../inputs/Input';
|
|
6
5
|
|
|
7
6
|
import InlineAlert, { InlineAlertProps } from './InlineAlert';
|
|
7
|
+
import { lorem40 } from '../test-utils';
|
|
8
8
|
|
|
9
9
|
export default {
|
|
10
10
|
component: InlineAlert,
|
|
@@ -29,81 +29,16 @@ export const Basic = () => {
|
|
|
29
29
|
|
|
30
30
|
const message = text('message', 'Please enter a password over 5 characters');
|
|
31
31
|
|
|
32
|
-
let typeClass = '';
|
|
33
|
-
switch (type) {
|
|
34
|
-
case Sentiment.ERROR:
|
|
35
|
-
case Sentiment.NEGATIVE:
|
|
36
|
-
typeClass = 'has-error';
|
|
37
|
-
break;
|
|
38
|
-
case Sentiment.SUCCESS:
|
|
39
|
-
case Sentiment.POSITIVE:
|
|
40
|
-
typeClass = 'has-success';
|
|
41
|
-
break;
|
|
42
|
-
case Sentiment.INFO:
|
|
43
|
-
case Sentiment.NEUTRAL:
|
|
44
|
-
typeClass = 'has-info';
|
|
45
|
-
break;
|
|
46
|
-
case Sentiment.WARNING:
|
|
47
|
-
typeClass = 'has-warning';
|
|
48
|
-
break;
|
|
49
|
-
case Sentiment.PENDING:
|
|
50
|
-
}
|
|
51
|
-
|
|
52
32
|
return (
|
|
53
33
|
<>
|
|
54
|
-
{/* eslint-disable-next-line react/no-adjacent-inline-elements */}
|
|
55
|
-
<p>
|
|
56
|
-
The styling for the input (the coloured border) and the visibility of the inline alert is
|
|
57
|
-
controlled through the use of <code>has-***</code> classes which are applied to the{' '}
|
|
58
|
-
<code>form-group</code> element. For example, to display an inline alert of type error, you
|
|
59
|
-
must also apply the class <code>has-error</code> to the parent <code>form-group</code>{' '}
|
|
60
|
-
element. The available classes are <code>has-error</code>, <code>has-info</code>,{' '}
|
|
61
|
-
<code>has-warning</code> and <code>has-success</code>.
|
|
62
|
-
</p>
|
|
63
34
|
<p>
|
|
64
|
-
|
|
65
|
-
<
|
|
66
|
-
|
|
35
|
+
Avoid using <code>InlineAlert</code> directly (unless for some custom use cases), for form
|
|
36
|
+
control validation, info messaging please use <code>Field</code> component
|
|
37
|
+
<pre>{'<Field sentiment={..} message={..}>'}</pre>
|
|
67
38
|
</p>
|
|
68
|
-
<
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
</label>
|
|
72
|
-
<Input id="id0" value="Neptune is cool" />
|
|
73
|
-
<InlineAlert type={type}>{message}</InlineAlert>
|
|
74
|
-
</div>
|
|
75
|
-
<div className="form-group has-error">
|
|
76
|
-
<label className="control-label" htmlFor="id1">
|
|
77
|
-
Negative
|
|
78
|
-
</label>
|
|
79
|
-
<Input id="id1" value="Neptune is cool" />
|
|
80
|
-
<InlineAlert type="negative">{message}</InlineAlert>
|
|
81
|
-
</div>
|
|
82
|
-
<div className="form-group has-success">
|
|
83
|
-
<label className="control-label" htmlFor="id2">
|
|
84
|
-
Positive
|
|
85
|
-
</label>
|
|
86
|
-
<Input id="id2" value="Neptune is cool" />
|
|
87
|
-
<InlineAlert type="positive">
|
|
88
|
-
Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt
|
|
89
|
-
ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation
|
|
90
|
-
ullamco laboris nisi ut aliquip ex ea commodo consequat.
|
|
91
|
-
</InlineAlert>
|
|
92
|
-
</div>
|
|
93
|
-
<div className="form-group has-neutral">
|
|
94
|
-
<label className="control-label" htmlFor="id3">
|
|
95
|
-
Neutral
|
|
96
|
-
</label>
|
|
97
|
-
<Input id="id3" value="Neptune is cool" />
|
|
98
|
-
<InlineAlert type="neutral">{message}</InlineAlert>
|
|
99
|
-
</div>
|
|
100
|
-
<div className="form-group">
|
|
101
|
-
<label className="control-label" htmlFor="id4">
|
|
102
|
-
No has-* class
|
|
103
|
-
</label>
|
|
104
|
-
<Input id="id4" value="Neptune is cool" />
|
|
105
|
-
<InlineAlert type="negative">{message}</InlineAlert>
|
|
106
|
-
</div>
|
|
39
|
+
<InlineAlert type={type}>{message}</InlineAlert>
|
|
40
|
+
<InlineAlert type="negative">{message}</InlineAlert>
|
|
41
|
+
<InlineAlert type="positive">{lorem40}</InlineAlert>
|
|
107
42
|
</>
|
|
108
43
|
);
|
|
109
44
|
};
|
|
@@ -3,6 +3,7 @@ import { ReactNode } from 'react';
|
|
|
3
3
|
|
|
4
4
|
import { Sentiment, Size } from '../common';
|
|
5
5
|
import StatusIcon from '../statusIcon';
|
|
6
|
+
import Body from '../body';
|
|
6
7
|
|
|
7
8
|
export interface InlineAlertProps {
|
|
8
9
|
id?: string;
|
|
@@ -19,6 +20,15 @@ const iconTypes = new Set<NonNullable<InlineAlertProps['type']>>([
|
|
|
19
20
|
Sentiment.WARNING,
|
|
20
21
|
]);
|
|
21
22
|
|
|
23
|
+
/**
|
|
24
|
+
* Avoid using `<InlineAlert>` component directly
|
|
25
|
+
* it's for edge cases when `<Field />` isn't suitable for some reasons.
|
|
26
|
+
*
|
|
27
|
+
* Example:
|
|
28
|
+
* ```
|
|
29
|
+
* <Field sentiment={..} message={..}>..</Field>
|
|
30
|
+
* ```
|
|
31
|
+
*/
|
|
22
32
|
export default function InlineAlert({
|
|
23
33
|
id,
|
|
24
34
|
type = 'neutral',
|
|
@@ -26,17 +36,18 @@ export default function InlineAlert({
|
|
|
26
36
|
children,
|
|
27
37
|
}: InlineAlertProps) {
|
|
28
38
|
return (
|
|
29
|
-
<
|
|
39
|
+
<Body
|
|
30
40
|
role="alert"
|
|
31
41
|
id={id}
|
|
32
42
|
className={clsx(
|
|
33
43
|
'alert alert-detach',
|
|
34
44
|
`alert-${type === Sentiment.NEGATIVE || type === Sentiment.ERROR ? 'danger' : type}`,
|
|
45
|
+
'd-flex',
|
|
35
46
|
className,
|
|
36
47
|
)}
|
|
37
48
|
>
|
|
38
49
|
{iconTypes.has(type) && <StatusIcon sentiment={type} size={Size.SMALL} />}
|
|
39
|
-
|
|
40
|
-
</
|
|
50
|
+
{children}
|
|
51
|
+
</Body>
|
|
41
52
|
);
|
|
42
53
|
}
|
|
@@ -2,6 +2,7 @@ import { text } from '@storybook/addon-knobs';
|
|
|
2
2
|
import { userEvent, within } from '@storybook/test';
|
|
3
3
|
|
|
4
4
|
import InputWithDisplayFormat from '.';
|
|
5
|
+
import { Field } from '../field/Field';
|
|
5
6
|
|
|
6
7
|
export default {
|
|
7
8
|
component: InputWithDisplayFormat,
|
|
@@ -13,10 +14,7 @@ export const Basic = () => {
|
|
|
13
14
|
const displayPattern = text('DisplayPattern', '**-**-**');
|
|
14
15
|
|
|
15
16
|
return (
|
|
16
|
-
|
|
17
|
-
<label id="template" htmlFor="Basic">
|
|
18
|
-
Display pattern is controlled via knobs
|
|
19
|
-
</label>
|
|
17
|
+
<Field label="Display pattern is controlled via knobs" id="Basic">
|
|
20
18
|
<InputWithDisplayFormat
|
|
21
19
|
id="Basic"
|
|
22
20
|
placeholder={placeholder}
|
|
@@ -26,7 +24,7 @@ export const Basic = () => {
|
|
|
26
24
|
onBlur={(v) => console.log(v)}
|
|
27
25
|
onFocus={(v) => console.log(v)}
|
|
28
26
|
/>
|
|
29
|
-
|
|
27
|
+
</Field>
|
|
30
28
|
);
|
|
31
29
|
};
|
|
32
30
|
|
|
@@ -37,10 +35,7 @@ const Template = (args) => {
|
|
|
37
35
|
const id = label.replaceAll(' ', '-').toLowerCase();
|
|
38
36
|
|
|
39
37
|
return (
|
|
40
|
-
|
|
41
|
-
<label id="template" htmlFor={id}>
|
|
42
|
-
{label}
|
|
43
|
-
</label>
|
|
38
|
+
<Field label={label} id={id}>
|
|
44
39
|
<InputWithDisplayFormat
|
|
45
40
|
id={id}
|
|
46
41
|
placeholder={placeholder}
|
|
@@ -50,7 +45,7 @@ const Template = (args) => {
|
|
|
50
45
|
onBlur={(v) => console.log(v)}
|
|
51
46
|
onFocus={(v) => console.log(v)}
|
|
52
47
|
/>
|
|
53
|
-
|
|
48
|
+
</Field>
|
|
54
49
|
);
|
|
55
50
|
};
|
|
56
51
|
|
|
@@ -214,6 +214,6 @@ describe('SelectInput', () => {
|
|
|
214
214
|
<SelectInput items={[{ type: 'option', value: 'USD' }]} value="USD" />
|
|
215
215
|
</Field>,
|
|
216
216
|
);
|
|
217
|
-
expect(screen.getByLabelText(
|
|
217
|
+
expect(screen.getByLabelText(/Currency/)).toHaveAttribute('aria-haspopup');
|
|
218
218
|
});
|
|
219
219
|
});
|
package/src/label/Label.spec.tsx
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Input } from '../inputs/Input';
|
|
2
|
-
import { render, screen } from '../test-utils';
|
|
2
|
+
import { lorem10, render, screen } from '../test-utils';
|
|
3
3
|
import { Label } from './Label';
|
|
4
4
|
|
|
5
5
|
describe('Label', () => {
|
|
@@ -11,16 +11,65 @@ describe('Label', () => {
|
|
|
11
11
|
</Label>,
|
|
12
12
|
);
|
|
13
13
|
|
|
14
|
-
expect(screen.getByLabelText(
|
|
14
|
+
expect(screen.getByLabelText(/Phone number/)).toBeInTheDocument();
|
|
15
15
|
});
|
|
16
|
-
|
|
16
|
+
|
|
17
|
+
it('renders string labels with (Optional) suffix', () => {
|
|
17
18
|
render(
|
|
18
19
|
<Label>
|
|
19
|
-
<
|
|
20
|
+
<Label.Optional>Phone number</Label.Optional>
|
|
20
21
|
<Input readOnly />
|
|
21
22
|
</Label>,
|
|
22
23
|
);
|
|
23
24
|
|
|
25
|
+
expect(screen.getByLabelText(/Phone number/)).toBeInTheDocument();
|
|
26
|
+
expect(screen.getByLabelText(/(Optional)/)).toBeInTheDocument();
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
it('renders description', () => {
|
|
30
|
+
const inputId = 'input-id';
|
|
31
|
+
const descriptionId = 'desc-test';
|
|
32
|
+
render(
|
|
33
|
+
<>
|
|
34
|
+
<Label htmlFor={inputId}>Phone number</Label>
|
|
35
|
+
<Label.Description id={descriptionId}>{lorem10}</Label.Description>
|
|
36
|
+
<Input id={inputId} readOnly aria-describedby={descriptionId} />
|
|
37
|
+
</>,
|
|
38
|
+
);
|
|
39
|
+
|
|
24
40
|
expect(screen.getByLabelText('Phone number')).toBeInTheDocument();
|
|
41
|
+
expect(screen.getByText(lorem10)).toBeInTheDocument();
|
|
42
|
+
expect(screen.getByText(lorem10)).toHaveAttribute('id', descriptionId);
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('renders description with optional', () => {
|
|
46
|
+
const descriptionId = 'desc-test';
|
|
47
|
+
render(
|
|
48
|
+
<>
|
|
49
|
+
<Label htmlFor="test">
|
|
50
|
+
<Label.Optional>Phone number</Label.Optional>
|
|
51
|
+
</Label>
|
|
52
|
+
<Label.Description id={descriptionId}>{lorem10}</Label.Description>
|
|
53
|
+
<Input id="test" readOnly aria-describedby={descriptionId} />
|
|
54
|
+
</>,
|
|
55
|
+
);
|
|
56
|
+
|
|
57
|
+
expect(screen.getByLabelText(/Phone number/)).toBeInTheDocument();
|
|
58
|
+
expect(screen.getByText(lorem10)).toBeInTheDocument();
|
|
59
|
+
expect(screen.getByText(lorem10)).toHaveAttribute('id', descriptionId);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('connects label with form control via `id` + `aria-labelledby`', () => {
|
|
63
|
+
const labelId = 'label-id';
|
|
64
|
+
render(
|
|
65
|
+
<>
|
|
66
|
+
<Label id={labelId}>Phone number</Label>
|
|
67
|
+
<div role="group" aria-labelledby={labelId}>
|
|
68
|
+
Custom complex component
|
|
69
|
+
</div>
|
|
70
|
+
</>,
|
|
71
|
+
);
|
|
72
|
+
|
|
73
|
+
expect(screen.getByLabelText('Phone number')).toHaveAttribute('aria-labelledby', labelId);
|
|
25
74
|
});
|
|
26
75
|
});
|
|
@@ -1,37 +1,43 @@
|
|
|
1
|
-
import
|
|
2
|
-
|
|
3
|
-
import Info from '../info/Info';
|
|
4
|
-
import { Input } from '../inputs/Input';
|
|
1
|
+
import Info from '../info';
|
|
5
2
|
import { Label } from './Label';
|
|
6
3
|
|
|
7
4
|
export default {
|
|
8
5
|
component: Label,
|
|
9
6
|
title: 'Label',
|
|
7
|
+
tags: ['autodocs'],
|
|
10
8
|
};
|
|
11
9
|
|
|
12
10
|
export const Basic = () => {
|
|
13
|
-
const [value, setValue] = useState<string | undefined>('This is some text');
|
|
14
|
-
return (
|
|
15
|
-
<Label>
|
|
16
|
-
Phone number
|
|
17
|
-
<Input value={value} id="input" onChange={({ target }) => setValue(target.value)} />
|
|
18
|
-
</Label>
|
|
19
|
-
);
|
|
20
|
-
};
|
|
21
|
-
|
|
22
|
-
export const WithInfo = () => {
|
|
23
|
-
const [value, setValue] = useState<string | undefined>('This is some text');
|
|
24
11
|
return (
|
|
25
|
-
|
|
26
|
-
<
|
|
27
|
-
|
|
28
|
-
<
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
12
|
+
<>
|
|
13
|
+
<p>
|
|
14
|
+
Avoid using <code>Label</code> component directly (unless for some custom use cases), for
|
|
15
|
+
form control labels and descriptions / info messages, please use <code>Field</code>{' '}
|
|
16
|
+
component
|
|
17
|
+
<pre>{'<Field label={..} description={..} required={..}>'}</pre>
|
|
18
|
+
</p>
|
|
19
|
+
<Label className="m-b-2">Text Input</Label>
|
|
20
|
+
<Label className="m-b-2">
|
|
21
|
+
<Label.Optional>Text Input</Label.Optional>
|
|
22
|
+
</Label>
|
|
23
|
+
<Label>
|
|
24
|
+
<Label.Optional>Text Input with Description</Label.Optional>
|
|
25
|
+
</Label>
|
|
26
|
+
<Label.Description className="m-b-2">This a field Description</Label.Description>
|
|
27
|
+
<Label>
|
|
28
|
+
<div>
|
|
29
|
+
Text Input with Description{' '}
|
|
30
|
+
<Info content="This is some help in popover" aria-label="The aria label" />
|
|
31
|
+
</div>
|
|
32
|
+
</Label>
|
|
33
|
+
<Label.Description className="m-b-2">This a field Description</Label.Description>
|
|
34
|
+
<Label>
|
|
35
|
+
<Label.Optional>
|
|
36
|
+
Text Input with Description{' '}
|
|
37
|
+
<Info content="This is some help in popover" aria-label="The aria label" />{' '}
|
|
38
|
+
</Label.Optional>
|
|
39
|
+
</Label>
|
|
40
|
+
<Label.Description className="m-b-2">This a field Description</Label.Description>
|
|
41
|
+
</>
|
|
36
42
|
);
|
|
37
43
|
};
|
package/src/label/Label.tsx
CHANGED
|
@@ -1,4 +1,9 @@
|
|
|
1
1
|
import { clsx } from 'clsx';
|
|
2
|
+
import messages from './Label.messages';
|
|
3
|
+
import { useIntl } from 'react-intl';
|
|
4
|
+
import Body from '../body';
|
|
5
|
+
import { CommonProps } from '../common';
|
|
6
|
+
import { PropsWithChildren } from 'react';
|
|
2
7
|
|
|
3
8
|
export type LabelProps = {
|
|
4
9
|
id?: string;
|
|
@@ -7,14 +12,54 @@ export type LabelProps = {
|
|
|
7
12
|
children?: React.ReactNode;
|
|
8
13
|
};
|
|
9
14
|
|
|
10
|
-
|
|
15
|
+
/**
|
|
16
|
+
* Avoid using `<Label>` component directly
|
|
17
|
+
* it's for edge cases when `<Field />` isn't suitable for some reasons.
|
|
18
|
+
*
|
|
19
|
+
* Example:
|
|
20
|
+
* ```
|
|
21
|
+
* <Field label={..} description={..} required={..}>..</Field>
|
|
22
|
+
* ```
|
|
23
|
+
*/
|
|
24
|
+
const Label = ({ className, children, htmlFor, id }: LabelProps) => {
|
|
11
25
|
return (
|
|
12
26
|
<label
|
|
13
27
|
id={id}
|
|
14
28
|
htmlFor={htmlFor}
|
|
15
|
-
className={clsx(
|
|
29
|
+
className={clsx(
|
|
30
|
+
'np-label d-flex flex-column np-text-body-default-bold text-primary m-b-0',
|
|
31
|
+
className,
|
|
32
|
+
)}
|
|
16
33
|
>
|
|
17
34
|
{children}
|
|
18
35
|
</label>
|
|
19
36
|
);
|
|
20
37
|
};
|
|
38
|
+
|
|
39
|
+
export type LabelOptionalProps = PropsWithChildren<CommonProps>;
|
|
40
|
+
|
|
41
|
+
// eslint-disable-next-line functional/immutable-data
|
|
42
|
+
Label.Optional = function Optional({ children, className }: LabelOptionalProps) {
|
|
43
|
+
const { formatMessage } = useIntl();
|
|
44
|
+
return (
|
|
45
|
+
<div>
|
|
46
|
+
{children}
|
|
47
|
+
<Body as="span" className={clsx('text-secondary', 'm-l-1', className)}>
|
|
48
|
+
{formatMessage(messages.optionalLabel)}
|
|
49
|
+
</Body>
|
|
50
|
+
</div>
|
|
51
|
+
);
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
export type LabelDescriptionProps = PropsWithChildren<CommonProps> & { id?: string };
|
|
55
|
+
|
|
56
|
+
// eslint-disable-next-line functional/immutable-data
|
|
57
|
+
Label.Description = function Description({ id, children, className }: LabelDescriptionProps) {
|
|
58
|
+
return children ? (
|
|
59
|
+
<Body id={id} className={clsx('text-secondary', className)}>
|
|
60
|
+
{children}
|
|
61
|
+
</Body>
|
|
62
|
+
) : null;
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
export { Label };
|
package/src/main.css
CHANGED
|
@@ -1719,18 +1719,10 @@ button.np-option {
|
|
|
1719
1719
|
white-space: nowrap;
|
|
1720
1720
|
width: 100%;
|
|
1721
1721
|
}
|
|
1722
|
-
.np-date-trigger .control-label {
|
|
1723
|
-
font-weight: 400;
|
|
1724
|
-
font-weight: var(--font-weight-regular);
|
|
1725
|
-
}
|
|
1726
1722
|
.np-theme-personal .np-date-trigger {
|
|
1727
1723
|
padding-left: 16px;
|
|
1728
1724
|
padding-left: var(--size-16);
|
|
1729
1725
|
}
|
|
1730
|
-
.np-theme-personal .np-date-trigger .control-label + span {
|
|
1731
|
-
font-weight: 400;
|
|
1732
|
-
font-weight: var(--font-weight-regular);
|
|
1733
|
-
}
|
|
1734
1726
|
.clear-btn {
|
|
1735
1727
|
transition: color 0.15s ease-in-out;
|
|
1736
1728
|
color: #c9cbce;
|
|
@@ -2385,6 +2377,10 @@ html:not([dir="rtl"]) .np-flow-navigation--sm .np-flow-navigation__stepper {
|
|
|
2385
2377
|
border-radius: 9999px !important;
|
|
2386
2378
|
border-radius: var(--radius-full) !important;
|
|
2387
2379
|
}
|
|
2380
|
+
.np-field-control {
|
|
2381
|
+
margin-top: 4px;
|
|
2382
|
+
margin-top: var(--size-4);
|
|
2383
|
+
}
|
|
2388
2384
|
.np-input-group {
|
|
2389
2385
|
display: inline-grid;
|
|
2390
2386
|
width: 100%;
|
package/src/main.less
CHANGED
|
@@ -4,8 +4,7 @@ import { Lock } from '@transferwise/icons';
|
|
|
4
4
|
import { useState } from 'react';
|
|
5
5
|
|
|
6
6
|
import MoneyInput, { CurrencyOptionItem } from '.';
|
|
7
|
-
import
|
|
8
|
-
import translations from '../i18n';
|
|
7
|
+
import { Field } from '../field/Field';
|
|
9
8
|
|
|
10
9
|
export default {
|
|
11
10
|
component: MoneyInput,
|
|
@@ -17,14 +16,13 @@ export default {
|
|
|
17
16
|
const handleOnCurrencyChange = (value: CurrencyOptionItem) => setSelectedCurrency(value);
|
|
18
17
|
|
|
19
18
|
return (
|
|
20
|
-
|
|
21
|
-
<label htmlFor={args.id}>Editable money input label</label>
|
|
19
|
+
<Field id={args.id} label="Editable money input label" required>
|
|
22
20
|
<MoneyInput
|
|
23
21
|
{...args}
|
|
24
22
|
selectedCurrency={selectedCurrency}
|
|
25
23
|
onCurrencyChange={handleOnCurrencyChange}
|
|
26
24
|
/>
|
|
27
|
-
|
|
25
|
+
</Field>
|
|
28
26
|
);
|
|
29
27
|
},
|
|
30
28
|
args: {
|
|
@@ -161,17 +159,19 @@ export const SmallInput: Story = {
|
|
|
161
159
|
render: (args) => {
|
|
162
160
|
return (
|
|
163
161
|
<>
|
|
164
|
-
<
|
|
165
|
-
|
|
162
|
+
<Field id={args.id} label="Money inputs" required>
|
|
163
|
+
<MoneyInput {...args} {...SingleCurrency.args} />
|
|
164
|
+
</Field>
|
|
166
165
|
<br />
|
|
167
166
|
<MoneyInput {...args} {...MultipleCurrencies.args} />
|
|
168
167
|
<hr />
|
|
169
|
-
<
|
|
170
|
-
<label htmlFor={args.id}>Error states</label>
|
|
168
|
+
<Field id={args.id} label="Error states" sentiment="negative" required>
|
|
171
169
|
<MoneyInput {...args} {...SingleCurrency.args} />
|
|
172
|
-
|
|
170
|
+
</Field>
|
|
171
|
+
<br />
|
|
172
|
+
<Field sentiment="negative">
|
|
173
173
|
<MoneyInput {...args} {...MultipleCurrencies.args} />
|
|
174
|
-
</
|
|
174
|
+
</Field>
|
|
175
175
|
</>
|
|
176
176
|
);
|
|
177
177
|
},
|