From 109c121c69ca084fe338be786052f8ab73164f1a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Tom=C3=A1=C5=A1=20Vy=C4=8D=C3=ADtal?= Date: Sat, 25 Jan 2020 14:16:08 +0100 Subject: [PATCH] fix(type-coercion): use Data Pipes (#259) * fix(type-coercion): use Data Pipes Switch from the deprecated type property of Data Set to Data Pipes. This solves some or maybe all issues from #254. PS: There is a bug in Vis Data that prevents this from working see visjs/vis-data/pull/78. * docs(type-coercion): add more documentation * fix(type-coersion): prevent potential NPE --- lib/timeline/Graph2d.js | 17 ++++--- lib/timeline/Timeline.js | 38 ++++++--------- lib/timeline/component/ItemSet.js | 44 ++++++++--------- lib/timeline/component/LineGraph.js | 9 ++-- lib/util.js | 73 +++++++++++++++++++++++++++++ package-lock.json | 6 +-- package.json | 2 +- test/Timeline.test.js | 6 +++ 8 files changed, 132 insertions(+), 63 deletions(-) diff --git a/lib/timeline/Graph2d.js b/lib/timeline/Graph2d.js index e067bff6d7..43cbaeeeb8 100644 --- a/lib/timeline/Graph2d.js +++ b/lib/timeline/Graph2d.js @@ -1,5 +1,5 @@ import moment from '../module/moment'; -import util from '../util'; +import util, { typeCoerceDataSet } from '../util'; import { DataSet } from 'vis-data'; import { DataView } from 'vis-data'; import Range from './Range'; @@ -193,21 +193,20 @@ Graph2d.prototype.setItems = function(items) { newDataSet = null; } else if (items instanceof DataSet || items instanceof DataView) { - newDataSet = items; + newDataSet = typeCoerceDataSet(items); } else { // turn an array into a dataset - newDataSet = new DataSet(items, { - type: { - start: 'Date', - end: 'Date' - } - }); + newDataSet = typeCoerceDataSet(new DataSet(items)); } // set items + if (this.itemsData) { + // stop maintaining a coerced version of the old data set + this.itemsData.dispose() + } this.itemsData = newDataSet; - this.linegraph && this.linegraph.setItems(newDataSet); + this.linegraph && this.linegraph.setItems(newDataSet != null ? newDataSet.rawDS : null); if (initialLoad) { if (this.options.start != undefined || this.options.end != undefined) { diff --git a/lib/timeline/Timeline.js b/lib/timeline/Timeline.js index a7f450b89c..f49ac3e4a8 100644 --- a/lib/timeline/Timeline.js +++ b/lib/timeline/Timeline.js @@ -1,5 +1,5 @@ import moment from '../module/moment'; -import util from '../util'; +import util, { typeCoerceDataSet } from '../util'; import { DataSet } from 'vis-data'; import { DataView } from 'vis-data'; import Range from './Range'; @@ -289,9 +289,9 @@ export default class Timeline extends Core { const itemsData = this.itemsData; if (itemsData) { const selection = this.getSelection(); - this.setItems(null); // remove all - this.setItems(itemsData); // add all - this.setSelection(selection); // restore selection + this.setItems(null); // remove all + this.setItems(itemsData.rawDS); // add all + this.setSelection(selection); // restore selection } } } @@ -310,21 +310,20 @@ export default class Timeline extends Core { newDataSet = null; } else if (items instanceof DataSet || items instanceof DataView) { - newDataSet = items; + newDataSet = typeCoerceDataSet(items); } else { // turn an array into a dataset - newDataSet = new DataSet(items, { - type: { - start: 'Date', - end: 'Date' - } - }); + newDataSet = typeCoerceDataSet(new DataSet(items)); } // set items + if (this.itemsData) { + // stop maintaining a coerced version of the old data set + this.itemsData.dispose(); + } this.itemsData = newDataSet; - this.itemSet && this.itemSet.setItems(newDataSet); + this.itemSet && this.itemSet.setItems(newDataSet != null ? newDataSet.rawDS : null); } /** @@ -420,12 +419,7 @@ export default class Timeline extends Core { const ids = Array.isArray(id) ? id : [id]; // get the specified item(s) - const itemsData = this.itemsData.getDataSet().get(ids, { - type: { - start: 'Date', - end: 'Date' - } - }); + const itemsData = this.itemsData.get(ids); // calculate minimum start and maximum end of specified items let start = null; @@ -534,8 +528,7 @@ export default class Timeline extends Core { const animation = (options && options.animation !== undefined) ? options.animation : true; let range; - const dataset = this.itemsData && this.itemsData.getDataSet(); - if (dataset.length === 1 && dataset.get()[0].end === undefined) { + if (this.itemsData.length === 1 && this.itemsData.get()[0].end === undefined) { // a single item -> don't fit, just show a range around the item from -4 to +3 days range = this.getDataRange(); this.moveTo(range.min.valueOf(), {animation}, callback); @@ -646,9 +639,8 @@ export default class Timeline extends Core { let min = null; let max = null; - const dataset = this.itemsData && this.itemsData.getDataSet(); - if (dataset) { - dataset.forEach(item => { + if (this.itemsData) { + this.itemsData.forEach(item => { const start = util.convert(item.start, 'Date').valueOf(); const end = util.convert(item.end != undefined ? item.end : item.start, 'Date').valueOf(); if (min === null || start < min) { diff --git a/lib/timeline/component/ItemSet.js b/lib/timeline/component/ItemSet.js index 9e1a5ebfec..1ee4bb5a06 100644 --- a/lib/timeline/component/ItemSet.js +++ b/lib/timeline/component/ItemSet.js @@ -1,5 +1,5 @@ import Hammer from '../../module/hammer'; -import util from '../../util'; +import util, { typeCoerceDataSet } from '../../util'; import { DataSet } from 'vis-data'; import { DataView } from 'vis-data'; import TimeStep from '../TimeStep'; @@ -130,12 +130,7 @@ class ItemSet extends Component { this.options = util.extend({}, this.defaultOptions); this.options.rtl = options.rtl; this.options.onTimeout = options.onTimeout; - - // options for getting items from the DataSet with the correct type - this.itemOptions = { - type: {start: 'Date', end: 'Date'} - }; - + this.conversion = { toScreen: body.util.toScreen, toTime: body.util.toTime @@ -963,7 +958,7 @@ class ItemSet extends Component { this.itemsData = null; } else if (items instanceof DataSet || items instanceof DataView) { - this.itemsData = items; + this.itemsData = typeCoerceDataSet(items); } else { throw new TypeError('Data must be an instance of DataSet or DataView'); @@ -975,6 +970,9 @@ class ItemSet extends Component { oldItemsData.off(event, callback); }); + // stop maintaining a coerced version of the old data set + oldItemsData.dispose() + // remove all drawn items ids = oldItemsData.getIds(); this._onRemove(ids); @@ -1003,7 +1001,7 @@ class ItemSet extends Component { * @returns {vis.DataSet | null} */ getItems() { - return this.itemsData; + return this.itemsData != null ? this.itemsData.rawDS : null; } /** @@ -1097,7 +1095,6 @@ class ItemSet extends Component { */ removeItem(id) { const item = this.itemsData.get(id); - const dataset = this.itemsData.getDataSet(); if (item) { // confirm deletion @@ -1105,7 +1102,7 @@ class ItemSet extends Component { if (item) { // remove by id here, it is possible that an item has no id defined // itself, so better not delete by the item itself - dataset.remove(id); + this.itemsData.remove(id); } }); } @@ -1146,7 +1143,7 @@ class ItemSet extends Component { const me = this; ids.forEach(id => { - const itemData = me.itemsData.get(id, me.itemOptions); + const itemData = me.itemsData.get(id); let item = me.items[id]; const type = itemData ? me._getType(itemData) : null; @@ -1610,7 +1607,7 @@ class ItemSet extends Component { }; const id = util.randomUUID(); - itemData[this.itemsData._idProp] = id; + itemData[this.itemsData.idProp] = id; const group = this.groupFromTarget(event); if (group) { @@ -1823,20 +1820,19 @@ class ItemSet extends Component { event.stopPropagation(); const me = this; - const dataset = this.itemsData.getDataSet(); const itemProps = this.touchParams.itemProps; this.touchParams.itemProps = null; itemProps.forEach(props => { const id = props.item.id; - const exists = me.itemsData.get(id, me.itemOptions) != null; + const exists = me.itemsData.get(id) != null; if (!exists) { // add a new item me.options.onAdd(props.item.data, itemData => { me._removeItem(props.item); // remove temporary item if (itemData) { - me.itemsData.getDataSet().add(itemData); + me.itemsData.add(itemData); } // force re-stacking of all items next redraw @@ -1849,8 +1845,8 @@ class ItemSet extends Component { me.options.onMove(itemData, itemData => { if (itemData) { // apply changes - itemData[dataset._idProp] = id; // ensure the item contains its id (can be undefined) - dataset.update(itemData); + itemData[this.itemsData.idProp] = id; // ensure the item contains its id (can be undefined) + this.itemsData.update(itemData); } else { // restore original values @@ -2304,7 +2300,7 @@ class ItemSet extends Component { const itemData = me.itemsData.get(item.id); // get a clone of the data from the dataset this.options.onUpdate(itemData, itemData => { if (itemData) { - me.itemsData.getDataSet().update(itemData); + me.itemsData.update(itemData); } }); } @@ -2348,7 +2344,7 @@ class ItemSet extends Component { newItemData.content = newItemData.content ? newItemData.content : 'new item'; newItemData.start = newItemData.start ? newItemData.start : (snap ? snap(start, scale, step) : start); newItemData.type = newItemData.type || 'box'; - newItemData[this.itemsData._idProp] = newItemData.id || util.randomUUID(); + newItemData[this.itemsData.idProp] = newItemData.id || util.randomUUID(); if (newItemData.type == 'range' && !newItemData.end) { end = this.body.util.toTime(x + this.props.width / 5); @@ -2359,7 +2355,7 @@ class ItemSet extends Component { start: snap ? snap(start, scale, step) : start, content: 'new item' }; - newItemData[this.itemsData._idProp] = util.randomUUID(); + newItemData[this.itemsData.idProp] = util.randomUUID(); // when default type is a range, add a default end date to the new item if (this.options.type === 'range') { @@ -2377,7 +2373,7 @@ class ItemSet extends Component { newItemData = this._cloneItemData(newItemData); // convert start and end to the correct type this.options.onAdd(newItemData, item => { if (item) { - me.itemsData.getDataSet().add(item); + me.itemsData.add(item); if (event.type == 'drop') { me.setSelection([item.id]); } @@ -2421,7 +2417,7 @@ class ItemSet extends Component { if (!this.options.multiselectPerGroup || lastSelectedGroup == undefined || lastSelectedGroup == itemGroup) { selection.push(item.id); } - const range = ItemSet._getItemRange(this.itemsData.get(selection, this.itemOptions)); + const range = ItemSet._getItemRange(this.itemsData.get(selection)); if (!this.options.multiselectPerGroup || lastSelectedGroup == itemGroup) { // select all items within the selection range @@ -2609,7 +2605,7 @@ class ItemSet extends Component { if (!type) { // convert start and end date to the type (Date, Moment, ...) configured in the DataSet - type = this.itemsData.getDataSet()._options.type; + type = this.itemsData.type; } if (clone.start != undefined) { diff --git a/lib/timeline/component/LineGraph.js b/lib/timeline/component/LineGraph.js index af5d6fc186..e73f0486a1 100644 --- a/lib/timeline/component/LineGraph.js +++ b/lib/timeline/component/LineGraph.js @@ -1,4 +1,4 @@ -import util from '../../util'; +import util, { typeCoerceDataSet } from '../../util'; import * as DOMutil from '../../DOMutil'; import { DataSet } from 'vis-data'; import { DataView } from 'vis-data'; @@ -254,7 +254,7 @@ LineGraph.prototype.setItems = function (items) { this.itemsData = null; } else if (items instanceof DataSet || items instanceof DataView) { - this.itemsData = items; + this.itemsData = typeCoerceDataSet(items); } else { throw new TypeError('Data must be an instance of DataSet or DataView'); @@ -266,6 +266,9 @@ LineGraph.prototype.setItems = function (items) { oldItemsData.off(event, callback); }); + // stop maintaining a coerced version of the old data set + oldItemsData.dispose() + // remove all drawn items ids = oldItemsData.getIds(); this._onRemove(ids); @@ -433,7 +436,7 @@ LineGraph.prototype._updateAllGroupData = function (ids, groupIds) { if (this.itemsData != null) { var groupsContent = {}; var items = this.itemsData.get(); - var fieldId = this.itemsData._idProp; + var fieldId = this.itemsData.idProp; var idMap = {}; if (ids){ ids.map(function (id) { diff --git a/lib/util.js b/lib/util.js index ff910597df..8b4b5acc76 100644 --- a/lib/util.js +++ b/lib/util.js @@ -3,6 +3,7 @@ export * from "vis-util"; import * as util from "vis-util"; import { getType, isNumber, isString } from "vis-util"; +import { DataSet, createNewDataPipeFrom } from "vis-data"; import moment from "moment"; @@ -154,6 +155,78 @@ export function convert(object, type) { } } +/** + * Create a Data Set like wrapper to seamlessly coerce data types. + * + * @param rawDS - The Data Set with raw uncoerced data. + * @param type - A record assigning a data type to property name. + * + * @remarks + * The write operations (`add`, `remove`, `update` and `updateOnly`) write into + * the raw (uncoerced) data set. These values are then picked up by a pipe + * which coerces the values using the [[convert]] function and feeds them into + * the coerced data set. When querying (`forEach`, `get`, `getIds`, `off` and + * `on`) the values are then fetched from the coerced data set and already have + * the required data types. The values are coerced only once when inserted and + * then the same value is returned each time until it is updated or deleted. + * + * For example: `typeCoercedDataSet.add({ id: 7, start: "2020-01-21" })` would + * result in `typeCoercedDataSet.get(7)` returning `{ id: 7, start: moment(new + * Date("2020-01-21")).toDate() }`. + * + * Use the dispose method prior to throwing a reference to this away. Otherwise + * the pipe connecting the two Data Sets will keep the unaccessible coerced + * Data Set alive and updated as long as the raw Data Set exists. + * + * @returns A Data Set like object that saves data into the raw Data Set and + * retrieves them from the coerced Data Set. + */ +export function typeCoerceDataSet( + rawDS, + type = { start: "Date", end: "Date" } +) { + const idProp = rawDS._idProp; + const coercedDS = new DataSet({ fieldId: idProp }); + + const pipe = createNewDataPipeFrom(rawDS) + .map(item => + Object.keys(item).reduce((acc, key) => { + acc[key] = convert(item[key], type[key]); + return acc; + }, {}) + ) + .to(coercedDS); + + pipe.all().start(); + + return { + // Write only. + add: (...args) => rawDS.getDataSet().add(...args), + remove: (...args) => rawDS.getDataSet().remove(...args), + update: (...args) => rawDS.getDataSet().update(...args), + updateOnly: (...args) => rawDS.getDataSet().updateOnly(...args), + + // Read only. + forEach: coercedDS.forEach.bind(coercedDS), + get: coercedDS.get.bind(coercedDS), + getIds: coercedDS.getIds.bind(coercedDS), + off: coercedDS.off.bind(coercedDS), + on: coercedDS.on.bind(coercedDS), + + get length() { + return coercedDS.length; + }, + + // Non standard. + idProp, + type, + + rawDS, + coercedDS, + dispose: () => pipe.stop() + }; +} + export default { ...util, convert diff --git a/package-lock.json b/package-lock.json index cee36e4df9..6522e0df5b 100644 --- a/package-lock.json +++ b/package-lock.json @@ -15333,9 +15333,9 @@ "dev": true }, "vis-data": { - "version": "6.3.3", - "resolved": "https://registry.npmjs.org/vis-data/-/vis-data-6.3.3.tgz", - "integrity": "sha512-yt0szLEgiBDIKE4LE+CC/K+5iFO4kUHNVwlPm97+t4+A7vCXSLCLaLo7WhKgwIZN/xbnfD2z8U4P7xzHmWADag==", + "version": "6.3.5", + "resolved": "https://registry.npmjs.org/vis-data/-/vis-data-6.3.5.tgz", + "integrity": "sha512-H8lD1NTmC5HimYeqTouxhbUya5qrYQGjpJhIHkxsm5R34EEdcm/D1cLTMLFFyIoZC6SQkADQj+dlFKf5DOnSww==", "requires": { "vis-util": "2.0.1", "vis-uuid": "1.1.3" diff --git a/package.json b/package.json index 89bd836221..c8a70bbcb6 100644 --- a/package.json +++ b/package.json @@ -50,7 +50,7 @@ "keycharm": "^0.3.0", "moment": "2.24.0", "propagating-hammerjs": "1.4.7", - "vis-data": "6.3.3", + "vis-data": "6.3.5", "vis-util": "2.1.0" }, "devDependencies": { diff --git a/test/Timeline.test.js b/test/Timeline.test.js index a23481dc3c..2f8081676b 100644 --- a/test/Timeline.test.js +++ b/test/Timeline.test.js @@ -44,4 +44,10 @@ describe('Timeline', () => { timeline .setSelection([events[0].id], {animation: false}); }); + + it("setItems(null) should not crash", function() { + const timeline = new Timeline(document.createElement("div"), []); + + timeline.setItems(null); + }); });