Skip to content

Commit 8886ca3

Browse files
authored
Merge pull request #541 from ckeditor/ck/fix-undefined-behavior
Fix: Behavior of `useCKEditorCloud` hook is now consistent on `Vite` and `Next` runtimes while changing properties. Internal: Fix incorrect typings of `useAsyncCallback` and `useAsyncValue` callback arguments types. Tests: Add missing cleanup methods to few tests.
2 parents 8f5232c + bd261f1 commit 8886ca3

12 files changed

+45
-29
lines changed

src/hooks/useAsyncCallback.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ import { useRefSafeCallback } from './useRefSafeCallback.js';
3434
* ```
3535
*/
3636
export const useAsyncCallback = <A extends Array<unknown>, R>(
37-
callback: ( ...args: Array<A> ) => Promise<R>
37+
callback: ( ...args: A ) => Promise<R>
3838
): AsyncCallbackHookResult<A, R> => {
3939
// The state of the asynchronous callback.
4040
const [ asyncState, setAsyncState ] = useState<AsyncCallbackState<R>>( {
@@ -50,7 +50,7 @@ export const useAsyncCallback = <A extends Array<unknown>, R>(
5050
const prevExecutionUIDRef = useRef<string | null>( null );
5151

5252
// The asynchronous executor function, which is a wrapped version of the original callback.
53-
const asyncExecutor = useRefSafeCallback( async ( ...args: Array<any> ) => {
53+
const asyncExecutor = useRefSafeCallback( async ( ...args: A ) => {
5454
if ( unmountedRef.current || isSSR() ) {
5555
return null;
5656
}
@@ -101,7 +101,7 @@ export const useAsyncCallback = <A extends Array<unknown>, R>(
101101
* Represents the result of the `useAsyncCallback` hook.
102102
*/
103103
export type AsyncCallbackHookResult<A extends Array<unknown>, R> = [
104-
( ...args: Array<A> ) => Promise<R | null>,
104+
( ...args: A ) => Promise<R | null>,
105105
AsyncCallbackState<R>
106106
];
107107

src/hooks/useAsyncValue.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ import { useAsyncCallback, type AsyncCallbackState } from './useAsyncCallback.js
3838
* ```
3939
*/
4040
export const useAsyncValue = <A extends Array<unknown>, R>(
41-
callback: ( ...args: Array<A> ) => Promise<R>,
41+
callback: ( ...args: A ) => Promise<R>,
4242
deps: DependencyList
4343
): AsyncValueHookResult<R> => {
4444
const [ asyncCallback, asyncState ] = useAsyncCallback( callback );

src/hooks/useInstantEffect.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { useRef, type DependencyList } from 'react';
6+
import { useState, type DependencyList } from 'react';
77
import { shallowCompareArrays } from '@ckeditor/ckeditor5-integrations-common';
88

99
/**
@@ -13,10 +13,10 @@ import { shallowCompareArrays } from '@ckeditor/ckeditor5-integrations-common';
1313
* @param deps The dependency list.
1414
*/
1515
export const useInstantEffect = ( fn: VoidFunction, deps: DependencyList ): void => {
16-
const prevDeps = useRef<any>( null );
16+
const [ prevDeps, setDeps ] = useState<any>( null );
1717

18-
if ( !shallowCompareArrays( prevDeps.current, deps ) ) {
19-
prevDeps.current = [ ...deps ];
18+
if ( !shallowCompareArrays( prevDeps, deps ) ) {
2019
fn();
20+
setDeps( [ ...deps ] );
2121
}
2222
};

tests/cloud/useCKEditorCloud.test.tsx

+6-3
Original file line numberDiff line numberDiff line change
@@ -3,16 +3,19 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { beforeEach, describe, expect, expectTypeOf, it } from 'vitest';
7-
import { renderHook, waitFor, act } from '@testing-library/react';
6+
import { afterEach, describe, expect, expectTypeOf, it } from 'vitest';
7+
import { renderHook, waitFor, act, cleanup } from '@testing-library/react';
88

99
import type { CKEditorCloudConfig } from '@ckeditor/ckeditor5-integrations-common';
1010
import { removeAllCkCdnResources } from '@ckeditor/ckeditor5-integrations-common/test-utils';
1111

1212
import useCKEditorCloud from '../../src/cloud/useCKEditorCloud.js';
1313

1414
describe( 'useCKEditorCloud', () => {
15-
beforeEach( removeAllCkCdnResources );
15+
afterEach( () => {
16+
cleanup();
17+
removeAllCkCdnResources();
18+
} );
1619

1720
it( 'should load CKEditor bundles from CDN', async () => {
1821
const { result } = renderHook( () => useCKEditorCloud( {

tests/cloud/withCKEditorCloud.test.tsx

+3-4
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
*/
55

66
import React, { type MutableRefObject } from 'react';
7-
import { afterEach, beforeEach, describe, expect, it } from 'vitest';
7+
import { afterEach, describe, expect, it } from 'vitest';
88
import { cleanup, render } from '@testing-library/react';
99

1010
import { createDefer } from '@ckeditor/ckeditor5-integrations-common';
@@ -17,9 +17,8 @@ describe( 'withCKEditorCloud', () => {
1717
current: null
1818
};
1919

20-
afterEach( cleanup );
21-
22-
beforeEach( () => {
20+
afterEach( () => {
21+
cleanup();
2322
removeAllCkCdnResources();
2423
lastRenderedMockProps.current = null;
2524
} );

tests/hooks/useAsyncCallback.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,14 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it, vi } from 'vitest';
7-
import { renderHook, act, waitFor } from '@testing-library/react';
6+
import { afterEach, describe, expect, it, vi } from 'vitest';
7+
import { renderHook, act, waitFor, cleanup } from '@testing-library/react';
88
import { useAsyncCallback } from '../../src/hooks/useAsyncCallback.js';
99
import { timeout } from '../_utils/timeout.js';
1010

1111
describe( 'useAsyncCallback', () => {
12+
afterEach( cleanup );
13+
1214
it( 'should execute the callback and update the state correctly when the callback resolves', async () => {
1315
const fetchData = vi.fn().mockResolvedValue( 'data' );
1416

tests/hooks/useAsyncValue.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it } from 'vitest';
7-
import { renderHook, waitFor } from '@testing-library/react';
6+
import { afterEach, describe, expect, it } from 'vitest';
7+
import { cleanup, renderHook, waitFor } from '@testing-library/react';
88
import { useAsyncValue } from '../../src/hooks/useAsyncValue.js';
99

1010
describe( 'useAsyncValue', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should return a mutable ref object', async () => {
1214
const { result } = renderHook( () => useAsyncValue( async () => 123, [] ) );
1315

tests/hooks/useInstantEditorEffect.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it, vi } from 'vitest';
7-
import { renderHook } from '@testing-library/react';
6+
import { afterEach, describe, expect, it, vi } from 'vitest';
7+
import { cleanup, renderHook } from '@testing-library/react';
88
import { useInstantEditorEffect } from '../../src/hooks/useInstantEditorEffect.js';
99

1010
describe( 'useInstantEditorEffect', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should execute the provided function after mounting of editor', () => {
1214
const semaphore = {
1315
runAfterMount: vi.fn()

tests/hooks/useInstantEffect.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it, vi } from 'vitest';
7-
import { renderHook } from '@testing-library/react';
6+
import { afterEach, describe, expect, it, vi } from 'vitest';
7+
import { cleanup, renderHook } from '@testing-library/react';
88
import { useInstantEffect } from '../../src/hooks/useInstantEffect.js';
99

1010
describe( 'useInstantEffect', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should call the effect function when dependencies change', () => {
1214
const effectFn = vi.fn();
1315
const { rerender } = renderHook( deps => useInstantEffect( effectFn, deps ), {

tests/hooks/useIsMountedRef.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it } from 'vitest';
7-
import { renderHook } from '@testing-library/react';
6+
import { afterEach, describe, expect, it } from 'vitest';
7+
import { cleanup, renderHook } from '@testing-library/react';
88
import { useIsMountedRef } from '../../src/hooks/useIsMountedRef.js';
99

1010
describe( 'useIsMountedRef', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should return a mutable ref object', () => {
1214
const { result } = renderHook( () => useIsMountedRef() );
1315

tests/hooks/useIsUnmountedRef.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { describe, expect, it } from 'vitest';
7-
import { renderHook } from '@testing-library/react';
6+
import { afterEach, describe, expect, it } from 'vitest';
7+
import { cleanup, renderHook } from '@testing-library/react';
88
import { useIsUnmountedRef } from '../../src/hooks/useIsUnmountedRef.js';
99

1010
describe( 'useIsUnmountedRef', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should return a mutable ref object', () => {
1214
const { result } = renderHook( () => useIsUnmountedRef() );
1315

tests/hooks/useRefSafeCallback.test.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
* For licensing, see LICENSE.md.
44
*/
55

6-
import { expect, it, describe, vi } from 'vitest';
7-
import { renderHook, act } from '@testing-library/react';
6+
import { expect, it, describe, vi, afterEach } from 'vitest';
7+
import { renderHook, act, cleanup } from '@testing-library/react';
88
import { useRefSafeCallback } from '../../src/hooks/useRefSafeCallback.js';
99

1010
describe( 'useRefSafeCallback', () => {
11+
afterEach( cleanup );
12+
1113
it( 'should return a function', () => {
1214
const { result } = renderHook( () => useRefSafeCallback( () => {} ) );
1315

0 commit comments

Comments
 (0)