diff --git a/src/options.js b/src/options.js index 387daa94fb..ce40aa6997 100644 --- a/src/options.js +++ b/src/options.js @@ -132,6 +132,7 @@ export function keyword(input, name, allowed) { // Promotes the specified data to an array as needed. export function arrayify(data) { + if (typeof data?.transform === "function") data = data.transform(); return data == null || data instanceof Array || data instanceof TypedArray ? data : Array.from(data); } diff --git a/src/transforms/group.js b/src/transforms/group.js index 07ae348359..542cf76d96 100644 --- a/src/transforms/group.js +++ b/src/transforms/group.js @@ -66,7 +66,7 @@ function groupn( x, // optionally group on x y, // optionally group on y { - data: reduceData = reduceIdentity, + data: reduceData = reduceIdentityLazy, filter, sort, reverse, @@ -153,6 +153,7 @@ function groupn( groupFacets.push(groupFacet); } maybeSort(groupFacets, sort, reverse); + if (reduceData === reduceIdentityLazy) groupData.transform = () => groupData.map((d) => d()); return {data: groupData, facets: groupFacets}; }), ...(!hasOutput(outputs, "x") && (GX ? {x: GX} : {x1, x2})), @@ -236,6 +237,7 @@ export function maybeGroup(I, X) { export function maybeReduce(reduce, value, fallback = invalidReduce) { if (reduce == null) return fallback(reduce); if (typeof reduce.reduceIndex === "function") return reduce; + if (typeof reduce.reduceIndexLazy === "function") return reduce; if (typeof reduce.reduce === "function" && isObject(reduce)) return reduceReduce(reduce); // N.B. array.reduce if (typeof reduce === "function") return reduceFunction(reduce); if (/^p\d{2}$/i.test(reduce)) return reduceAccessor(percentile(reduce)); @@ -366,6 +368,12 @@ export const reduceIdentity = { } }; +const reduceIdentityLazy = { + reduceIndex(I, X) { + return () => take(X, I); + } +}; + export const reduceFirst = { reduceIndex(I, X) { return X[I[0]];