forked from observablehq/plot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathselect.js
110 lines (97 loc) · 2.94 KB
/
select.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
import {greatest, group, least} from "d3";
import {maybeZ, valueof} from "../options.js";
import {basic} from "./basic.js";
/** @jsdoc select */
export function select(selector, options = {}) {
// If specified selector is a string or function, it’s a selector without an
// input channel such as first or last.
if (typeof selector === "string") {
switch (selector.toLowerCase()) {
case "first":
return selectFirst(options);
case "last":
return selectLast(options);
}
}
if (typeof selector === "function") {
return selectChannel(null, selector, options);
}
// Otherwise the selector is an option {name: value} where name is a channel
// name and value is a selector definition that additionally takes the given
// channel values as input. The selector object must have exactly one key.
let key, value;
for (key in selector) {
if (value !== undefined) throw new Error("ambiguous selector; multiple inputs");
value = maybeSelector(selector[key]);
}
if (value === undefined) throw new Error(`invalid selector: ${selector}`);
return selectChannel(key, value, options);
}
function maybeSelector(selector) {
if (typeof selector === "function") return selector;
switch (`${selector}`.toLowerCase()) {
case "min":
return selectorMin;
case "max":
return selectorMax;
}
throw new Error(`unknown selector: ${selector}`);
}
/** @jsdoc selectFirst */
export function selectFirst(options) {
return selectChannel(null, selectorFirst, options);
}
/** @jsdoc selectLast */
export function selectLast(options) {
return selectChannel(null, selectorLast, options);
}
/** @jsdoc selectMinX */
export function selectMinX(options) {
return selectChannel("x", selectorMin, options);
}
/** @jsdoc selectMinY */
export function selectMinY(options) {
return selectChannel("y", selectorMin, options);
}
/** @jsdoc selectMaxX */
export function selectMaxX(options) {
return selectChannel("x", selectorMax, options);
}
/** @jsdoc selectMaxY */
export function selectMaxY(options) {
return selectChannel("y", selectorMax, options);
}
function* selectorFirst(I) {
yield I[0];
}
function* selectorLast(I) {
yield I[I.length - 1];
}
function* selectorMin(I, X) {
yield least(I, (i) => X[i]);
}
function* selectorMax(I, X) {
yield greatest(I, (i) => X[i]);
}
function selectChannel(v, selector, options) {
if (v != null) {
if (options[v] == null) throw new Error(`missing channel: ${v}`);
v = options[v];
}
const z = maybeZ(options);
return basic(options, (data, facets) => {
const Z = valueof(data, z);
const V = valueof(data, v);
const selectFacets = [];
for (const facet of facets) {
const selectFacet = [];
for (const I of Z ? group(facet, (i) => Z[i]).values() : [facet]) {
for (const i of selector(I, V)) {
selectFacet.push(i);
}
}
selectFacets.push(selectFacet);
}
return {data, facets: selectFacets};
});
}