pass emit update fn through setter so we control when tile can update

This commit is contained in:
Bruno Windels 2020-03-21 14:28:09 +01:00
parent a3714f49cc
commit f729108226
2 changed files with 36 additions and 12 deletions

View file

@ -2,6 +2,10 @@ import BaseObservableList from "../../../../observable/list/BaseObservableList.j
import sortedIndex from "../../../../utils/sortedIndex.js"; import sortedIndex from "../../../../utils/sortedIndex.js";
// maps 1..n entries to 0..1 tile. Entries are what is stored in the timeline, either an event or fragmentboundary // maps 1..n entries to 0..1 tile. Entries are what is stored in the timeline, either an event or fragmentboundary
// for now, tileCreator should be stable in whether it returns a tile or not.
// e.g. the decision to create a tile or not should be based on properties
// not updated later on (e.g. event type)
// also see big comment in onUpdate
export default class TilesCollection extends BaseObservableList { export default class TilesCollection extends BaseObservableList {
constructor(entries, tileCreator) { constructor(entries, tileCreator) {
super(); super();
@ -28,7 +32,7 @@ export default class TilesCollection extends BaseObservableList {
let currentTile = null; let currentTile = null;
for (let entry of this._entries) { for (let entry of this._entries) {
if (!currentTile || !currentTile.tryIncludeEntry(entry)) { if (!currentTile || !currentTile.tryIncludeEntry(entry)) {
currentTile = this._tileCreator(entry, this._emitSpontanousUpdate); currentTile = this._tileCreator(entry);
if (currentTile) { if (currentTile) {
this._tiles.push(currentTile); this._tiles.push(currentTile);
} }
@ -45,6 +49,11 @@ export default class TilesCollection extends BaseObservableList {
if (prevTile) { if (prevTile) {
prevTile.updateNextSibling(null); prevTile.updateNextSibling(null);
} }
// now everything is wired up,
// allow tiles to emit updates
for (const tile of this._tiles) {
tile.setUpdateEmit(this._emitSpontanousUpdate);
}
} }
_findTileIdx(entry) { _findTileIdx(entry) {
@ -93,10 +102,11 @@ export default class TilesCollection extends BaseObservableList {
return; return;
} }
const newTile = this._tileCreator(entry, this._emitSpontanousUpdate); const newTile = this._tileCreator(entry);
if (newTile) { if (newTile) {
if (prevTile) { if (prevTile) {
prevTile.updateNextSibling(newTile); prevTile.updateNextSibling(newTile);
// this emits an update while the add hasn't been emitted yet
newTile.updatePreviousSibling(prevTile); newTile.updatePreviousSibling(prevTile);
} }
if (nextTile) { if (nextTile) {
@ -105,6 +115,9 @@ export default class TilesCollection extends BaseObservableList {
} }
this._tiles.splice(tileIdx, 0, newTile); this._tiles.splice(tileIdx, 0, newTile);
this.emitAdd(tileIdx, newTile); this.emitAdd(tileIdx, newTile);
// add event is emitted, now the tile
// can emit updates
newTile.setUpdateEmit(this._emitSpontanousUpdate);
} }
// find position by sort key // find position by sort key
// ask siblings to be included? both? yes, twice: a (insert c here) b, ask a(c), if yes ask b(a), else ask b(c)? if yes then b(a)? // ask siblings to be included? both? yes, twice: a (insert c here) b, ask a(c), if yes ask b(a), else ask b(c)? if yes then b(a)?
@ -141,6 +154,7 @@ export default class TilesCollection extends BaseObservableList {
this._tiles.splice(tileIdx, 1); this._tiles.splice(tileIdx, 1);
prevTile && prevTile.updateNextSibling(nextTile); prevTile && prevTile.updateNextSibling(nextTile);
nextTile && nextTile.updatePreviousSibling(prevTile); nextTile && nextTile.updatePreviousSibling(prevTile);
tile.setUpdateEmit(null);
this.emitRemove(tileIdx, tile); this.emitRemove(tileIdx, tile);
} }
@ -177,10 +191,12 @@ import ObservableArray from "../../../../observable/list/ObservableArray.js";
import UpdateAction from "./UpdateAction.js"; import UpdateAction from "./UpdateAction.js";
export function tests() { export function tests() {
class TestTile { class TestTile {
constructor(entry, update) { constructor(entry) {
this.entry = entry; this.entry = entry;
this.update = null;
}
setUpdateEmit(update) {
this.update = update; this.update = update;
} }
tryIncludeEntry() { tryIncludeEntry() {
@ -211,15 +227,15 @@ export function tests() {
class UpdateOnSiblingTile extends TestTile { class UpdateOnSiblingTile extends TestTile {
updateNextSibling() { updateNextSibling() {
// this happens with isContinuation // this happens with isContinuation
this.update(this, "next"); this.update && this.update(this, "next");
} }
updatePreviousSibling() { updatePreviousSibling() {
// this happens with isContinuation // this happens with isContinuation
this.update(this, "previous"); this.update && this.update(this, "previous");
} }
} }
const entries = new ObservableArray([{n: 5}, {n: 10}]); const entries = new ObservableArray([{n: 5}, {n: 10}]);
const tiles = new TilesCollection(entries, (e, u) => new UpdateOnSiblingTile(e, u)); const tiles = new TilesCollection(entries, entry => new UpdateOnSiblingTile(entry));
let receivedAdd = false; let receivedAdd = false;
tiles.subscribe({ tiles.subscribe({
onAdd(idx, tile) { onAdd(idx, tile) {
@ -227,8 +243,9 @@ export function tests() {
receivedAdd = true; receivedAdd = true;
}, },
onUpdate(idx, tile) { onUpdate(idx, tile) {
assert(tile.entry.n, 7); if (tile.entry.n === 7) {
assert(!receivedAdd, "receiving update before add"); assert(!receivedAdd, "receiving update before add");
}
} }
}); });
entries.insert(1, {n: 7}); entries.insert(1, {n: 7});

View file

@ -1,9 +1,9 @@
import UpdateAction from "../UpdateAction.js"; import UpdateAction from "../UpdateAction.js";
export default class SimpleTile { export default class SimpleTile {
constructor({entry, emitUpdate}) { constructor({entry}) {
this._entry = entry; this._entry = entry;
this._emitUpdate = emitUpdate; this._emitUpdate = null;
} }
// view model props for all subclasses // view model props for all subclasses
// hmmm, could also do instanceof ... ? // hmmm, could also do instanceof ... ?
@ -23,7 +23,9 @@ export default class SimpleTile {
} }
emitUpdate(paramName) { emitUpdate(paramName) {
this._emitUpdate(this, paramName); if (this._emitUpdate) {
this._emitUpdate(this, paramName);
}
} }
get internalId() { get internalId() {
@ -33,6 +35,11 @@ export default class SimpleTile {
get isPending() { get isPending() {
return this._entry.isPending; return this._entry.isPending;
} }
// TilesCollection contract below
setUpdateEmit(emitUpdate) {
this._emitUpdate = emitUpdate;
}
get upperEntry() { get upperEntry() {
return this._entry; return this._entry;
} }