Skip to content

Commit 0d45da5

Browse files
authored
Merge pull request #8103 from marmelab/useGetRecordId
Feat: Add useGetRecordId in ra-core
2 parents f430559 + ccc828b commit 0d45da5

File tree

2 files changed

+89
-0
lines changed

2 files changed

+89
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
import * as React from 'react';
2+
import { useGetRecordId } from './useGetRecordId';
3+
import { render, screen } from '@testing-library/react';
4+
import { MemoryRouter, Route, Routes } from 'react-router-dom';
5+
import { RecordContextProvider } from '..';
6+
7+
describe('useGetRecordId', () => {
8+
const UseGetRecordId = (props: any) => {
9+
const recordId = useGetRecordId(props.id);
10+
return <div>{recordId}</div>;
11+
};
12+
13+
it('should return the record id it received in options', () => {
14+
render(<UseGetRecordId id="abc" />);
15+
expect(screen.queryByText('abc')).not.toBeNull();
16+
});
17+
18+
it('should return the record id it received in options even if it is falsy', () => {
19+
render(<UseGetRecordId id={0} />);
20+
expect(screen.queryByText('0')).not.toBeNull();
21+
});
22+
23+
it('should return the record id it received through the record context', () => {
24+
render(
25+
<RecordContextProvider value={{ id: 'abc' }}>
26+
<UseGetRecordId />
27+
</RecordContextProvider>
28+
);
29+
expect(screen.queryByText('abc')).not.toBeNull();
30+
});
31+
32+
it('should return the record id it received through the record context even if it is falsy', () => {
33+
render(
34+
<RecordContextProvider value={{ id: 0 }}>
35+
<UseGetRecordId />
36+
</RecordContextProvider>
37+
);
38+
expect(screen.queryByText('0')).not.toBeNull();
39+
});
40+
41+
it('should return the record id parsed from the location', () => {
42+
render(
43+
<MemoryRouter initialEntries={['/posts/abc']}>
44+
<Routes>
45+
<Route path="/posts/:id" element={<UseGetRecordId />} />
46+
</Routes>
47+
</MemoryRouter>
48+
);
49+
expect(screen.queryByText('abc')).not.toBeNull();
50+
});
51+
52+
it('should return the record id parsed from the location even if it is falsy', () => {
53+
render(
54+
<MemoryRouter initialEntries={['/posts/0']}>
55+
<Routes>
56+
<Route path="/posts/:id" element={<UseGetRecordId />} />
57+
</Routes>
58+
</MemoryRouter>
59+
);
60+
expect(screen.queryByText('0')).not.toBeNull();
61+
});
62+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
import { useParams } from 'react-router-dom';
2+
import { useRecordContext } from '../controller';
3+
import { Identifier } from '../types';
4+
5+
/**
6+
* Helper hook to get the current `recordId`.
7+
*
8+
* `recordId` is obtained from parameters if passed as a parameter, or from the `RecordContext` if there is one, or, lastly, from the react-router URL.
9+
*
10+
* @param {any} recordId optional if used inside a RecordContextProvider or if recordId can be guessed from the URL
11+
*
12+
* @returns The `recordId` determined in this manner.
13+
*
14+
* @example
15+
* const recordId = useGetRecordId();
16+
*/
17+
export function useGetRecordId(recordId?: Identifier): Identifier {
18+
const contextRecord = useRecordContext();
19+
const { id: routeId } = useParams<'id'>();
20+
const actualRecordId = recordId ?? contextRecord?.id ?? routeId;
21+
if (actualRecordId == null)
22+
throw new Error(
23+
`useGetRecordId could not find the current record id. You need to use it inside a RecordContextProvider, or inside a supported route, or provide the record id to the hook yourself.`
24+
);
25+
26+
return actualRecordId;
27+
}

0 commit comments

Comments
 (0)