Skip to content

Commit 17d75d8

Browse files
authored
Convert React Aria Components examples to TypeScript (#4212)
1 parent 67f618d commit 17d75d8

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

64 files changed

+416
-212
lines changed

packages/@react-aria/calendar/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ export {useRangeCalendar} from './useRangeCalendar';
1515
export {useCalendarGrid} from './useCalendarGrid';
1616
export {useCalendarCell} from './useCalendarCell';
1717

18-
export type {AriaCalendarProps, AriaRangeCalendarProps, CalendarProps, RangeCalendarProps} from '@react-types/calendar';
18+
export type {AriaCalendarProps, AriaRangeCalendarProps, CalendarProps, DateValue, RangeCalendarProps} from '@react-types/calendar';
1919
export type {CalendarAria} from './useCalendarBase';
2020
export type {AriaCalendarGridProps, CalendarGridAria} from './useCalendarGrid';
2121
export type {AriaCalendarCellProps, CalendarCellAria} from './useCalendarCell';

packages/@react-aria/combobox/src/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,12 @@
33
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
44
* you may not use this file except in compliance with the License. You may obtain a copy
55
* of the License at http://www.apache.org/licenses/LICENSE-2.0
6-
*
6+
*
77
* Unless required by applicable law or agreed to in writing, software distributed under
88
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
99
* OF ANY KIND, either express or implied. See the License for the specific language
1010
* governing permissions and limitations under the License.
1111
*/
1212
export {useComboBox} from './useComboBox';
1313
export type {AriaComboBoxOptions, ComboBoxAria} from './useComboBox';
14+
export type {AriaComboBoxProps} from '@react-types/combobox';

packages/@react-aria/datepicker/src/index.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -16,8 +16,8 @@ export {useDateField, useTimeField} from './useDateField';
1616
export {useDateRangePicker} from './useDateRangePicker';
1717
export {useDisplayNames} from './useDisplayNames';
1818

19-
export type {AriaDatePickerProps, AriaDateRangePickerProps} from '@react-types/datepicker';
20-
export type {AriaDateFieldProps, DateFieldAria} from './useDateField';
19+
export type {AriaDateFieldProps, AriaDatePickerProps, AriaDateRangePickerProps, DateRange, DateValue, TimeValue} from '@react-types/datepicker';
20+
export type {AriaDateFieldOptions, DateFieldAria} from './useDateField';
2121
export type {DatePickerAria} from './useDatePicker';
2222
export type {DateRangePickerAria} from './useDateRangePicker';
2323
export type {DateSegmentAria} from './useDateSegment';

packages/@react-aria/datepicker/src/useDateField.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ import {useFocusWithin} from '@react-aria/interactions';
2424
import {useLocalizedStringFormatter} from '@react-aria/i18n';
2525

2626
// Allows this hook to also be used with TimeField
27-
export interface AriaDateFieldProps<T extends DateValue> extends Omit<AriaDateFieldPropsBase<T>, 'value' | 'defaultValue' | 'onChange' | 'minValue' | 'maxValue' | 'placeholderValue'> {}
27+
export interface AriaDateFieldOptions<T extends DateValue> extends Omit<AriaDateFieldPropsBase<T>, 'value' | 'defaultValue' | 'onChange' | 'minValue' | 'maxValue' | 'placeholderValue'> {}
2828

2929
export interface DateFieldAria {
3030
/** Props for the field's visible label element, if any. */
@@ -57,7 +57,7 @@ export const focusManagerSymbol = '__focusManager_' + Date.now();
5757
* A date field allows users to enter and edit date and time values using a keyboard.
5858
* Each part of a date value is displayed in an individually editable segment.
5959
*/
60-
export function useDateField<T extends DateValue>(props: AriaDateFieldProps<T>, state: DateFieldState, ref: RefObject<Element>): DateFieldAria {
60+
export function useDateField<T extends DateValue>(props: AriaDateFieldOptions<T>, state: DateFieldState, ref: RefObject<Element>): DateFieldAria {
6161
let {labelProps, fieldProps, descriptionProps, errorMessageProps} = useField({
6262
...props,
6363
labelElementType: 'span'

packages/@react-aria/overlays/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -28,3 +28,4 @@ export type {DismissButtonProps} from './DismissButton';
2828
export type {AriaPopoverProps, PopoverAria} from './usePopover';
2929
export type {AriaModalOverlayProps, ModalOverlayAria} from './useModalOverlay';
3030
export type {OverlayProps} from './Overlay';
31+
export type {Placement, PlacementAxis, PositionProps} from '@react-types/overlays';

packages/@react-aria/radio/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -12,5 +12,6 @@
1212
export {useRadio} from './useRadio';
1313
export {useRadioGroup} from './useRadioGroup';
1414
export type {AriaRadioGroupProps, AriaRadioProps} from '@react-types/radio';
15+
export type {Orientation} from '@react-types/shared';
1516
export type {RadioAria} from './useRadio';
1617
export type {RadioGroupAria} from './useRadioGroup';

packages/@react-aria/select/src/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
44
* you may not use this file except in compliance with the License. You may obtain a copy
55
* of the License at http://www.apache.org/licenses/LICENSE-2.0
6-
*
6+
*
77
* Unless required by applicable law or agreed to in writing, software distributed under
88
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
99
* OF ANY KIND, either express or implied. See the License for the specific language
@@ -13,3 +13,4 @@ export {useSelect} from './useSelect';
1313
export {useHiddenSelect, HiddenSelect} from './HiddenSelect';
1414
export type {AriaSelectOptions, SelectAria} from './useSelect';
1515
export type {AriaHiddenSelectProps, HiddenSelectProps} from './HiddenSelect';
16+
export type {AriaSelectProps} from '@react-types/select';

packages/@react-aria/slider/src/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -14,3 +14,5 @@ export {useSliderThumb} from './useSliderThumb';
1414
export type {AriaSliderProps} from '@react-types/slider';
1515
export type {SliderAria} from './useSlider';
1616
export type {AriaSliderThumbOptions, SliderThumbAria} from './useSliderThumb';
17+
export type {AriaSliderThumbProps} from '@react-types/slider';
18+
export type {Orientation} from '@react-types/shared';

packages/@react-aria/tabs/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ export {useTab} from './useTab';
1313
export {useTabPanel} from './useTabPanel';
1414
export {useTabList} from './useTabList';
1515
export type {AriaTabListProps, AriaTabPanelProps, AriaTabProps} from '@react-types/tabs';
16+
export type {Orientation} from '@react-types/shared';
1617
export type {TabAria} from './useTab';
1718
export type {TabPanelAria} from './useTabPanel';
1819
export type {TabListAria} from './useTabList';

packages/@react-stately/table/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,5 +22,5 @@ export {Column} from './Column';
2222
export {Row} from './Row';
2323
export {Cell} from './Cell';
2424
export {Section} from '@react-stately/collections';
25-
export {TableCollection} from './TableCollection';
25+
export {TableCollection, buildHeaderRows} from './TableCollection';
2626
export {TableColumnLayout} from './TableColumnLayout';

packages/dev/docs/src/docs.css

+8
Original file line numberDiff line numberDiff line change
@@ -365,6 +365,10 @@ footer li:last-child:after {
365365
margin-bottom: 0;
366366
}
367367

368+
.propTable td[data-column="CSS Selector"] code {
369+
white-space: nowrap;
370+
}
371+
368372
.type-description {
369373
margin-top: 0;
370374
}
@@ -720,6 +724,10 @@ h2.sectionHeader {
720724
main {
721725
margin-right: 0;
722726
}
727+
728+
.propTable td[data-column="CSS Selector"] code {
729+
white-space: normal;
730+
}
723731
}
724732

725733
@define-mixin small-propTable {

packages/dev/parcel-transformer-docs/DocsTransformer.js

+12-5
Original file line numberDiff line numberDiff line change
@@ -649,12 +649,19 @@ module.exports = new Transformer({
649649
return false;
650650
}
651651

652-
if (!t.isMemberExpression(path.node.callee)) {
653-
return path.get('callee').referencesImport(module, name);
652+
let callee = path.node.callee;
653+
let calleePath = path.get('callee');
654+
if (t.isTSAsExpression(callee)) {
655+
callee = callee.expression;
656+
calleePath = calleePath.get('expression');
654657
}
655658

656-
if (path.get('callee.object').referencesImport(module, 'default')) {
657-
return t.isIdentifier(path.node.callee.property, {name});
659+
if (!t.isMemberExpression(callee)) {
660+
return calleePath.referencesImport(module, name);
661+
}
662+
663+
if (calleePath.get('object').referencesImport(module, 'default')) {
664+
return t.isIdentifier(callee.property, {name});
658665
}
659666

660667
return false;
@@ -664,7 +671,7 @@ module.exports = new Transformer({
664671
if (path.isFunction()) {
665672
if (
666673
path.node.returnType &&
667-
t.isTSTypeReference(path.node.returnType.typeAnnotation) &&
674+
t.isTSTypeReference(path.node.returnType.typeAnnotation) &&
668675
t.isTSQualifiedName(path.node.returnType.typeAnnotation.typeName) &&
669676
t.isIdentifier(path.node.returnType.typeAnnotation.typeName.left, {name: 'JSX'}) &&
670677
t.isIdentifier(path.node.returnType.typeAnnotation.typeName.right, {name: 'Element'})

packages/react-aria-components/docs/Calendar.mdx

+7-2
Original file line numberDiff line numberDiff line change
@@ -382,9 +382,14 @@ If you will use a Calendar in multiple places in your app, you can wrap all of t
382382
This example also shows how to use the `errorMessage` slot to render help text in case of a validation error ([see below for details](#error-message)).
383383

384384
```tsx example export=true
385+
import type {CalendarProps, DateValue} from 'react-aria-components';
385386
import {Text} from 'react-aria-components';
386387

387-
function MyCalendar({errorMessage, ...props}) {
388+
interface MyCalendarProps<T extends DateValue> extends CalendarProps<T> {
389+
errorMessage?: string
390+
}
391+
392+
function MyCalendar<T extends DateValue>({errorMessage, ...props}: MyCalendarProps<T>) {
388393
return (
389394
<Calendar {...props}>
390395
<header>
@@ -468,7 +473,7 @@ The below example displays a `Calendar` in the Hindi language, using the Indian
468473
import {I18nProvider} from '@react-aria/i18n';
469474

470475
function Example() {
471-
let [date, setDate] = React.useState(null);
476+
let [date, setDate] = React.useState<DateValue>(null);
472477
return (
473478
<I18nProvider locale="hi-IN-u-ca-indian">
474479
<MyCalendar aria-label="Date" value={date} onChange={setDate} />

packages/react-aria-components/docs/Checkbox.mdx

+3-1
Original file line numberDiff line numberDiff line change
@@ -260,7 +260,9 @@ If you will use a Checkbox in multiple places in your app, you can wrap all of t
260260
This example wraps `Checkbox` and all of its children together into a single component. It also shows how to use render props to conditionally render a different indicator icon when the checkbox is in an indeterminate state.
261261

262262
```tsx example export=true
263-
function MyCheckbox({children, ...props}) {
263+
import type {CheckboxProps} from 'react-aria-components';
264+
265+
function MyCheckbox({children, ...props}: CheckboxProps) {
264266
return (
265267
<Checkbox {...props}>
266268
{({isIndeterminate}) => <>

packages/react-aria-components/docs/CheckboxGroup.mdx

+9-2
Original file line numberDiff line numberDiff line change
@@ -341,15 +341,22 @@ If you will use a CheckboxGroup in multiple places in your app, you can wrap all
341341
This example wraps `CheckboxGroup` and all of its children together into a single component which accepts a `label` prop, which is passed to the right place. It also shows how to use the `description` and `errorMessage` slots to render help text ([see below for details](#help-text)).
342342

343343
```tsx example export=true
344+
import type {CheckboxGroupProps, CheckboxProps} from 'react-aria-components';
344345
import {Text} from 'react-aria-components';
345346

347+
interface MyCheckboxGroupProps extends CheckboxGroupProps {
348+
label?: string,
349+
description?: string,
350+
errorMessage?: string
351+
}
352+
346353
function MyCheckboxGroup({
347354
label,
348355
description,
349356
errorMessage,
350357
children,
351358
...props
352-
}) {
359+
}: MyCheckboxGroupProps) {
353360
return (
354361
<CheckboxGroup {...props}>
355362
{label}
@@ -360,7 +367,7 @@ function MyCheckboxGroup({
360367
);
361368
}
362369

363-
function MyCheckbox({children, ...props}) {
370+
function MyCheckbox({children, ...props}: CheckboxProps) {
364371
return (
365372
<Checkbox {...props}>
366373
<div className="checkbox" aria-hidden="true">

packages/react-aria-components/docs/ComboBox.mdx

+18-6
Original file line numberDiff line numberDiff line change
@@ -462,9 +462,17 @@ If you will use a ComboBox in multiple places in your app, you can wrap all of t
462462
This example wraps `ComboBox` and all of its children together into a single component which accepts a `label` prop and `children`, which are passed through to the right places. It also shows how to use the `description` and `errorMessage` slots to render help text ([see below for details](#help-text)). The `Item` component is also wrapped to apply class names based on the current state, as described [above](#styling).
463463

464464
```tsx example export=true
465+
import type {ComboBoxProps, ItemProps} from 'react-aria-components';
465466
import {Text} from 'react-aria-components';
466467

467-
function MyComboBox({label, description, errorMessage, children, ...props}) {
468+
interface MyComboBoxProps<T extends object> extends ComboBoxProps<T> {
469+
label?: string,
470+
description?: string | null,
471+
errorMessage?: string | null,
472+
children: React.ReactNode | ((item: T) => React.ReactNode)
473+
}
474+
475+
function MyComboBox<T extends object>({label, description, errorMessage, children, ...props}: MyComboBoxProps<T>) {
468476
return (
469477
<ComboBox {...props}>
470478
<Label>{label}</Label>
@@ -483,7 +491,7 @@ function MyComboBox({label, description, errorMessage, children, ...props}) {
483491
);
484492
}
485493

486-
function MyItem(props) {
494+
function MyItem(props: ItemProps) {
487495
return <Item {...props} className={({isFocused, isSelected}) => `my-item ${isFocused ? 'focused' : ''} ${isSelected ? 'selected' : ''}`} />
488496
}
489497

@@ -565,7 +573,7 @@ function Example() {
565573
{id: 8, name: 'Agricultural'},
566574
{id: 9, name: 'Electrical'}
567575
];
568-
let [majorId, setMajorId] = React.useState();
576+
let [majorId, setMajorId] = React.useState<React.Key | null>(null);
569577

570578
return (
571579
<>
@@ -791,7 +799,7 @@ function ControlledComboBox() {
791799
let onInputChange = (value) => {
792800
setFieldState(prevState => ({
793801
inputValue: value,
794-
selectedKey: value === '' ? null : prevState.selectedKey,
802+
selectedKey: value === '' ? '' : prevState.selectedKey,
795803
items: optionList.filter(item => startsWith(item.name, value))
796804
}));
797805
};
@@ -875,8 +883,12 @@ of a ComboBox supporting those features.
875883
```tsx example
876884
import {useAsyncList} from '@react-stately/data';
877885

886+
interface Character {
887+
name: string
888+
}
889+
878890
function AsyncLoadingExample() {
879-
let list = useAsyncList({
891+
let list = useAsyncList<Character>({
880892
async load({signal, filterText}) {
881893
let res = await fetch(
882894
`https://swapi.py4e.com/api/people/?search=${filterText}`,
@@ -908,7 +920,7 @@ The `description` slot can be used to associate additional help text with a Comb
908920

909921
```tsx example
910922
function Example() {
911-
let [animalId, setAnimalId] = React.useState();
923+
let [animalId, setAnimalId] = React.useState<React.Key | null>(null);
912924
let options = [
913925
{ id: 1, name: 'Aardvark' },
914926
{ id: 2, name: 'Cat' },

packages/react-aria-components/docs/DateField.mdx

+9-2
Original file line numberDiff line numberDiff line change
@@ -308,9 +308,16 @@ If you will use a DateField in multiple places in your app, you can wrap all of
308308
This example wraps `DateField` and all of its children together into a single component which accepts a `label` prop, which is passed to the right place. It also shows how to use the `description` and `errorMessage` slots to render help text ([see below for details](#help-text)).
309309

310310
```tsx example export=true
311+
import type {DateFieldProps, DateValue} from 'react-aria-components';
311312
import {Text} from 'react-aria-components';
312313

313-
function MyDateField({label, description, errorMessage, ...props}) {
314+
interface MyDateFieldProps<T extends DateValue> extends DateFieldProps<T> {
315+
label?: string,
316+
description?: string,
317+
errorMessage?: string
318+
}
319+
320+
function MyDateField<T extends DateValue>({label, description, errorMessage, ...props}: MyDateFieldProps<T>) {
314321
return (
315322
<DateField {...props}>
316323
<Label>{label}</Label>
@@ -464,7 +471,7 @@ The below example displays a `DateField` in the Hindi language, using the Indian
464471
import {I18nProvider} from '@react-aria/i18n';
465472

466473
function Example() {
467-
let [date, setDate] = React.useState(null);
474+
let [date, setDate] = React.useState<DateValue>(null);
468475
return (
469476
<I18nProvider locale="hi-IN-u-ca-indian">
470477
<MyDateField label="Date" value={date} onChange={setDate} />

packages/react-aria-components/docs/DatePicker.mdx

+11-4
Original file line numberDiff line numberDiff line change
@@ -664,9 +664,16 @@ If you will use a DatePicker in multiple places in your app, you can wrap all of
664664
This example wraps `DatePicker` and all of its children together into a single component which accepts a `label` prop, which is passed to the right place. It also shows how to use the `description` and `errorMessage` slots to render help text ([see below for details](#help-text)).
665665

666666
```tsx example export=true
667+
import type {DatePickerProps, DateValue} from 'react-aria-components';
667668
import {Text} from 'react-aria-components';
668669

669-
function MyDatePicker({label, description, errorMessage, ...props}) {
670+
interface MyDatePickerProps<T extends DateValue> extends DatePickerProps<T> {
671+
label?: string,
672+
description?: string,
673+
errorMessage?: string
674+
}
675+
676+
function MyDatePicker<T extends DateValue>({label, description, errorMessage, ...props}: MyDatePickerProps<T>) {
670677
return (
671678
<DatePicker {...props}>
672679
<Label>{label}</Label>
@@ -838,7 +845,7 @@ The below example displays a `DatePicker` in the Hindi language, using the India
838845
import {I18nProvider} from '@react-aria/i18n';
839846

840847
function Example() {
841-
let [date, setDate] = React.useState(null);
848+
let [date, setDate] = React.useState<DateValue>(null);
842849
return (
843850
<I18nProvider locale="hi-IN-u-ca-indian">
844851
<MyDatePicker label="Date" value={date} onChange={setDate} />
@@ -909,8 +916,8 @@ function Example() {
909916
value={date}
910917
onChange={setDate}
911918
validationState={isInvalid ? 'invalid' : 'valid'}
912-
description={isInvalid ? null : 'Select a weekday'}
913-
errorMessage={isInvalid ? 'We are closed on weekends' : null} />
919+
description={isInvalid ? undefined : 'Select a weekday'}
920+
errorMessage={isInvalid ? 'We are closed on weekends' : undefined} />
914921
);
915922
}
916923
```

0 commit comments

Comments
 (0)