@sanity/sdk-react 0.0.0-alpha.20 → 0.0.0-alpha.4
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/README.md +73 -34
- package/dist/_chunks-es/context.js +8 -0
- package/dist/_chunks-es/context.js.map +1 -0
- package/dist/_chunks-es/useLogOut.js +44 -0
- package/dist/_chunks-es/useLogOut.js.map +1 -0
- package/dist/components.d.ts +83 -0
- package/dist/components.js +132 -0
- package/dist/components.js.map +1 -0
- package/dist/context.d.ts +39 -0
- package/dist/context.js +5 -0
- package/dist/context.js.map +1 -0
- package/dist/hooks.d.ts +277 -0
- package/dist/hooks.js +153 -0
- package/dist/hooks.js.map +1 -0
- package/dist/index.d.ts +2 -4742
- package/dist/index.js +2 -1054
- package/dist/index.js.map +1 -1
- package/package.json +43 -22
- package/src/_exports/components.ts +2 -0
- package/src/_exports/context.ts +2 -0
- package/src/_exports/hooks.ts +21 -0
- package/src/_exports/index.ts +10 -66
- package/src/components/Login/LoginLinks.test.tsx +3 -2
- package/src/components/SanityApp.test.tsx +2 -104
- package/src/components/SanityApp.tsx +16 -80
- package/src/components/auth/AuthBoundary.test.tsx +10 -4
- package/src/components/auth/AuthBoundary.tsx +2 -18
- package/src/components/auth/Login.test.tsx +9 -2
- package/src/components/auth/Login.tsx +0 -1
- package/src/components/auth/LoginCallback.test.tsx +9 -2
- package/src/components/auth/LoginError.test.tsx +10 -2
- package/src/components/auth/LoginFooter.test.tsx +9 -2
- package/src/components/auth/LoginLayout.test.tsx +9 -2
- package/src/context/SanityProvider.test.tsx +1 -1
- package/src/context/SanityProvider.tsx +13 -21
- package/src/hooks/auth/useAuthState.tsx +5 -4
- package/src/hooks/auth/useAuthToken.tsx +1 -1
- package/src/hooks/auth/useCurrentUser.tsx +4 -27
- package/src/hooks/auth/useHandleCallback.tsx +0 -1
- package/src/hooks/auth/useLogOut.tsx +1 -1
- package/src/hooks/auth/useLoginUrls.tsx +0 -1
- package/src/hooks/client/useClient.test.tsx +130 -0
- package/src/hooks/client/useClient.ts +30 -8
- package/src/hooks/comlink/useFrameConnection.test.tsx +10 -55
- package/src/hooks/comlink/useFrameConnection.ts +47 -43
- package/src/hooks/comlink/useWindowConnection.test.ts +12 -53
- package/src/hooks/comlink/useWindowConnection.ts +33 -73
- package/src/hooks/context/useSanityInstance.test.tsx +1 -1
- package/src/hooks/context/useSanityInstance.ts +7 -23
- package/src/hooks/documentCollection/useDocuments.test.ts +130 -0
- package/src/hooks/documentCollection/useDocuments.ts +87 -0
- package/src/hooks/helpers/createCallbackHook.tsx +2 -3
- package/src/hooks/helpers/createStateSourceHook.test.tsx +0 -66
- package/src/hooks/helpers/createStateSourceHook.tsx +10 -29
- package/src/hooks/preview/usePreview.test.tsx +10 -19
- package/src/hooks/preview/usePreview.tsx +12 -66
- package/src/components/SDKProvider.test.tsx +0 -79
- package/src/components/SDKProvider.tsx +0 -42
- package/src/components/auth/authTestHelpers.tsx +0 -11
- package/src/components/utils.ts +0 -22
- package/src/context/SanityInstanceContext.ts +0 -4
- package/src/hooks/_synchronous-groq-js.mjs +0 -4
- package/src/hooks/auth/useDashboardOrganizationId.test.tsx +0 -42
- package/src/hooks/auth/useDashboardOrganizationId.tsx +0 -29
- package/src/hooks/comlink/useManageFavorite.test.ts +0 -106
- package/src/hooks/comlink/useManageFavorite.ts +0 -101
- package/src/hooks/comlink/useRecordDocumentHistoryEvent.test.ts +0 -77
- package/src/hooks/comlink/useRecordDocumentHistoryEvent.ts +0 -79
- package/src/hooks/dashboard/useNavigateToStudioDocument.ts +0 -97
- package/src/hooks/dashboard/useStudioWorkspacesByResourceId.test.tsx +0 -274
- package/src/hooks/dashboard/useStudioWorkspacesByResourceId.ts +0 -91
- package/src/hooks/datasets/useDatasets.ts +0 -37
- package/src/hooks/document/useApplyActions.test.ts +0 -25
- package/src/hooks/document/useApplyActions.ts +0 -74
- package/src/hooks/document/useDocument.test.ts +0 -81
- package/src/hooks/document/useDocument.ts +0 -107
- package/src/hooks/document/useDocumentEvent.test.ts +0 -63
- package/src/hooks/document/useDocumentEvent.ts +0 -54
- package/src/hooks/document/useDocumentSyncStatus.test.ts +0 -16
- package/src/hooks/document/useDocumentSyncStatus.ts +0 -30
- package/src/hooks/document/useEditDocument.test.ts +0 -179
- package/src/hooks/document/useEditDocument.ts +0 -195
- package/src/hooks/document/usePermissions.ts +0 -82
- package/src/hooks/infiniteList/useInfiniteList.test.tsx +0 -152
- package/src/hooks/infiniteList/useInfiniteList.ts +0 -174
- package/src/hooks/paginatedList/usePaginatedList.test.tsx +0 -259
- package/src/hooks/paginatedList/usePaginatedList.ts +0 -290
- package/src/hooks/projection/useProjection.test.tsx +0 -218
- package/src/hooks/projection/useProjection.ts +0 -147
- package/src/hooks/projects/useProject.ts +0 -45
- package/src/hooks/projects/useProjects.ts +0 -41
- package/src/hooks/query/useQuery.test.tsx +0 -188
- package/src/hooks/query/useQuery.ts +0 -103
- package/src/hooks/users/useUsers.test.ts +0 -163
- package/src/hooks/users/useUsers.ts +0 -107
- package/src/utils/getEnv.ts +0 -21
- package/src/version.ts +0 -8
|
@@ -1,41 +0,0 @@
|
|
|
1
|
-
import {type SanityProject} from '@sanity/client'
|
|
2
|
-
import {getProjectsState, resolveProjects, type SanityInstance, type StateSource} from '@sanity/sdk'
|
|
3
|
-
|
|
4
|
-
import {createStateSourceHook} from '../helpers/createStateSourceHook'
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* @public
|
|
8
|
-
* @category Types
|
|
9
|
-
*/
|
|
10
|
-
export type ProjectWithoutMembers = Omit<SanityProject, 'members'>
|
|
11
|
-
|
|
12
|
-
type UseProjects = {
|
|
13
|
-
/**
|
|
14
|
-
*
|
|
15
|
-
* Returns metadata for each project in your organization.
|
|
16
|
-
*
|
|
17
|
-
* @category Projects
|
|
18
|
-
* @returns An array of metadata (minus the projects’ members) for each project in your organization
|
|
19
|
-
* @example
|
|
20
|
-
* ```tsx
|
|
21
|
-
* const projects = useProjects()
|
|
22
|
-
*
|
|
23
|
-
* return (
|
|
24
|
-
* <select>
|
|
25
|
-
* {projects.map((project) => (
|
|
26
|
-
* <option key={project.id}>{project.displayName}</option>
|
|
27
|
-
* ))}
|
|
28
|
-
* </select>
|
|
29
|
-
* )
|
|
30
|
-
* ```
|
|
31
|
-
*/
|
|
32
|
-
(): ProjectWithoutMembers[]
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/** @public */
|
|
36
|
-
export const useProjects: UseProjects = createStateSourceHook({
|
|
37
|
-
// remove `undefined` since we're suspending when that is the case
|
|
38
|
-
getState: getProjectsState as (instance: SanityInstance) => StateSource<ProjectWithoutMembers[]>,
|
|
39
|
-
shouldSuspend: (instance) => getProjectsState(instance).getCurrent() === undefined,
|
|
40
|
-
suspender: resolveProjects,
|
|
41
|
-
})
|
|
@@ -1,188 +0,0 @@
|
|
|
1
|
-
import {getQueryState, resolveQuery, type StateSource} from '@sanity/sdk'
|
|
2
|
-
import {act, render, screen} from '@testing-library/react'
|
|
3
|
-
import {Suspense, useState} from 'react'
|
|
4
|
-
import {type Observable, Subject} from 'rxjs'
|
|
5
|
-
import {beforeEach, describe, expect, it, vi} from 'vitest'
|
|
6
|
-
|
|
7
|
-
import {useQuery} from './useQuery'
|
|
8
|
-
|
|
9
|
-
// Mock the functions from '@sanity/sdk'
|
|
10
|
-
vi.mock('@sanity/sdk', async (importOriginal) => {
|
|
11
|
-
const original = await importOriginal<typeof import('@sanity/sdk')>()
|
|
12
|
-
return {
|
|
13
|
-
...original,
|
|
14
|
-
getQueryState: vi.fn(),
|
|
15
|
-
resolveQuery: vi.fn(),
|
|
16
|
-
}
|
|
17
|
-
})
|
|
18
|
-
|
|
19
|
-
// Mock the Sanity instance hook to return a dummy instance
|
|
20
|
-
vi.mock('../context/useSanityInstance', () => ({
|
|
21
|
-
useSanityInstance: vi.fn().mockReturnValue({}),
|
|
22
|
-
}))
|
|
23
|
-
|
|
24
|
-
describe('useQuery', () => {
|
|
25
|
-
beforeEach(() => {
|
|
26
|
-
vi.resetAllMocks()
|
|
27
|
-
})
|
|
28
|
-
|
|
29
|
-
it('should render data immediately when available', () => {
|
|
30
|
-
const getCurrent = vi.fn().mockReturnValue('test data')
|
|
31
|
-
vi.mocked(getQueryState).mockReturnValue({
|
|
32
|
-
getCurrent,
|
|
33
|
-
subscribe: vi.fn(),
|
|
34
|
-
get observable(): Observable<unknown> {
|
|
35
|
-
throw new Error('Not implemented')
|
|
36
|
-
},
|
|
37
|
-
} as StateSource<unknown>)
|
|
38
|
-
|
|
39
|
-
function TestComponent() {
|
|
40
|
-
const {data, isPending} = useQuery<string>('test query')
|
|
41
|
-
return (
|
|
42
|
-
<div data-testid="output">
|
|
43
|
-
{data} - {isPending ? 'pending' : 'not pending'}
|
|
44
|
-
</div>
|
|
45
|
-
)
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
render(<TestComponent />)
|
|
49
|
-
|
|
50
|
-
// Verify that the output contains the data and that isPending is false
|
|
51
|
-
expect(screen.getByTestId('output').textContent).toContain('test data')
|
|
52
|
-
expect(screen.getByTestId('output').textContent).toContain('not pending')
|
|
53
|
-
})
|
|
54
|
-
|
|
55
|
-
it('should suspend rendering until data is resolved via Suspense', async () => {
|
|
56
|
-
const ref = {current: undefined as string | undefined}
|
|
57
|
-
const getCurrent = vi.fn(() => ref.current)
|
|
58
|
-
const storeChanged$ = new Subject<void>()
|
|
59
|
-
|
|
60
|
-
vi.mocked(getQueryState).mockReturnValue({
|
|
61
|
-
getCurrent,
|
|
62
|
-
subscribe: vi.fn((cb) => {
|
|
63
|
-
const subscription = storeChanged$.subscribe(cb)
|
|
64
|
-
return () => subscription.unsubscribe()
|
|
65
|
-
}),
|
|
66
|
-
get observable(): Observable<unknown> {
|
|
67
|
-
throw new Error('Not implemented')
|
|
68
|
-
},
|
|
69
|
-
} as StateSource<unknown>)
|
|
70
|
-
|
|
71
|
-
// Create a controllable promise to simulate the query resolution
|
|
72
|
-
let resolvePromise: () => void
|
|
73
|
-
// Mock resolveQuery to return our fake promise
|
|
74
|
-
vi.mocked(resolveQuery).mockReturnValue(
|
|
75
|
-
new Promise<void>((resolve) => {
|
|
76
|
-
resolvePromise = () => {
|
|
77
|
-
ref.current = 'resolved data'
|
|
78
|
-
storeChanged$.next()
|
|
79
|
-
resolve()
|
|
80
|
-
}
|
|
81
|
-
}),
|
|
82
|
-
)
|
|
83
|
-
|
|
84
|
-
function TestComponent() {
|
|
85
|
-
const {data} = useQuery<string>('test query')
|
|
86
|
-
return <div data-testid="output">{data}</div>
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
render(
|
|
90
|
-
<Suspense fallback={<div data-testid="fallback">Loading...</div>}>
|
|
91
|
-
<TestComponent />
|
|
92
|
-
</Suspense>,
|
|
93
|
-
)
|
|
94
|
-
|
|
95
|
-
// Initially, since storeValue is undefined, the component should suspend and fallback is shown
|
|
96
|
-
expect(screen.getByTestId('fallback')).toBeInTheDocument()
|
|
97
|
-
|
|
98
|
-
// Now simulate that data becomes available
|
|
99
|
-
await act(async () => {
|
|
100
|
-
resolvePromise()
|
|
101
|
-
})
|
|
102
|
-
|
|
103
|
-
expect(screen.getByTestId('output').textContent).toContain('resolved data')
|
|
104
|
-
})
|
|
105
|
-
|
|
106
|
-
it('should display transition pending state during query change', async () => {
|
|
107
|
-
const ref = {current: undefined as string | undefined}
|
|
108
|
-
const getCurrent = vi.fn(() => ref.current)
|
|
109
|
-
const storeChanged$ = new Subject<void>()
|
|
110
|
-
|
|
111
|
-
vi.mocked(getQueryState).mockImplementation((_instance, query) => {
|
|
112
|
-
if (query === 'query1') {
|
|
113
|
-
return {
|
|
114
|
-
getCurrent: vi.fn().mockReturnValue('data1'),
|
|
115
|
-
subscribe: vi.fn(),
|
|
116
|
-
get observable(): Observable<unknown> {
|
|
117
|
-
throw new Error('Not implemented')
|
|
118
|
-
},
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
return {
|
|
123
|
-
getCurrent,
|
|
124
|
-
subscribe: vi.fn((cb) => {
|
|
125
|
-
const subscription = storeChanged$.subscribe(cb)
|
|
126
|
-
return () => subscription.unsubscribe()
|
|
127
|
-
}),
|
|
128
|
-
get observable(): Observable<unknown> {
|
|
129
|
-
throw new Error('Not implemented')
|
|
130
|
-
},
|
|
131
|
-
}
|
|
132
|
-
})
|
|
133
|
-
|
|
134
|
-
// Create a controllable promise to simulate the query resolution
|
|
135
|
-
let resolvePromise: () => void
|
|
136
|
-
// Mock resolveQuery to return our fake promise
|
|
137
|
-
vi.mocked(resolveQuery).mockReturnValue(
|
|
138
|
-
new Promise<void>((resolve) => {
|
|
139
|
-
resolvePromise = () => {
|
|
140
|
-
ref.current = 'data2'
|
|
141
|
-
storeChanged$.next()
|
|
142
|
-
resolve()
|
|
143
|
-
}
|
|
144
|
-
}),
|
|
145
|
-
)
|
|
146
|
-
|
|
147
|
-
function WrapperComponent() {
|
|
148
|
-
const [query, setQuery] = useState('query1')
|
|
149
|
-
const {data, isPending} = useQuery<string>(query)
|
|
150
|
-
return (
|
|
151
|
-
<div>
|
|
152
|
-
<div data-testid="output">
|
|
153
|
-
{data} - {isPending ? 'pending' : 'not pending'}
|
|
154
|
-
</div>
|
|
155
|
-
<button data-testid="button" onClick={() => setQuery('query2')}>
|
|
156
|
-
Change Query
|
|
157
|
-
</button>
|
|
158
|
-
</div>
|
|
159
|
-
)
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
render(<WrapperComponent />)
|
|
163
|
-
|
|
164
|
-
// Initially, should show data1 and not pending
|
|
165
|
-
expect(screen.getByTestId('output').textContent).toContain('data1')
|
|
166
|
-
expect(screen.getByTestId('output').textContent).toContain('not pending')
|
|
167
|
-
|
|
168
|
-
// Trigger query change to "query2"
|
|
169
|
-
act(() => {
|
|
170
|
-
screen.getByTestId('button').click()
|
|
171
|
-
})
|
|
172
|
-
|
|
173
|
-
// Immediately after clicking, deferredQueryKey is still for query1,
|
|
174
|
-
// so the hook returns data from the previous query ('data1') but isPending should now be true.
|
|
175
|
-
expect(screen.getByTestId('output').textContent).toContain('data1')
|
|
176
|
-
expect(screen.getByTestId('output').textContent).toContain('pending')
|
|
177
|
-
|
|
178
|
-
// Simulate the completion of the transition.
|
|
179
|
-
await act(async () => {
|
|
180
|
-
// Update the global variable so that getCurrent now returns data2 for the new query.
|
|
181
|
-
resolvePromise()
|
|
182
|
-
})
|
|
183
|
-
|
|
184
|
-
// Now, the component should render with the new deferred query and display new data.
|
|
185
|
-
expect(screen.getByTestId('output').textContent).toContain('data2')
|
|
186
|
-
expect(screen.getByTestId('output').textContent).toContain('not pending')
|
|
187
|
-
})
|
|
188
|
-
})
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
getQueryKey,
|
|
3
|
-
getQueryState,
|
|
4
|
-
parseQueryKey,
|
|
5
|
-
type QueryOptions,
|
|
6
|
-
resolveQuery,
|
|
7
|
-
} from '@sanity/sdk'
|
|
8
|
-
import {useEffect, useMemo, useState, useSyncExternalStore, useTransition} from 'react'
|
|
9
|
-
|
|
10
|
-
import {useSanityInstance} from '../context/useSanityInstance'
|
|
11
|
-
|
|
12
|
-
/**
|
|
13
|
-
* Executes GROQ queries against a Sanity dataset.
|
|
14
|
-
*
|
|
15
|
-
* This hook provides a convenient way to fetch and subscribe to real-time updates
|
|
16
|
-
* for your Sanity content. Changes made to the dataset’s content will trigger
|
|
17
|
-
* automatic updates.
|
|
18
|
-
*
|
|
19
|
-
* @remarks
|
|
20
|
-
* The returned `isPending` flag indicates when a React transition is in progress,
|
|
21
|
-
* which can be used to show loading states for query changes.
|
|
22
|
-
*
|
|
23
|
-
* @beta
|
|
24
|
-
* @category GROQ
|
|
25
|
-
* @param query - GROQ query string to execute
|
|
26
|
-
* @param options - Optional configuration for the query
|
|
27
|
-
* @returns Object containing the query result and a pending state flag
|
|
28
|
-
*
|
|
29
|
-
* @example Basic usage
|
|
30
|
-
* ```tsx
|
|
31
|
-
* const {data, isPending} = useQuery<Movie[]>('*[_type == "movie"]')
|
|
32
|
-
* ```
|
|
33
|
-
*
|
|
34
|
-
* @example Using parameters
|
|
35
|
-
* ```tsx
|
|
36
|
-
* // With parameters
|
|
37
|
-
* const {data} = useQuery<Movie>('*[_type == "movie" && _id == $id][0]', {
|
|
38
|
-
* params: { id: 'movie-123' }
|
|
39
|
-
* })
|
|
40
|
-
* ```
|
|
41
|
-
*
|
|
42
|
-
* @example With a loading state for transitions
|
|
43
|
-
* ```tsx
|
|
44
|
-
* const {data, isPending} = useQuery<Movie[]>('*[_type == "movie"]')
|
|
45
|
-
* return (
|
|
46
|
-
* <div>
|
|
47
|
-
* {isPending && <div>Updating...</div>}
|
|
48
|
-
* <ul>
|
|
49
|
-
* {data.map(movie => <li key={movie._id}>{movie.title}</li>)}
|
|
50
|
-
* </ul>
|
|
51
|
-
* </div>
|
|
52
|
-
* )
|
|
53
|
-
* ```
|
|
54
|
-
*
|
|
55
|
-
*/
|
|
56
|
-
export function useQuery<T>(query: string, options?: QueryOptions): {data: T; isPending: boolean} {
|
|
57
|
-
const instance = useSanityInstance(options?.resourceId)
|
|
58
|
-
// Use React's useTransition to avoid UI jank when queries change
|
|
59
|
-
const [isPending, startTransition] = useTransition()
|
|
60
|
-
|
|
61
|
-
// Get the unique key for this query and its options
|
|
62
|
-
const queryKey = getQueryKey(query, options)
|
|
63
|
-
// Use a deferred state to avoid immediate re-renders when the query changes
|
|
64
|
-
const [deferredQueryKey, setDeferredQueryKey] = useState(queryKey)
|
|
65
|
-
// Parse the deferred query key back into a query and options
|
|
66
|
-
const deferred = useMemo(() => parseQueryKey(deferredQueryKey), [deferredQueryKey])
|
|
67
|
-
|
|
68
|
-
// Create an AbortController to cancel in-flight requests when needed
|
|
69
|
-
const [ref, setRef] = useState<AbortController>(new AbortController())
|
|
70
|
-
|
|
71
|
-
// When the query or options change, start a transition to update the query
|
|
72
|
-
useEffect(() => {
|
|
73
|
-
if (queryKey === deferredQueryKey) return
|
|
74
|
-
|
|
75
|
-
startTransition(() => {
|
|
76
|
-
// Abort any in-flight requests for the previous query
|
|
77
|
-
if (ref && !ref.signal.aborted) {
|
|
78
|
-
ref.abort()
|
|
79
|
-
setRef(new AbortController())
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
setDeferredQueryKey(queryKey)
|
|
83
|
-
})
|
|
84
|
-
}, [deferredQueryKey, queryKey, ref])
|
|
85
|
-
|
|
86
|
-
// Get the state source for this query from the query store
|
|
87
|
-
const {getCurrent, subscribe} = useMemo(
|
|
88
|
-
() => getQueryState(instance, deferred.query, deferred.options),
|
|
89
|
-
[instance, deferred],
|
|
90
|
-
)
|
|
91
|
-
|
|
92
|
-
// If data isn't available yet, suspend rendering until it is
|
|
93
|
-
// This is the React Suspense integration - throwing a promise
|
|
94
|
-
// will cause React to show the nearest Suspense fallback
|
|
95
|
-
if (getCurrent() === undefined) {
|
|
96
|
-
throw resolveQuery(instance, deferred.query, {...deferred.options, signal: ref.signal})
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
// Subscribe to updates and get the current data
|
|
100
|
-
// useSyncExternalStore ensures the component re-renders when the data changes
|
|
101
|
-
const data = useSyncExternalStore(subscribe, getCurrent) as T
|
|
102
|
-
return {data, isPending}
|
|
103
|
-
}
|
|
@@ -1,163 +0,0 @@
|
|
|
1
|
-
import {createUsersStore, type ResourceType, type SanityUser} from '@sanity/sdk'
|
|
2
|
-
import {act, renderHook} from '@testing-library/react'
|
|
3
|
-
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
|
4
|
-
|
|
5
|
-
import {useSanityInstance} from '../context/useSanityInstance'
|
|
6
|
-
import {useUsers} from './useUsers'
|
|
7
|
-
|
|
8
|
-
vi.mock('@sanity/sdk')
|
|
9
|
-
vi.mock('../context/useSanityInstance')
|
|
10
|
-
|
|
11
|
-
describe('useUsers', () => {
|
|
12
|
-
const mockInstance = {}
|
|
13
|
-
const mockUser: SanityUser = {
|
|
14
|
-
profile: {
|
|
15
|
-
id: 'user1',
|
|
16
|
-
displayName: 'Test User',
|
|
17
|
-
email: 'test@test.com',
|
|
18
|
-
provider: 'test',
|
|
19
|
-
createdAt: '2021-01-01',
|
|
20
|
-
},
|
|
21
|
-
sanityUserId: 'user1',
|
|
22
|
-
memberships: [],
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
const getCurrent = vi.fn().mockReturnValue({
|
|
26
|
-
users: [],
|
|
27
|
-
totalCount: 0,
|
|
28
|
-
nextCursor: null,
|
|
29
|
-
hasMore: false,
|
|
30
|
-
initialFetchCompleted: false,
|
|
31
|
-
options: {
|
|
32
|
-
resourceType: '' as ResourceType,
|
|
33
|
-
resourceId: '',
|
|
34
|
-
limit: 100,
|
|
35
|
-
},
|
|
36
|
-
})
|
|
37
|
-
const unsubscribe = vi.fn()
|
|
38
|
-
const subscribe = vi.fn().mockReturnValue(unsubscribe)
|
|
39
|
-
const dispose = vi.fn()
|
|
40
|
-
|
|
41
|
-
const mockUsersStore: ReturnType<typeof createUsersStore> = {
|
|
42
|
-
setOptions: vi.fn(),
|
|
43
|
-
loadMore: vi.fn(),
|
|
44
|
-
resolveUsers: vi.fn(),
|
|
45
|
-
getState: vi.fn().mockReturnValue({getCurrent, subscribe}),
|
|
46
|
-
dispose,
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
beforeEach(() => {
|
|
50
|
-
vi.mocked(useSanityInstance).mockReturnValue(
|
|
51
|
-
mockInstance as unknown as ReturnType<typeof useSanityInstance>,
|
|
52
|
-
)
|
|
53
|
-
vi.mocked(createUsersStore).mockReturnValue(
|
|
54
|
-
mockUsersStore as unknown as ReturnType<typeof createUsersStore>,
|
|
55
|
-
)
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
afterEach(() => {
|
|
59
|
-
vi.clearAllMocks()
|
|
60
|
-
})
|
|
61
|
-
|
|
62
|
-
it('should initialize with given params', () => {
|
|
63
|
-
renderHook(() =>
|
|
64
|
-
useUsers({
|
|
65
|
-
resourceType: 'project',
|
|
66
|
-
resourceId: 'proj1',
|
|
67
|
-
}),
|
|
68
|
-
)
|
|
69
|
-
|
|
70
|
-
expect(createUsersStore).toHaveBeenCalledWith(mockInstance)
|
|
71
|
-
expect(mockUsersStore.setOptions).toHaveBeenCalledWith({
|
|
72
|
-
resourceType: 'project',
|
|
73
|
-
resourceId: 'proj1',
|
|
74
|
-
})
|
|
75
|
-
})
|
|
76
|
-
|
|
77
|
-
it('should subscribe to users store changes', () => {
|
|
78
|
-
renderHook(() =>
|
|
79
|
-
useUsers({
|
|
80
|
-
resourceType: 'organization',
|
|
81
|
-
resourceId: 'org1',
|
|
82
|
-
}),
|
|
83
|
-
)
|
|
84
|
-
expect(subscribe).toHaveBeenCalledTimes(1)
|
|
85
|
-
})
|
|
86
|
-
|
|
87
|
-
it('should return current users state', () => {
|
|
88
|
-
const mockState = {users: [mockUser], hasMore: true}
|
|
89
|
-
getCurrent.mockReturnValue(mockState)
|
|
90
|
-
|
|
91
|
-
const {result} = renderHook(() =>
|
|
92
|
-
useUsers({
|
|
93
|
-
resourceType: 'project',
|
|
94
|
-
resourceId: 'proj1',
|
|
95
|
-
}),
|
|
96
|
-
)
|
|
97
|
-
expect(result.current).toMatchObject(mockState)
|
|
98
|
-
})
|
|
99
|
-
|
|
100
|
-
it('should call loadMore when loadMore is invoked', () => {
|
|
101
|
-
const {result} = renderHook(() =>
|
|
102
|
-
useUsers({
|
|
103
|
-
resourceType: 'project',
|
|
104
|
-
resourceId: 'proj1',
|
|
105
|
-
}),
|
|
106
|
-
)
|
|
107
|
-
|
|
108
|
-
act(() => {
|
|
109
|
-
result.current.loadMore()
|
|
110
|
-
})
|
|
111
|
-
|
|
112
|
-
expect(mockUsersStore.loadMore).toHaveBeenCalled()
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
it('should update options when params change', () => {
|
|
116
|
-
const initialParams = {resourceType: 'project' as ResourceType, resourceId: 'proj1'}
|
|
117
|
-
const {rerender} = renderHook(({params}) => useUsers(params), {
|
|
118
|
-
initialProps: {params: initialParams},
|
|
119
|
-
})
|
|
120
|
-
|
|
121
|
-
const newParams = {resourceType: 'organization' as ResourceType, resourceId: 'org1'}
|
|
122
|
-
rerender({params: newParams})
|
|
123
|
-
|
|
124
|
-
expect(mockUsersStore.setOptions).toHaveBeenCalledWith(newParams)
|
|
125
|
-
})
|
|
126
|
-
|
|
127
|
-
it('should resolve users if initial fetch not completed', () => {
|
|
128
|
-
getCurrent.mockReturnValue({initialFetchCompleted: false})
|
|
129
|
-
|
|
130
|
-
renderHook(() =>
|
|
131
|
-
useUsers({
|
|
132
|
-
resourceType: 'project',
|
|
133
|
-
resourceId: 'proj1',
|
|
134
|
-
}),
|
|
135
|
-
)
|
|
136
|
-
expect(mockUsersStore.resolveUsers).toHaveBeenCalled()
|
|
137
|
-
})
|
|
138
|
-
|
|
139
|
-
it('should not resolve users if initial fetch already completed', () => {
|
|
140
|
-
getCurrent.mockReturnValue({initialFetchCompleted: true})
|
|
141
|
-
|
|
142
|
-
renderHook(() =>
|
|
143
|
-
useUsers({
|
|
144
|
-
resourceType: 'project',
|
|
145
|
-
resourceId: 'proj1',
|
|
146
|
-
}),
|
|
147
|
-
)
|
|
148
|
-
expect(mockUsersStore.resolveUsers).not.toHaveBeenCalled()
|
|
149
|
-
})
|
|
150
|
-
|
|
151
|
-
it('should clean up store on unmount', () => {
|
|
152
|
-
const {unmount} = renderHook(() =>
|
|
153
|
-
useUsers({
|
|
154
|
-
resourceType: 'project',
|
|
155
|
-
resourceId: 'proj1',
|
|
156
|
-
}),
|
|
157
|
-
)
|
|
158
|
-
|
|
159
|
-
unmount()
|
|
160
|
-
expect(mockUsersStore.dispose).toHaveBeenCalled()
|
|
161
|
-
expect(unsubscribe).toHaveBeenCalled()
|
|
162
|
-
})
|
|
163
|
-
})
|
|
@@ -1,107 +0,0 @@
|
|
|
1
|
-
import {createUsersStore, type ResourceType, type SanityUser} from '@sanity/sdk'
|
|
2
|
-
import {useCallback, useEffect, useState, useSyncExternalStore} from 'react'
|
|
3
|
-
|
|
4
|
-
import {useSanityInstance} from '../context/useSanityInstance'
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* @public
|
|
8
|
-
* @category Types
|
|
9
|
-
*/
|
|
10
|
-
export interface UseUsersParams {
|
|
11
|
-
/**
|
|
12
|
-
* The type of resource to fetch users for.
|
|
13
|
-
*/
|
|
14
|
-
resourceType: ResourceType
|
|
15
|
-
/**
|
|
16
|
-
* The ID of the resource to fetch users for.
|
|
17
|
-
*/
|
|
18
|
-
resourceId: string
|
|
19
|
-
/**
|
|
20
|
-
* The limit of users to fetch.
|
|
21
|
-
*/
|
|
22
|
-
limit?: number
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* @public
|
|
27
|
-
* @category Types
|
|
28
|
-
*/
|
|
29
|
-
export interface UseUsersResult {
|
|
30
|
-
/**
|
|
31
|
-
* The users fetched.
|
|
32
|
-
*/
|
|
33
|
-
users: SanityUser[]
|
|
34
|
-
/**
|
|
35
|
-
* Whether there are more users to fetch.
|
|
36
|
-
*/
|
|
37
|
-
hasMore: boolean
|
|
38
|
-
/**
|
|
39
|
-
* Load more users.
|
|
40
|
-
*/
|
|
41
|
-
loadMore: () => void
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
/**
|
|
45
|
-
*
|
|
46
|
-
* @public
|
|
47
|
-
*
|
|
48
|
-
* Retrieves the users for a given resource (either a project or an organization).
|
|
49
|
-
*
|
|
50
|
-
* @category Users
|
|
51
|
-
* @param params - The resource type and its ID, and the limit of users to fetch
|
|
52
|
-
* @returns A list of users, a boolean indicating whether there are more users to fetch, and a function to load more users
|
|
53
|
-
*
|
|
54
|
-
* @example
|
|
55
|
-
* ```
|
|
56
|
-
* const { users, hasMore, loadMore } = useUsers({
|
|
57
|
-
* resourceType: 'organization',
|
|
58
|
-
* resourceId: 'my-org-id',
|
|
59
|
-
* limit: 10,
|
|
60
|
-
* })
|
|
61
|
-
*
|
|
62
|
-
* return (
|
|
63
|
-
* <div>
|
|
64
|
-
* {users.map(user => (
|
|
65
|
-
* <figure key={user.sanityUserId}>
|
|
66
|
-
* <img src={user.profile.imageUrl} alt='' />
|
|
67
|
-
* <figcaption>{user.profile.displayName}</figcaption>
|
|
68
|
-
* <address>{user.profile.email}</address>
|
|
69
|
-
* </figure>
|
|
70
|
-
* ))}
|
|
71
|
-
* {hasMore && <button onClick={loadMore}>Load More</button>}
|
|
72
|
-
* </div>
|
|
73
|
-
* )
|
|
74
|
-
* ```
|
|
75
|
-
*/
|
|
76
|
-
export function useUsers(params: UseUsersParams): UseUsersResult {
|
|
77
|
-
const instance = useSanityInstance(params.resourceId)
|
|
78
|
-
const [store] = useState(() => createUsersStore(instance))
|
|
79
|
-
|
|
80
|
-
useEffect(() => {
|
|
81
|
-
store.setOptions({
|
|
82
|
-
resourceType: params.resourceType,
|
|
83
|
-
resourceId: params.resourceId,
|
|
84
|
-
})
|
|
85
|
-
}, [params.resourceType, params.resourceId, store])
|
|
86
|
-
|
|
87
|
-
const subscribe = useCallback(
|
|
88
|
-
(onStoreChanged: () => void) => {
|
|
89
|
-
if (store.getState().getCurrent().initialFetchCompleted === false) {
|
|
90
|
-
store.resolveUsers()
|
|
91
|
-
}
|
|
92
|
-
const unsubscribe = store.getState().subscribe(onStoreChanged)
|
|
93
|
-
|
|
94
|
-
return () => {
|
|
95
|
-
unsubscribe()
|
|
96
|
-
store.dispose()
|
|
97
|
-
}
|
|
98
|
-
},
|
|
99
|
-
[store],
|
|
100
|
-
)
|
|
101
|
-
|
|
102
|
-
const getSnapshot = useCallback(() => store.getState().getCurrent(), [store])
|
|
103
|
-
|
|
104
|
-
const {users, hasMore} = useSyncExternalStore(subscribe, getSnapshot) || {}
|
|
105
|
-
|
|
106
|
-
return {users, hasMore, loadMore: store.loadMore}
|
|
107
|
-
}
|
package/src/utils/getEnv.ts
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
// Local type declaration for Remix
|
|
2
|
-
type WindowWithEnv = Window &
|
|
3
|
-
typeof globalThis & {
|
|
4
|
-
ENV?: Record<string, unknown>
|
|
5
|
-
}
|
|
6
|
-
|
|
7
|
-
type KnownEnvVar = 'DEV' | 'PKG_VERSION'
|
|
8
|
-
|
|
9
|
-
export function getEnv(key: KnownEnvVar): unknown {
|
|
10
|
-
if (typeof import.meta !== 'undefined' && import.meta.env) {
|
|
11
|
-
// Vite environment variables
|
|
12
|
-
return (import.meta.env as unknown as Record<string, unknown>)[key]
|
|
13
|
-
} else if (typeof process !== 'undefined' && process.env) {
|
|
14
|
-
// Node.js or server-side environment variables
|
|
15
|
-
return process.env[key]
|
|
16
|
-
} else if (typeof window !== 'undefined' && (window as WindowWithEnv).ENV) {
|
|
17
|
-
// Remix-style client-side environment variables
|
|
18
|
-
return (window as WindowWithEnv).ENV?.[key]
|
|
19
|
-
}
|
|
20
|
-
return undefined
|
|
21
|
-
}
|
package/src/version.ts
DELETED