Skip to content

Commit 8cf7538

Browse files
committed
Improve types for some array internals
1 parent 59637e7 commit 8cf7538

File tree

3 files changed

+39
-23
lines changed

3 files changed

+39
-23
lines changed

packages/@ember/-internals/metal/lib/array.ts

+23-13
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,13 @@
1-
import { Array as EmberArray, MutableArray, NativeArray } from '@ember/-internals/runtime';
1+
import { Array as EmberArray, MutableArray } from '@ember/-internals/runtime';
2+
import { assert } from '@ember/debug';
23
import { arrayContentDidChange, arrayContentWillChange } from './array_events';
34
import { addListener, removeListener } from './events';
45

56
const EMPTY_ARRAY = Object.freeze([]);
67

7-
interface ObservedObject<T> extends EmberArray<T> {
8+
type ObservedArray<T> = (T[] | EmberArray<T>) & ObservedObject;
9+
10+
interface ObservedObject {
811
_revalidate?: () => void;
912
}
1013

@@ -16,16 +19,23 @@ export function objectAt<T>(array: T[] | EmberArray<T>, index: number): T | unde
1619
}
1720
}
1821

22+
// Ideally, we'd use MutableArray.detect but for unknown reasons this causes
23+
// the node tests to fail strangely.
24+
function isMutableArray<T>(obj: unknown): obj is MutableArray<T> {
25+
return obj != null && typeof (obj as MutableArray<T>).replace === 'function';
26+
}
27+
1928
export function replace<T>(
20-
array: NativeArray<T> | MutableArray<T>,
29+
array: T[] | MutableArray<T>,
2130
start: number,
2231
deleteCount: number,
2332
items: T[] = EMPTY_ARRAY as []
2433
): void {
25-
if (Array.isArray(array)) {
26-
replaceInNativeArray(array, start, deleteCount, items);
27-
} else {
34+
if (isMutableArray(array)) {
2835
array.replace(start, deleteCount, items);
36+
} else {
37+
assert('Can only replace content of a native array or MutableArray', Array.isArray(array));
38+
replaceInNativeArray(array, start, deleteCount, items);
2939
}
3040
}
3141

@@ -34,7 +44,7 @@ const CHUNK_SIZE = 60000;
3444
// To avoid overflowing the stack, we splice up to CHUNK_SIZE items at a time.
3545
// See https://code.google.com/p/chromium/issues/detail?id=56588 for more details.
3646
export function replaceInNativeArray<T>(
37-
array: T[] | NativeArray<T>,
47+
array: T[],
3848
start: number,
3949
deleteCount: number,
4050
items: ReadonlyArray<T>
@@ -61,18 +71,18 @@ interface ArrayObserverOptions {
6171
}
6272

6373
type Operation<T> = (
64-
obj: ObservedObject<T>,
74+
obj: ObservedArray<T>,
6575
eventName: string,
6676
target: object | Function | null,
6777
callbackName: string
6878
) => void;
6979

7080
function arrayObserversHelper<T>(
71-
obj: ObservedObject<T>,
81+
obj: ObservedArray<T>,
7282
target: object | Function | null,
7383
opts: ArrayObserverOptions,
7484
operation: Operation<T>
75-
): ObservedObject<T> {
85+
): ObservedArray<T> {
7686
let { willChange, didChange } = opts;
7787

7888
operation(obj, '@array:before', target, willChange);
@@ -91,14 +101,14 @@ export function addArrayObserver<T>(
91101
array: EmberArray<T>,
92102
target: object | Function | null,
93103
opts: ArrayObserverOptions
94-
): ObservedObject<T> {
104+
): ObservedArray<T> {
95105
return arrayObserversHelper(array, target, opts, addListener);
96106
}
97107

98108
export function removeArrayObserver<T>(
99-
array: EmberArray<T>,
109+
array: T[] | EmberArray<T>,
100110
target: object | Function | null,
101111
opts: ArrayObserverOptions
102-
): ObservedObject<T> {
112+
): ObservedArray<T> {
103113
return arrayObserversHelper(array, target, opts, removeListener);
104114
}

packages/@ember/-internals/runtime/lib/mixins/array.d.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -107,4 +107,4 @@ export function uniqBy<T>(
107107
array: T[] | EmberArray<T>,
108108
keyOrFunc: string | ((item: T) => unknown)
109109
): T[] | EmberArray<T>;
110-
export function isArray(obj: unknown): boolean;
110+
export function isArray(obj: unknown): obj is ArrayLike<unknown> | EmberArray<unknown>;

packages/@ember/-internals/runtime/lib/system/array_proxy.ts

+15-9
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ import {
1616
} from '@ember/-internals/metal';
1717
import { isObject } from '@ember/-internals/utils';
1818
import EmberObject from './object';
19-
import EmberArray, { isArray, MutableArray } from '../mixins/array';
19+
import EmberArray, { MutableArray } from '../mixins/array';
2020
import { assert } from '@ember/debug';
2121
import { setCustomTagFor } from '@glimmer/manager';
2222
import {
@@ -30,7 +30,7 @@ import {
3030
} from '@glimmer/validator';
3131
import { PropertyDidChange } from '@ember/-internals/metal/lib/property_events';
3232

33-
function isMutable<T>(obj: EmberArray<T>): obj is MutableArray<T> {
33+
function isMutable<T>(obj: T[] | EmberArray<T>): obj is T[] | MutableArray<T> {
3434
return Array.isArray(obj) || typeof (obj as MutableArray<T>).replace === 'function';
3535
}
3636

@@ -116,8 +116,8 @@ function customTagForArrayProxy(proxy: object, key: string) {
116116
@public
117117
*/
118118
// eslint-disable-next-line @typescript-eslint/no-empty-interface, @typescript-eslint/no-unused-vars
119-
interface ArrayProxy<T, C extends EmberArray<T> = EmberArray<T>> extends MutableArray<T> {}
120-
class ArrayProxy<T, C extends EmberArray<T> = EmberArray<T>>
119+
interface ArrayProxy<T, C extends EmberArray<T> | T[] = T[]> extends MutableArray<T> {}
120+
class ArrayProxy<T, C extends EmberArray<T> | T[] = T[]>
121121
extends EmberObject
122122
implements PropertyDidChange
123123
{
@@ -218,15 +218,15 @@ class ArrayProxy<T, C extends EmberArray<T> = EmberArray<T>>
218218
@method replaceContent
219219
@param {Number} idx The starting index
220220
@param {Number} amt The number of items to remove from the content.
221-
@param {EmberArray} objects Optional array of objects to insert.
221+
@param {Array} objects Optional array of objects to insert.
222222
@return {void}
223223
@public
224224
*/
225225
replaceContent(idx: number, amt: number, objects?: T[]) {
226226
let content = get(this, 'content');
227-
assert('[BUG] Called objectAtContent without content', content);
227+
assert('[BUG] Called replaceContent without content', content);
228228
assert('Mutating a non-mutable array is not allowed', isMutable(content));
229-
content.replace(idx, amt, objects);
229+
replace<T>(content, idx, amt, objects);
230230
}
231231

232232
// Overriding objectAt is not supported.
@@ -312,8 +312,14 @@ class ArrayProxy<T, C extends EmberArray<T> = EmberArray<T>>
312312
// @ts-expect-error This check is still good for ensuring correctness
313313
assert("Can't set ArrayProxy's content to itself", arrangedContent !== this);
314314
assert(
315-
`ArrayProxy expects an Array or ArrayProxy, but you passed ${typeof arrangedContent}`,
316-
isArray(arrangedContent) || (arrangedContent as any).isDestroyed
315+
`ArrayProxy expects a native Array, EmberArray, or ArrayProxy, but you passed ${typeof arrangedContent}`,
316+
(function (arr: unknown): arr is EmberArray<unknown> {
317+
return Array.isArray(arr) || EmberArray.detect(arr);
318+
})(arrangedContent)
319+
);
320+
assert(
321+
'ArrayProxy expected its contents to not be destroyed',
322+
!(arrangedContent as any).isDestroyed
317323
);
318324

319325
addArrayObserver(arrangedContent, this, ARRAY_OBSERVER_MAPPING);

0 commit comments

Comments
 (0)