@d3plus/dom 3.0.0-alpha.0

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.
@@ -0,0 +1,463 @@
1
+ /*
2
+ @d3plus/dom v3.0.0
3
+ JavaScript functions for manipulating and analyzing DOM elements.
4
+ Copyright (c) 2025 D3plus - https://d3plus.org
5
+ @license MIT
6
+ */
7
+
8
+ (function (factory) {
9
+ typeof define === 'function' && define.amd ? define(factory) :
10
+ factory();
11
+ })((function () { 'use strict';
12
+
13
+ if (typeof window !== "undefined") {
14
+ (function () {
15
+ try {
16
+ if (typeof SVGElement === 'undefined' || Boolean(SVGElement.prototype.innerHTML)) {
17
+ return;
18
+ }
19
+ } catch (e) {
20
+ return;
21
+ }
22
+
23
+ function serializeNode (node) {
24
+ switch (node.nodeType) {
25
+ case 1:
26
+ return serializeElementNode(node);
27
+ case 3:
28
+ return serializeTextNode(node);
29
+ case 8:
30
+ return serializeCommentNode(node);
31
+ }
32
+ }
33
+
34
+ function serializeTextNode (node) {
35
+ return node.textContent.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;');
36
+ }
37
+
38
+ function serializeCommentNode (node) {
39
+ return '<!--' + node.nodeValue + '-->'
40
+ }
41
+
42
+ function serializeElementNode (node) {
43
+ var output = '';
44
+
45
+ output += '<' + node.tagName;
46
+
47
+ if (node.hasAttributes()) {
48
+ [].forEach.call(node.attributes, function(attrNode) {
49
+ output += ' ' + attrNode.name + '="' + attrNode.value + '"';
50
+ });
51
+ }
52
+
53
+ output += '>';
54
+
55
+ if (node.hasChildNodes()) {
56
+ [].forEach.call(node.childNodes, function(childNode) {
57
+ output += serializeNode(childNode);
58
+ });
59
+ }
60
+
61
+ output += '</' + node.tagName + '>';
62
+
63
+ return output;
64
+ }
65
+
66
+ Object.defineProperty(SVGElement.prototype, 'innerHTML', {
67
+ get: function () {
68
+ var output = '';
69
+
70
+ [].forEach.call(this.childNodes, function(childNode) {
71
+ output += serializeNode(childNode);
72
+ });
73
+
74
+ return output;
75
+ },
76
+ set: function (markup) {
77
+ while (this.firstChild) {
78
+ this.removeChild(this.firstChild);
79
+ }
80
+
81
+ try {
82
+ var dXML = new DOMParser();
83
+ dXML.async = false;
84
+
85
+ var sXML = '<svg xmlns=\'http://www.w3.org/2000/svg\' xmlns:xlink=\'http://www.w3.org/1999/xlink\'>' + markup + '</svg>';
86
+ var svgDocElement = dXML.parseFromString(sXML, 'text/xml').documentElement;
87
+
88
+ [].forEach.call(svgDocElement.childNodes, function(childNode) {
89
+ this.appendChild(this.ownerDocument.importNode(childNode, true));
90
+ }.bind(this));
91
+ } catch (e) {
92
+ throw new Error('Error parsing markup string');
93
+ }
94
+ }
95
+ });
96
+
97
+ Object.defineProperty(SVGElement.prototype, 'innerSVG', {
98
+ get: function () {
99
+ return this.innerHTML;
100
+ },
101
+ set: function (markup) {
102
+ this.innerHTML = markup;
103
+ }
104
+ });
105
+
106
+ })();
107
+ }
108
+
109
+ }));
110
+
111
+ (function (global, factory) {
112
+ typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('d3-selection'), require('d3-transition'), require('@d3plus/text')) :
113
+ typeof define === 'function' && define.amd ? define('@d3plus/dom', ['exports', 'd3-selection', 'd3-transition', '@d3plus/text'], factory) :
114
+ (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.d3plus = {}, global.d3Selection, global.d3Transition, global.text));
115
+ })(this, (function (exports, d3Selection, d3Transition, text) { 'use strict';
116
+
117
+ /**
118
+ @function isObject
119
+ @desc Detects if a variable is a javascript Object.
120
+ @param {*} item
121
+ */ function isObject(item) {
122
+ return item && typeof item === "object" && (typeof window === "undefined" || item !== window && item !== window.document && !(item instanceof Element)) && !Array.isArray(item) ? true : false;
123
+ }
124
+
125
+ /**
126
+ @function validObject
127
+ @desc Determines if the object passed is the document or window.
128
+ @param {Object} obj
129
+ @private
130
+ */ function validObject(obj) {
131
+ if (typeof window === "undefined") return true;
132
+ else return obj !== window && obj !== document;
133
+ }
134
+ /**
135
+ @function assign
136
+ @desc A deeply recursive version of `Object.assign`.
137
+ @param {...Object} objects
138
+ @example <caption>this</caption>
139
+ assign({id: "foo", deep: {group: "A"}}, {id: "bar", deep: {value: 20}}));
140
+ @example <caption>returns this</caption>
141
+ {id: "bar", deep: {group: "A", value: 20}}
142
+ */ function assign(...objects) {
143
+ const target = objects[0];
144
+ for(let i = 1; i < objects.length; i++){
145
+ const source = objects[i];
146
+ if (!isObject(source)) continue;
147
+ Object.keys(source).forEach((prop)=>{
148
+ const value = source[prop];
149
+ if (isObject(value) && validObject(value)) {
150
+ if (Object.prototype.hasOwnProperty.call(target, prop) && isObject(target[prop])) target[prop] = assign({}, target[prop], value);
151
+ else target[prop] = assign({}, value);
152
+ } else if (Array.isArray(value)) target[prop] = value.slice();
153
+ else target[prop] = value;
154
+ });
155
+ }
156
+ return target;
157
+ }
158
+
159
+ /**
160
+ @function attrize
161
+ @desc Applies each key/value in an object as an attr.
162
+ @param {D3selection} elem The D3 element to apply the styles to.
163
+ @param {Object} attrs An object of key/value attr pairs.
164
+ */ function attrize(e, a = {}) {
165
+ for(const k in a)if (({}).hasOwnProperty.call(a, k)) e.attr(k, a[k]);
166
+ }
167
+
168
+ /**
169
+ @function date
170
+ @summary Parses numbers and strings to valid Javascript Date objects.
171
+ @description Returns a javascript Date object for a given a Number (representing either a 4-digit year or milliseconds since epoch), a String representing a Quarter (ie. "Q2 1987", mapping to the last day in that quarter), or a String that is in [valid dateString format](http://dygraphs.com/date-formats.html). Besides the 4-digit year parsing, this function is useful when needing to parse negative (BC) years, which the vanilla Date object cannot parse.
172
+ @param {Number|String} *date*
173
+ */ function date(d) {
174
+ // returns if falsey or already Date object
175
+ if ([
176
+ false,
177
+ undefined,
178
+ NaN
179
+ ].includes(d) || d.constructor === Date) return d;
180
+ else if (d.constructor === Number && `${d}`.length > 5 && d % 1 === 0) return new Date(d);
181
+ let s = `${d}`;
182
+ // tests for MM/DD/YYYY and MM-DD-YYYY format
183
+ const dayFormat = new RegExp(/^\d{1,2}[./-]\d{1,2}[./-](-*\d{1,4})$/g).exec(s);
184
+ if (dayFormat) {
185
+ const year = dayFormat[1];
186
+ if (year.indexOf("-") === 0) s = s.replace(year, year.substring(1));
187
+ const date = new Date(s);
188
+ date.setFullYear(year);
189
+ return date;
190
+ }
191
+ // tests for full Date object string format
192
+ const strFormat = new RegExp(/^[A-z]{1,3} [A-z]{1,3} \d{1,2} (-*\d{1,4}) \d{1,2}:\d{1,2}:\d{1,2} [A-z]{1,3}-*\d{1,4} \([A-z]{1,3}\)/g).exec(s);
193
+ if (strFormat) {
194
+ const year = strFormat[1];
195
+ if (year.indexOf("-") === 0) s = s.replace(year, year.substring(1));
196
+ const date = new Date(s);
197
+ date.setFullYear(year);
198
+ return date;
199
+ }
200
+ // tests for quarterly formats (ie. "QX YYYY")
201
+ const quarterPrefix = new RegExp(/^([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})[\s|-]{0,1}(-*\d{1,4})$/g).exec(s);
202
+ const quarterSuffix = new RegExp(/^(-*\d{1,4})[\s|-]{0,1}([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})$/g).exec(s);
203
+ if (quarterPrefix || quarterSuffix) {
204
+ const quarter = +(quarterPrefix ? quarterPrefix[1] : quarterSuffix[2]).toLowerCase().replace("q", "");
205
+ const year = +(quarterPrefix ? quarterPrefix[2] : quarterSuffix[1]);
206
+ const date = new Date(year, quarter * 3 - 3, 1);
207
+ date.setFullYear(year);
208
+ return date;
209
+ }
210
+ // detects if only passing a year value
211
+ if (!s.includes("/") && !s.includes(" ") && (!s.includes("-") || !s.indexOf("-"))) {
212
+ const date = new Date(+s, 0, 1);
213
+ date.setFullYear(d);
214
+ return date;
215
+ }
216
+ // falls back to Date object
217
+ return new Date(s);
218
+ }
219
+
220
+ /**
221
+ @function elem
222
+ @desc Manages the enter/update/exit pattern for a single DOM element.
223
+ @param {String} selector A D3 selector, which must include the tagname and a class and/or ID.
224
+ @param {Object} params Additional parameters.
225
+ @param {Boolean} [params.condition = true] Whether or not the element should be rendered (or removed).
226
+ @param {Object} [params.enter = {}] A collection of key/value pairs that map to attributes to be given on enter.
227
+ @param {Object} [params.exit = {}] A collection of key/value pairs that map to attributes to be given on exit.
228
+ @param {D3Selection} [params.parent = d3.select("body")] The parent element for this new element to be appended to.
229
+ @param {Number} [params.duration = 0] The duration for the d3 transition.
230
+ @param {Object} [params.update = {}] A collection of key/value pairs that map to attributes to be given on update.
231
+ */ function elem(selector, p) {
232
+ // overrides default params
233
+ p = Object.assign({}, {
234
+ condition: true,
235
+ enter: {},
236
+ exit: {},
237
+ duration: 0,
238
+ parent: d3Selection.select("body"),
239
+ update: {}
240
+ }, p);
241
+ const className = /\.([^#]+)/g.exec(selector), id = /#([^.]+)/g.exec(selector), t = d3Transition.transition().duration(p.duration), tag = /^([^.^#]+)/g.exec(selector)[1];
242
+ const elem = p.parent.selectAll(selector.includes(":") ? selector.split(":")[1] : selector).data(p.condition ? [
243
+ null
244
+ ] : []);
245
+ const enter = elem.enter().append(tag).call(attrize, p.enter);
246
+ if (id) enter.attr("id", id[1]);
247
+ if (className) enter.attr("class", className[1]);
248
+ if (p.duration) elem.exit().transition(t).call(attrize, p.exit).remove();
249
+ else elem.exit().call(attrize, p.exit).remove();
250
+ const update = enter.merge(elem);
251
+ if (p.duration) update.transition(t).call(attrize, p.update);
252
+ else update.call(attrize, p.update);
253
+ return update;
254
+ }
255
+
256
+ /**
257
+ * Strips HTML and "un-escapes" escape characters.
258
+ * @param {String} input
259
+ */ function htmlDecode(input) {
260
+ if (input.replace(/\s+/g, "") === "") return input;
261
+ const doc = new DOMParser().parseFromString(input.replace(/<[^>]+>/g, ""), "text/html");
262
+ return doc.documentElement ? doc.documentElement.textContent : input;
263
+ }
264
+ /**
265
+ @function textWidth
266
+ @desc Given a text string, returns the predicted pixel width of the string when placed into DOM.
267
+ @param {String|Array} text Can be either a single string or an array of strings to analyze.
268
+ @param {Object} [style] An object of CSS font styles to apply. Accepts any of the valid [CSS font property](http://www.w3schools.com/cssref/pr_font_font.asp) values.
269
+ */ function textWidth(text, style) {
270
+ style = Object.assign({
271
+ "font-size": 10,
272
+ "font-family": "sans-serif",
273
+ "font-style": "normal",
274
+ "font-weight": 400,
275
+ "font-variant": "normal"
276
+ }, style);
277
+ const context = document.createElement("canvas").getContext("2d");
278
+ const font = [];
279
+ font.push(style["font-style"]);
280
+ font.push(style["font-variant"]);
281
+ font.push(style["font-weight"]);
282
+ font.push(typeof style["font-size"] === "string" ? style["font-size"] : `${style["font-size"]}px`);
283
+ font.push(style["font-family"]);
284
+ context.font = font.join(" ");
285
+ if (text instanceof Array) return text.map((t)=>context.measureText(htmlDecode(t)).width);
286
+ return context.measureText(htmlDecode(text)).width;
287
+ }
288
+
289
+ const alpha = "abcdefghiABCDEFGHI_!@#$%^&*()_+1234567890", checked = {}, height = 32;
290
+ let dejavu, macos, monospace, proportional;
291
+ /**
292
+ @function fontExists
293
+ @desc Given either a single font-family or a list of fonts, returns the name of the first font that can be rendered, or `false` if none are installed on the user's machine.
294
+ @param {String|Array} font Can be either a valid CSS font-family string (single or comma-separated names) or an Array of string names.
295
+ */ const fontExists = (font)=>{
296
+ if (!dejavu) {
297
+ dejavu = textWidth(alpha, {
298
+ "font-family": "DejaVuSans",
299
+ "font-size": height
300
+ });
301
+ macos = textWidth(alpha, {
302
+ "font-family": "-apple-system",
303
+ "font-size": height
304
+ });
305
+ monospace = textWidth(alpha, {
306
+ "font-family": "monospace",
307
+ "font-size": height
308
+ });
309
+ proportional = textWidth(alpha, {
310
+ "font-family": "sans-serif",
311
+ "font-size": height
312
+ });
313
+ }
314
+ if (!(font instanceof Array)) font = font.split(",");
315
+ font = font.map((f)=>text.trim(f));
316
+ for(let i = 0; i < font.length; i++){
317
+ const fam = font[i];
318
+ if (checked[fam] || [
319
+ "-apple-system",
320
+ "monospace",
321
+ "sans-serif",
322
+ "DejaVuSans"
323
+ ].includes(fam)) return fam;
324
+ else if (checked[fam] === false) continue;
325
+ const width = textWidth(alpha, {
326
+ "font-family": fam,
327
+ "font-size": height
328
+ });
329
+ checked[fam] = width !== monospace;
330
+ if (checked[fam]) checked[fam] = width !== proportional;
331
+ if (macos && checked[fam]) checked[fam] = width !== macos;
332
+ if (dejavu && checked[fam]) checked[fam] = width !== dejavu;
333
+ if (checked[fam]) return fam;
334
+ }
335
+ return false;
336
+ };
337
+
338
+ /**
339
+ @desc Given an HTMLElement and a "width" or "height" string, this function returns the current calculated size for the DOM element.
340
+ @private
341
+ */ function _elementSize(element, s) {
342
+ if (!element) return undefined;
343
+ if (element.tagName === undefined || [
344
+ "BODY",
345
+ "HTML"
346
+ ].indexOf(element.tagName) >= 0) {
347
+ let val = window[`inner${s.charAt(0).toUpperCase() + s.slice(1)}`];
348
+ const elem = d3Selection.select(element);
349
+ if (s === "width") {
350
+ val -= parseFloat(elem.style("margin-left"), 10);
351
+ val -= parseFloat(elem.style("margin-right"), 10);
352
+ val -= parseFloat(elem.style("padding-left"), 10);
353
+ val -= parseFloat(elem.style("padding-right"), 10);
354
+ } else {
355
+ val -= parseFloat(elem.style("margin-top"), 10);
356
+ val -= parseFloat(elem.style("margin-bottom"), 10);
357
+ val -= parseFloat(elem.style("padding-top"), 10);
358
+ val -= parseFloat(elem.style("padding-bottom"), 10);
359
+ }
360
+ return val;
361
+ } else {
362
+ const val = parseFloat(d3Selection.select(element).style(s), 10);
363
+ if (typeof val === "number" && val > 0) return val;
364
+ else return _elementSize(element.parentNode, s);
365
+ }
366
+ }
367
+ /**
368
+ @function getSize
369
+ @desc Finds the available width and height for a specified HTMLElement, traversing it's parents until it finds something with constrained dimensions. Falls back to the inner dimensions of the browser window if none is found.
370
+ @param {HTMLElement} elem The HTMLElement to find dimensions for.
371
+ @private
372
+ */ function getSize(elem) {
373
+ return [
374
+ _elementSize(elem, "width"),
375
+ _elementSize(elem, "height")
376
+ ];
377
+ }
378
+
379
+ /**
380
+ @module inViewport
381
+ @desc Returns a *Boolean* denoting whether or not a given DOM element is visible in the current window.
382
+ @param {DOMElement} elem The DOM element to analyze.
383
+ @param {Number} [buffer = 0] A pixel offset from the edge of the top and bottom of the screen. If a positive value, the element will be deemed visible when it is that many pixels away from entering the viewport. If negative, the element will have to enter the viewport by that many pixels before being deemed visible.
384
+ @private
385
+ */ function inViewport(elem, buffer = 0) {
386
+ const pageX = window.pageXOffset !== undefined ? window.pageXOffset : (document.documentElement || document.body.parentNode || document.body).scrollLeft;
387
+ const pageY = window.pageYOffset !== undefined ? window.pageYOffset : (document.documentElement || document.body.parentNode || document.body).scrollTop;
388
+ const bounds = elem.getBoundingClientRect();
389
+ const height = bounds.height, left = bounds.left + pageX, top = bounds.top + pageY, width = bounds.width;
390
+ return pageY + window.innerHeight > top + buffer && pageY + buffer < top + height && pageX + window.innerWidth > left + buffer && pageX + buffer < left + width;
391
+ }
392
+
393
+ /**
394
+ @function parseSides
395
+ @desc Converts a string of directional CSS shorthand values into an object with the values expanded.
396
+ @param {String|Number} sides The CSS shorthand string to expand.
397
+ */ function parseSides(sides) {
398
+ let values;
399
+ if (typeof sides === "number") values = [
400
+ sides
401
+ ];
402
+ else values = sides.split(/\s+/);
403
+ if (values.length === 1) values = [
404
+ values[0],
405
+ values[0],
406
+ values[0],
407
+ values[0]
408
+ ];
409
+ else if (values.length === 2) values = values.concat(values);
410
+ else if (values.length === 3) values.push(values[1]);
411
+ return [
412
+ "top",
413
+ "right",
414
+ "bottom",
415
+ "left"
416
+ ].reduce((acc, direction, i)=>{
417
+ const value = parseFloat(values[i]);
418
+ acc[direction] = value || 0;
419
+ return acc;
420
+ }, {});
421
+ }
422
+
423
+ /**
424
+ @function prefix
425
+ @desc Returns the appropriate CSS vendor prefix, given the current browser.
426
+ */ function prefix() {
427
+ if ("-webkit-transform" in document.body.style) return "-webkit-";
428
+ else if ("-moz-transform" in document.body.style) return "-moz-";
429
+ else if ("-ms-transform" in document.body.style) return "-ms-";
430
+ else if ("-o-transform" in document.body.style) return "-o-";
431
+ else return "";
432
+ }
433
+
434
+ /**
435
+ @function rtl
436
+ @desc Returns `true` if the HTML or body element has either the "dir" HTML attribute or the "direction" CSS property set to "rtl".
437
+ */ var rtl = (()=>d3Selection.select("html").attr("dir") === "rtl" || d3Selection.select("body").attr("dir") === "rtl" || d3Selection.select("html").style("direction") === "rtl" || d3Selection.select("body").style("direction") === "rtl");
438
+
439
+ /**
440
+ @function stylize
441
+ @desc Applies each key/value in an object as a style.
442
+ @param {D3selection} elem The D3 element to apply the styles to.
443
+ @param {Object} styles An object of key/value style pairs.
444
+ */ function stylize(e, s = {}) {
445
+ for(const k in s)if (({}).hasOwnProperty.call(s, k)) e.style(k, s[k]);
446
+ }
447
+
448
+ exports.assign = assign;
449
+ exports.attrize = attrize;
450
+ exports.date = date;
451
+ exports.elem = elem;
452
+ exports.fontExists = fontExists;
453
+ exports.getSize = getSize;
454
+ exports.inViewport = inViewport;
455
+ exports.isObject = isObject;
456
+ exports.parseSides = parseSides;
457
+ exports.prefix = prefix;
458
+ exports.rtl = rtl;
459
+ exports.stylize = stylize;
460
+ exports.textWidth = textWidth;
461
+
462
+ }));
463
+ //# sourceMappingURL=d3plus-dom.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"d3plus-dom.js","sources":["../src/isObject.js","../src/assign.js","../src/attrize.js","../src/date.js","../src/elem.js","../src/textWidth.js","../src/fontExists.js","../src/getSize.js","../src/inViewport.js","../src/parseSides.js","../src/prefix.js","../src/rtl.js","../src/stylize.js"],"sourcesContent":["/**\n @function isObject\n @desc Detects if a variable is a javascript Object.\n @param {*} item\n*/\nexport default function(item) {\n return item &&\n typeof item === \"object\" &&\n (typeof window === \"undefined\" || item !== window && item !== window.document && !(item instanceof Element)) &&\n !Array.isArray(item)\n ? true : false;\n}\n","import isObject from \"./isObject.js\";\n\n/**\n @function validObject\n @desc Determines if the object passed is the document or window.\n @param {Object} obj\n @private\n*/\nfunction validObject(obj) {\n if (typeof window === \"undefined\") return true;\n else return obj !== window && obj !== document;\n}\n\n/**\n @function assign\n @desc A deeply recursive version of `Object.assign`.\n @param {...Object} objects\n @example <caption>this</caption>\nassign({id: \"foo\", deep: {group: \"A\"}}, {id: \"bar\", deep: {value: 20}}));\n @example <caption>returns this</caption>\n{id: \"bar\", deep: {group: \"A\", value: 20}}\n*/\nfunction assign(...objects) {\n\n const target = objects[0];\n for (let i = 1; i < objects.length; i++) {\n\n const source = objects[i];\n if (!isObject(source)) continue;\n\n Object.keys(source).forEach(prop => {\n\n const value = source[prop];\n\n if (isObject(value) && validObject(value)) {\n if (Object.prototype.hasOwnProperty.call(target, prop) && isObject(target[prop])) target[prop] = assign({}, target[prop], value);\n else target[prop] = assign({}, value);\n }\n else if (Array.isArray(value)) target[prop] = value.slice();\n else target[prop] = value;\n\n });\n }\n\n return target;\n\n}\n\nexport default assign;\n","/**\n @function attrize\n @desc Applies each key/value in an object as an attr.\n @param {D3selection} elem The D3 element to apply the styles to.\n @param {Object} attrs An object of key/value attr pairs.\n*/\nexport default function(e, a = {}) {\n for (const k in a) if ({}.hasOwnProperty.call(a, k)) e.attr(k, a[k]);\n}\n","/**\n @function date\n @summary Parses numbers and strings to valid Javascript Date objects.\n @description Returns a javascript Date object for a given a Number (representing either a 4-digit year or milliseconds since epoch), a String representing a Quarter (ie. \"Q2 1987\", mapping to the last day in that quarter), or a String that is in [valid dateString format](http://dygraphs.com/date-formats.html). Besides the 4-digit year parsing, this function is useful when needing to parse negative (BC) years, which the vanilla Date object cannot parse.\n @param {Number|String} *date*\n*/\nexport default function(d) {\n \n // returns if falsey or already Date object\n if ([false, undefined, NaN].includes(d) || d.constructor === Date) return d;\n // detects if milliseconds\n else if (d.constructor === Number && `${d}`.length > 5 && d % 1 === 0) return new Date(d);\n\n let s = `${d}`;\n\n // tests for MM/DD/YYYY and MM-DD-YYYY format\n const dayFormat = new RegExp(/^\\d{1,2}[./-]\\d{1,2}[./-](-*\\d{1,4})$/g).exec(s);\n if (dayFormat) {\n const year = dayFormat[1];\n if (year.indexOf(\"-\") === 0) s = s.replace(year, year.substring(1));\n const date = new Date(s);\n date.setFullYear(year);\n return date;\n }\n\n // tests for full Date object string format\n const strFormat = new RegExp(/^[A-z]{1,3} [A-z]{1,3} \\d{1,2} (-*\\d{1,4}) \\d{1,2}:\\d{1,2}:\\d{1,2} [A-z]{1,3}-*\\d{1,4} \\([A-z]{1,3}\\)/g).exec(s);\n if (strFormat) {\n const year = strFormat[1];\n if (year.indexOf(\"-\") === 0) s = s.replace(year, year.substring(1));\n const date = new Date(s);\n date.setFullYear(year);\n return date;\n }\n\n // tests for quarterly formats (ie. \"QX YYYY\")\n const quarterPrefix = new RegExp(/^([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})[\\s|-]{0,1}(-*\\d{1,4})$/g).exec(s);\n const quarterSuffix = new RegExp(/^(-*\\d{1,4})[\\s|-]{0,1}([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})$/g).exec(s);\n if (quarterPrefix || quarterSuffix) {\n const quarter = +(quarterPrefix ? quarterPrefix[1] : quarterSuffix[2]).toLowerCase().replace(\"q\", \"\");\n const year = +(quarterPrefix ? quarterPrefix[2] : quarterSuffix[1]);\n const date = new Date(year, quarter * 3 - 3, 1);\n date.setFullYear(year);\n return date;\n }\n\n // detects if only passing a year value\n if (!s.includes(\"/\") && !s.includes(\" \") && (!s.includes(\"-\") || !s.indexOf(\"-\"))) {\n const date = new Date(+s, 0, 1);\n date.setFullYear(d);\n return date;\n }\n\n // falls back to Date object\n return new Date(s);\n\n}\n","import {select} from \"d3-selection\";\nimport {transition} from \"d3-transition\";\n\nimport {default as attrize} from \"./attrize.js\";\n\n/**\n @function elem\n @desc Manages the enter/update/exit pattern for a single DOM element.\n @param {String} selector A D3 selector, which must include the tagname and a class and/or ID.\n @param {Object} params Additional parameters.\n @param {Boolean} [params.condition = true] Whether or not the element should be rendered (or removed).\n @param {Object} [params.enter = {}] A collection of key/value pairs that map to attributes to be given on enter.\n @param {Object} [params.exit = {}] A collection of key/value pairs that map to attributes to be given on exit.\n @param {D3Selection} [params.parent = d3.select(\"body\")] The parent element for this new element to be appended to.\n @param {Number} [params.duration = 0] The duration for the d3 transition.\n @param {Object} [params.update = {}] A collection of key/value pairs that map to attributes to be given on update.\n*/\nexport default function(selector, p) {\n\n // overrides default params\n p = Object.assign({}, {\n condition: true,\n enter: {},\n exit: {},\n duration: 0,\n parent: select(\"body\"),\n update: {}\n }, p);\n\n const className = (/\\.([^#]+)/g).exec(selector),\n id = (/#([^.]+)/g).exec(selector),\n t = transition().duration(p.duration),\n tag = (/^([^.^#]+)/g).exec(selector)[1];\n\n const elem = p.parent.selectAll(selector.includes(\":\") ? selector.split(\":\")[1] : selector)\n .data(p.condition ? [null] : []);\n\n const enter = elem.enter().append(tag).call(attrize, p.enter);\n\n if (id) enter.attr(\"id\", id[1]);\n if (className) enter.attr(\"class\", className[1]);\n\n if (p.duration) elem.exit().transition(t).call(attrize, p.exit).remove();\n else elem.exit().call(attrize, p.exit).remove();\n\n const update = enter.merge(elem);\n if (p.duration) update.transition(t).call(attrize, p.update);\n else update.call(attrize, p.update);\n\n return update;\n\n}\n","/**\n * Strips HTML and \"un-escapes\" escape characters.\n * @param {String} input\n */\nfunction htmlDecode(input) {\n if (input.replace(/\\s+/g, \"\") === \"\") return input;\n const doc = new DOMParser().parseFromString(input.replace(/<[^>]+>/g, \"\"), \"text/html\");\n return doc.documentElement ? doc.documentElement.textContent : input;\n}\n\n\n/**\n @function textWidth\n @desc Given a text string, returns the predicted pixel width of the string when placed into DOM.\n @param {String|Array} text Can be either a single string or an array of strings to analyze.\n @param {Object} [style] An object of CSS font styles to apply. Accepts any of the valid [CSS font property](http://www.w3schools.com/cssref/pr_font_font.asp) values.\n*/\nexport default function(text, style) {\n\n style = Object.assign({\n \"font-size\": 10,\n \"font-family\": \"sans-serif\",\n \"font-style\": \"normal\",\n \"font-weight\": 400,\n \"font-variant\": \"normal\"\n }, style);\n\n const context = document.createElement(\"canvas\").getContext(\"2d\");\n\n const font = [];\n font.push(style[\"font-style\"]);\n font.push(style[\"font-variant\"]);\n font.push(style[\"font-weight\"]);\n font.push(typeof style[\"font-size\"] === \"string\" ? style[\"font-size\"] : `${style[\"font-size\"]}px`);\n font.push(style[\"font-family\"]);\n\n context.font = font.join(\" \");\n\n if (text instanceof Array) return text.map(t => context.measureText(htmlDecode(t)).width);\n return context.measureText(htmlDecode(text)).width;\n\n}\n","import {default as textWidth} from \"./textWidth.js\";\nimport {trim} from \"@d3plus/text\";\n\nconst alpha = \"abcdefghiABCDEFGHI_!@#$%^&*()_+1234567890\",\n checked = {},\n height = 32;\n\nlet dejavu, macos, monospace, proportional;\n\n/**\n @function fontExists\n @desc Given either a single font-family or a list of fonts, returns the name of the first font that can be rendered, or `false` if none are installed on the user's machine.\n @param {String|Array} font Can be either a valid CSS font-family string (single or comma-separated names) or an Array of string names.\n*/\nconst fontExists = font => {\n\n if (!dejavu) {\n dejavu = textWidth(alpha, {\"font-family\": \"DejaVuSans\", \"font-size\": height});\n macos = textWidth(alpha, {\"font-family\": \"-apple-system\", \"font-size\": height});\n monospace = textWidth(alpha, {\"font-family\": \"monospace\", \"font-size\": height});\n proportional = textWidth(alpha, {\"font-family\": \"sans-serif\", \"font-size\": height});\n }\n\n if (!(font instanceof Array)) font = font.split(\",\");\n font = font.map(f => trim(f));\n\n for (let i = 0; i < font.length; i++) {\n const fam = font[i];\n if (checked[fam] || [\"-apple-system\", \"monospace\", \"sans-serif\", \"DejaVuSans\"].includes(fam)) return fam;\n else if (checked[fam] === false) continue;\n const width = textWidth(alpha, {\"font-family\": fam, \"font-size\": height});\n checked[fam] = width !== monospace;\n if (checked[fam]) checked[fam] = width !== proportional;\n if (macos && checked[fam]) checked[fam] = width !== macos;\n if (dejavu && checked[fam]) checked[fam] = width !== dejavu;\n if (checked[fam]) return fam;\n }\n\n return false;\n\n};\n\n\nexport default fontExists;\n","import {select} from \"d3-selection\";\n\n/**\n @desc Given an HTMLElement and a \"width\" or \"height\" string, this function returns the current calculated size for the DOM element.\n @private\n*/\nfunction _elementSize(element, s) {\n\n if (!element) return undefined;\n\n if (element.tagName === undefined || [\"BODY\", \"HTML\"].indexOf(element.tagName) >= 0) {\n\n let val = window[`inner${s.charAt(0).toUpperCase() + s.slice(1)}`];\n const elem = select(element);\n\n if (s === \"width\") {\n val -= parseFloat(elem.style(\"margin-left\"), 10);\n val -= parseFloat(elem.style(\"margin-right\"), 10);\n val -= parseFloat(elem.style(\"padding-left\"), 10);\n val -= parseFloat(elem.style(\"padding-right\"), 10);\n }\n else {\n val -= parseFloat(elem.style(\"margin-top\"), 10);\n val -= parseFloat(elem.style(\"margin-bottom\"), 10);\n val -= parseFloat(elem.style(\"padding-top\"), 10);\n val -= parseFloat(elem.style(\"padding-bottom\"), 10);\n }\n\n return val;\n\n }\n else {\n\n const val = parseFloat(select(element).style(s), 10);\n if (typeof val === \"number\" && val > 0) return val;\n else return _elementSize(element.parentNode, s);\n\n }\n}\n\n/**\n @function getSize\n @desc Finds the available width and height for a specified HTMLElement, traversing it's parents until it finds something with constrained dimensions. Falls back to the inner dimensions of the browser window if none is found.\n @param {HTMLElement} elem The HTMLElement to find dimensions for.\n @private\n*/\nexport default function(elem) {\n return [_elementSize(elem, \"width\"), _elementSize(elem, \"height\")];\n}\n","/**\n @module inViewport\n @desc Returns a *Boolean* denoting whether or not a given DOM element is visible in the current window.\n @param {DOMElement} elem The DOM element to analyze.\n @param {Number} [buffer = 0] A pixel offset from the edge of the top and bottom of the screen. If a positive value, the element will be deemed visible when it is that many pixels away from entering the viewport. If negative, the element will have to enter the viewport by that many pixels before being deemed visible.\n @private\n*/\nexport default function(elem, buffer = 0) {\n\n const pageX = window.pageXOffset !== undefined ? window.pageXOffset\n : (document.documentElement || document.body.parentNode || document.body).scrollLeft;\n\n const pageY = window.pageYOffset !== undefined ? window.pageYOffset\n : (document.documentElement || document.body.parentNode || document.body).scrollTop;\n\n const bounds = elem.getBoundingClientRect();\n const height = bounds.height,\n left = bounds.left + pageX,\n top = bounds.top + pageY,\n width = bounds.width;\n\n return pageY + window.innerHeight > top + buffer && pageY + buffer < top + height &&\n pageX + window.innerWidth > left + buffer && pageX + buffer < left + width;\n\n}\n","/**\n @function parseSides\n @desc Converts a string of directional CSS shorthand values into an object with the values expanded.\n @param {String|Number} sides The CSS shorthand string to expand.\n */\nexport default function(sides) {\n let values;\n if (typeof sides === \"number\") values = [sides];\n else values = sides.split(/\\s+/);\n\n if (values.length === 1) values = [values[0], values[0], values[0], values[0]];\n else if (values.length === 2) values = values.concat(values);\n else if (values.length === 3) values.push(values[1]);\n\n return [\n \"top\",\n \"right\",\n \"bottom\",\n \"left\"\n ].reduce((acc, direction, i) => {\n const value = parseFloat(values[i]);\n acc[direction] = value || 0;\n return acc;\n }, {});\n}\n","/**\n @function prefix\n @desc Returns the appropriate CSS vendor prefix, given the current browser.\n*/\nexport default function() {\n if (\"-webkit-transform\" in document.body.style) return \"-webkit-\";\n else if (\"-moz-transform\" in document.body.style) return \"-moz-\";\n else if (\"-ms-transform\" in document.body.style) return \"-ms-\";\n else if (\"-o-transform\" in document.body.style) return \"-o-\";\n else return \"\";\n}\n","import {select} from \"d3-selection\";\n\n/**\n @function rtl\n @desc Returns `true` if the HTML or body element has either the \"dir\" HTML attribute or the \"direction\" CSS property set to \"rtl\".\n*/\nexport default () =>\n select(\"html\").attr(\"dir\") === \"rtl\" ||\n select(\"body\").attr(\"dir\") === \"rtl\" ||\n select(\"html\").style(\"direction\") === \"rtl\" ||\n select(\"body\").style(\"direction\") === \"rtl\";\n","/**\n @function stylize\n @desc Applies each key/value in an object as a style.\n @param {D3selection} elem The D3 element to apply the styles to.\n @param {Object} styles An object of key/value style pairs.\n*/\nexport default function(e, s = {}) {\n for (const k in s) if ({}.hasOwnProperty.call(s, k)) e.style(k, s[k]);\n}\n"],"names":["item","window","document","Element","Array","isArray","validObject","obj","assign","objects","target","i","length","source","isObject","Object","keys","forEach","prop","value","prototype","hasOwnProperty","call","slice","e","a","k","attr","d","undefined","NaN","includes","constructor","Date","Number","s","dayFormat","RegExp","exec","year","indexOf","replace","substring","date","setFullYear","strFormat","quarterPrefix","quarterSuffix","quarter","toLowerCase","selector","p","condition","enter","exit","duration","parent","select","update","className","id","t","transition","tag","elem","selectAll","split","data","append","attrize","remove","merge","htmlDecode","input","doc","DOMParser","parseFromString","documentElement","textContent","text","style","context","createElement","getContext","font","push","join","map","measureText","width","alpha","checked","height","dejavu","macos","monospace","proportional","fontExists","textWidth","f","trim","fam","_elementSize","element","tagName","val","charAt","toUpperCase","parseFloat","parentNode","buffer","pageX","pageXOffset","body","scrollLeft","pageY","pageYOffset","scrollTop","bounds","getBoundingClientRect","left","top","innerHeight","innerWidth","sides","values","concat","reduce","acc","direction"],"mappings":"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"}
@@ -0,0 +1,108 @@
1
+ /*
2
+ @d3plus/dom v3.0.0
3
+ JavaScript functions for manipulating and analyzing DOM elements.
4
+ Copyright (c) 2025 D3plus - https://d3plus.org
5
+ @license MIT
6
+ */
7
+ (e=>{"function"==typeof define&&define.amd?define(e):e()})(function(){if("undefined"!=typeof window){try{if("undefined"==typeof SVGElement||Boolean(SVGElement.prototype.innerHTML))return}catch(e){return}function r(e){switch(e.nodeType){case 1:var t=e,n="";return n+="<"+t.tagName,t.hasAttributes()&&[].forEach.call(t.attributes,function(e){n+=" "+e.name+'="'+e.value+'"'}),n+=">",t.hasChildNodes()&&[].forEach.call(t.childNodes,function(e){n+=r(e)}),n+="</"+t.tagName+">";case 3:return e.textContent.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;");case 8:return"\x3c!--"+e.nodeValue+"--\x3e"}}Object.defineProperty(SVGElement.prototype,"innerHTML",{get:function(){var t="";return[].forEach.call(this.childNodes,function(e){t+=r(e)}),t},set:function(e){for(;this.firstChild;)this.removeChild(this.firstChild);try{var t=new DOMParser,n=(t.async=!1,"<svg xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink'>"+e+"</svg>"),r=t.parseFromString(n,"text/xml").documentElement;[].forEach.call(r.childNodes,function(e){this.appendChild(this.ownerDocument.importNode(e,!0))}.bind(this))}catch(e){throw new Error("Error parsing markup string")}}}),Object.defineProperty(SVGElement.prototype,"innerSVG",{get:function(){return this.innerHTML},set:function(e){this.innerHTML=e}})}}),((e,t)=>{"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("d3-selection"),require("d3-transition"),require("@d3plus/text")):"function"==typeof define&&define.amd?define("@d3plus/dom",["exports","d3-selection","d3-transition","@d3plus/text"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).d3plus={},e.d3Selection,e.d3Transition,e.text)})(this,function(e,a,s,o){
8
+ /**
9
+ @function isObject
10
+ @desc Detects if a variable is a javascript Object.
11
+ @param {*} item
12
+ */function l(e){return!(!e||"object"!=typeof e||"undefined"!=typeof window&&(e===window||e===window.document||e instanceof Element)||Array.isArray(e))}
13
+ /**
14
+ @function validObject
15
+ @desc Determines if the object passed is the document or window.
16
+ @param {Object} obj
17
+ @private
18
+ */
19
+ /**
20
+ @function attrize
21
+ @desc Applies each key/value in an object as an attr.
22
+ @param {D3selection} elem The D3 element to apply the styles to.
23
+ @param {Object} attrs An object of key/value attr pairs.
24
+ */function d(e,t={}){for(var n in t)!{}.hasOwnProperty.call(t,n)||e.attr(n,t[n])}
25
+ /**
26
+ @function date
27
+ @summary Parses numbers and strings to valid Javascript Date objects.
28
+ @description Returns a javascript Date object for a given a Number (representing either a 4-digit year or milliseconds since epoch), a String representing a Quarter (ie. "Q2 1987", mapping to the last day in that quarter), or a String that is in [valid dateString format](http://dygraphs.com/date-formats.html). Besides the 4-digit year parsing, this function is useful when needing to parse negative (BC) years, which the vanilla Date object cannot parse.
29
+ @param {Number|String} *date*
30
+ */
31
+ /**
32
+ * Strips HTML and "un-escapes" escape characters.
33
+ * @param {String} input
34
+ */function i(e){var t;return""!==e.replace(/\s+/g,"")&&(t=(new DOMParser).parseFromString(e.replace(/<[^>]+>/g,""),"text/html")).documentElement?t.documentElement.textContent:e}
35
+ /**
36
+ @function textWidth
37
+ @desc Given a text string, returns the predicted pixel width of the string when placed into DOM.
38
+ @param {String|Array} text Can be either a single string or an array of strings to analyze.
39
+ @param {Object} [style] An object of CSS font styles to apply. Accepts any of the valid [CSS font property](http://www.w3schools.com/cssref/pr_font_font.asp) values.
40
+ */function c(e,t){t=Object.assign({"font-size":10,"font-family":"sans-serif","font-style":"normal","font-weight":400,"font-variant":"normal"},t);let n=document.createElement("canvas").getContext("2d");var r=[];return r.push(t["font-style"]),r.push(t["font-variant"]),r.push(t["font-weight"]),r.push("string"==typeof t["font-size"]?t["font-size"]:t["font-size"]+"px"),r.push(t["font-family"]),n.font=r.join(" "),e instanceof Array?e.map(e=>n.measureText(i(e)).width):n.measureText(i(e)).width}let f="abcdefghiABCDEFGHI_!@#$%^&*()_+1234567890",u={},p,m,g,y;
41
+ /**
42
+ @desc Given an HTMLElement and a "width" or "height" string, this function returns the current calculated size for the DOM element.
43
+ @private
44
+ */function h(t,n){if(t){if(void 0===t.tagName||0<=["BODY","HTML"].indexOf(t.tagName)){let e=window["inner"+(n.charAt(0).toUpperCase()+n.slice(1))];var r=a.select(t);return e="width"===n?(e=(e=(e-=parseFloat(r.style("margin-left"),10))-parseFloat(r.style("margin-right"),10))-parseFloat(r.style("padding-left"),10))-parseFloat(r.style("padding-right"),10):(e=(e=(e-=parseFloat(r.style("margin-top"),10))-parseFloat(r.style("margin-bottom"),10))-parseFloat(r.style("padding-top"),10))-parseFloat(r.style("padding-bottom"),10)}return"number"==typeof(r=parseFloat(a.select(t).style(n),10))&&0<r?r:h(t.parentNode,n)}}
45
+ /**
46
+ @function getSize
47
+ @desc Finds the available width and height for a specified HTMLElement, traversing it's parents until it finds something with constrained dimensions. Falls back to the inner dimensions of the browser window if none is found.
48
+ @param {HTMLElement} elem The HTMLElement to find dimensions for.
49
+ @private
50
+ */e.assign=
51
+ /**
52
+ @function assign
53
+ @desc A deeply recursive version of `Object.assign`.
54
+ @param {...Object} objects
55
+ @example <caption>this</caption>
56
+ assign({id: "foo", deep: {group: "A"}}, {id: "bar", deep: {value: 20}}));
57
+ @example <caption>returns this</caption>
58
+ {id: "bar", deep: {group: "A", value: 20}}
59
+ */function o(...t){let i=t[0];for(let e=1;e<t.length;e++){let r=t[e];l(r)&&Object.keys(r).forEach(e=>{var t,n=r[e];l(n)&&(t=n,"undefined"==typeof window||t!==window&&t!==document)?Object.prototype.hasOwnProperty.call(i,e)&&l(i[e])?i[e]=o({},i[e],n):i[e]=o({},n):Array.isArray(n)?i[e]=n.slice():i[e]=n})}return i},e.attrize=d,e.date=function(e){
60
+ // returns if falsey or already Date object
61
+ if([!1,void 0,NaN].includes(e)||e.constructor===Date)return e;if(e.constructor===Number&&5<(""+e).length&&e%1==0)return new Date(e);let t=""+e;
62
+ // tests for MM/DD/YYYY and MM-DD-YYYY format
63
+ var n,r,o=new RegExp(/^\d{1,2}[./-]\d{1,2}[./-](-*\d{1,4})$/g).exec(t);return o?(0===(o=o[1]).indexOf("-")&&(t=t.replace(o,o.substring(1))),(n=new Date(t)).setFullYear(o),n):(o=new RegExp(/^[A-z]{1,3} [A-z]{1,3} \d{1,2} (-*\d{1,4}) \d{1,2}:\d{1,2}:\d{1,2} [A-z]{1,3}-*\d{1,4} \([A-z]{1,3}\)/g).exec(t))?(0===(n=o[1]).indexOf("-")&&(t=t.replace(n,n.substring(1))),(o=new Date(t)).setFullYear(n),o):(n=new RegExp(/^([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})[\s|-]{0,1}(-*\d{1,4})$/g).exec(t),o=new RegExp(/^(-*\d{1,4})[\s|-]{0,1}([qQ]{1}[1-4]{1}|[1-4]{1}[qQ]{1})$/g).exec(t),n||o?(r=+(n?n[1]:o[2]).toLowerCase().replace("q",""),n=+(n?n[2]:o[1]),(o=new Date(n,3*r-3,1)).setFullYear(n),o):
64
+ // detects if only passing a year value
65
+ t.includes("/")||t.includes(" ")||t.includes("-")&&t.indexOf("-")?new Date(t):((r=new Date(+t,0,1)).setFullYear(e),r));
66
+ // tests for full Date object string format
67
+ }
68
+ /**
69
+ @function elem
70
+ @desc Manages the enter/update/exit pattern for a single DOM element.
71
+ @param {String} selector A D3 selector, which must include the tagname and a class and/or ID.
72
+ @param {Object} params Additional parameters.
73
+ @param {Boolean} [params.condition = true] Whether or not the element should be rendered (or removed).
74
+ @param {Object} [params.enter = {}] A collection of key/value pairs that map to attributes to be given on enter.
75
+ @param {Object} [params.exit = {}] A collection of key/value pairs that map to attributes to be given on exit.
76
+ @param {D3Selection} [params.parent = d3.select("body")] The parent element for this new element to be appended to.
77
+ @param {Number} [params.duration = 0] The duration for the d3 transition.
78
+ @param {Object} [params.update = {}] A collection of key/value pairs that map to attributes to be given on update.
79
+ */,e.elem=function(e,t){
80
+ // overrides default params
81
+ t=Object.assign({},{condition:!0,enter:{},exit:{},duration:0,parent:a.select("body"),update:{}},t);var n=/\.([^#]+)/g.exec(e),r=/#([^.]+)/g.exec(e),o=s.transition().duration(t.duration),i=/^([^.^#]+)/g.exec(e)[1],i=(e=t.parent.selectAll(e.includes(":")?e.split(":")[1]:e).data(t.condition?[null]:[])).enter().append(i).call(d,t.enter),r=(r&&i.attr("id",r[1]),n&&i.attr("class",n[1]),(t.duration?e.exit().transition(o):e.exit()).call(d,t.exit).remove(),i.merge(e));return(t.duration?r.transition(o):r).call(d,t.update),r},e.fontExists=t=>{p||(p=c(f,{"font-family":"DejaVuSans","font-size":32}),m=c(f,{"font-family":"-apple-system","font-size":32}),g=c(f,{"font-family":"monospace","font-size":32}),y=c(f,{"font-family":"sans-serif","font-size":32})),t=(t=t instanceof Array?t:t.split(",")).map(e=>o.trim(e));for(let e=0;e<t.length;e++){var n=t[e];if(u[n]||["-apple-system","monospace","sans-serif","DejaVuSans"].includes(n))return n;if(!1!==u[n]){var r=c(f,{"font-family":n,"font-size":32});if(u[n]=r!==g,u[n]&&(u[n]=r!==y),m&&u[n]&&(u[n]=r!==m),p&&u[n]&&(u[n]=r!==p),u[n])return n}}return!1},e.getSize=function(e){return[h(e,"width"),h(e,"height")]}
82
+ /**
83
+ @module inViewport
84
+ @desc Returns a *Boolean* denoting whether or not a given DOM element is visible in the current window.
85
+ @param {DOMElement} elem The DOM element to analyze.
86
+ @param {Number} [buffer = 0] A pixel offset from the edge of the top and bottom of the screen. If a positive value, the element will be deemed visible when it is that many pixels away from entering the viewport. If negative, the element will have to enter the viewport by that many pixels before being deemed visible.
87
+ @private
88
+ */,e.inViewport=function(e,t=0){var n=void 0!==window.pageXOffset?window.pageXOffset:(document.documentElement||document.body.parentNode||document.body).scrollLeft,r=void 0!==window.pageYOffset?window.pageYOffset:(document.documentElement||document.body.parentNode||document.body).scrollTop,o=(e=e.getBoundingClientRect()).height,i=e.left+n,a=e.top+r;return r+window.innerHeight>a+t&&r+t<a+o&&n+window.innerWidth>i+t&&n+t<i+e.width}
89
+ /**
90
+ @function parseSides
91
+ @desc Converts a string of directional CSS shorthand values into an object with the values expanded.
92
+ @param {String|Number} sides The CSS shorthand string to expand.
93
+ */,e.isObject=l,e.parseSides=function(e){let r;return 1===(r="number"==typeof e?[e]:e.split(/\s+/)).length?r=[r[0],r[0],r[0],r[0]]:2===r.length?r=r.concat(r):3===r.length&&r.push(r[1]),["top","right","bottom","left"].reduce((e,t,n)=>{n=parseFloat(r[n]);return e[t]=n||0,e},{})}
94
+ /**
95
+ @function prefix
96
+ @desc Returns the appropriate CSS vendor prefix, given the current browser.
97
+ */,e.prefix=function(){return"-webkit-transform"in document.body.style?"-webkit-":"-moz-transform"in document.body.style?"-moz-":"-ms-transform"in document.body.style?"-ms-":"-o-transform"in document.body.style?"-o-":""}
98
+ /**
99
+ @function rtl
100
+ @desc Returns `true` if the HTML or body element has either the "dir" HTML attribute or the "direction" CSS property set to "rtl".
101
+ */,e.rtl=()=>"rtl"===a.select("html").attr("dir")||"rtl"===a.select("body").attr("dir")||"rtl"===a.select("html").style("direction")||"rtl"===a.select("body").style("direction"),e.stylize=
102
+ /**
103
+ @function stylize
104
+ @desc Applies each key/value in an object as a style.
105
+ @param {D3selection} elem The D3 element to apply the styles to.
106
+ @param {Object} styles An object of key/value style pairs.
107
+ */function(e,t={}){for(var n in t)!{}.hasOwnProperty.call(t,n)||e.style(n,t[n])},e.textWidth=c});
108
+ //# sourceMappingURL=d3plus-dom.js.map