Microsoft/fast-dna

View on GitHub
packages/web-components/fast-element/src/templating/repeat.spec.ts

Summary

Maintainability
F
6 days
Test Coverage
import { observable } from "../observation/observable.js";
import { RepeatBehavior, RepeatDirective, repeat } from "./repeat.js";
import { expect } from "chai";
import { html } from "./template.js";
import { toHTML } from "../__test__/helpers.js";
import { Updates } from "../observation/update-queue.js";
import type { ViewBehaviorTargets, ViewController } from "./html-directive.js";
import { Fake } from "../testing/fakes.js";

describe("The repeat", () => {
    function createLocation() {
        const parent = document.createElement("div");
        const location = document.createComment("");
        const nodeId = 'r';
        const targets = { [nodeId]: location };

        parent.appendChild(location);

        return { parent, targets, nodeId };
    }

    function expectViewPositionToBeCorrect(behavior: RepeatBehavior<any>) {
        for (let i = 0, ii = behavior.views.length; i < ii; ++i) {
            const context = behavior.views[i].context;
            expect(context.index).equal(i);
            expect(context.length).equal(ii);
        }
    }

    context("template function", () => {
        class ViewModel {
            items = ["a", "b", "c"]
        }

        it("returns a RepeatDirective", () => {
            const directive = repeat(
                () => [],
                html`test`
            );
            expect(directive).to.be.instanceOf(RepeatDirective);
        });

        it("returns a RepeatDirective with optional properties set to default values", () => {
            const directive = repeat(
                () => [],
                html`test`
            ) as RepeatDirective;
            expect(directive).to.be.instanceOf(RepeatDirective);
            expect(directive.options).to.deep.equal({positioning: false, recycle: true})
        });

        it("returns a RepeatDirective with recycle property set to default value when positioning is set to different value", () => {
            const directive = repeat(
                () => [],
                html`test`,
                {positioning: true}
            ) as RepeatDirective;
            expect(directive).to.be.instanceOf(RepeatDirective);
            expect(directive.options).to.deep.equal({positioning: true, recycle: true})
        });

        it("returns a RepeatDirective with positioning property set to default value when recycle is set to different value", () => {
            const directive = repeat(
                () => [],
                html`test`,
                {recycle: false}
            ) as RepeatDirective;
            expect(directive).to.be.instanceOf(RepeatDirective);
            expect(directive.options).to.deep.equal({positioning: false, recycle: false})
        });

        it("returns a RepeatDirective with optional properties set to different values", () => {
            const directive = repeat(
                () => [],
                html`test`,
                {positioning: true, recycle: false}
            ) as RepeatDirective;
            expect(directive).to.be.instanceOf(RepeatDirective);
            expect(directive.options).to.deep.equal({positioning: true, recycle: false})
        });

        it("creates a data binding that evaluates the provided binding", () => {
            const source = new ViewModel();
            const directive = repeat<ViewModel>(x => x.items, html`test`) as RepeatDirective;

            const data = directive.dataBinding.evaluate(source, Fake.executionContext());

            expect(data).to.equal(source.items);
        });

        it("creates a data binding that evaluates to a provided array", () => {
            const array = ["a", "b", "c"];
            const itemTemplate = html`test`;
            const directive = repeat(array, itemTemplate) as RepeatDirective;

            const data = directive.dataBinding.evaluate({}, Fake.executionContext());

            expect(data).to.equal(array);
        });

        it("creates a template binding when a template is provided", () => {
            const source = new ViewModel();
            const itemTemplate = html`test`;
            const directive = repeat<ViewModel>(x => x.items, itemTemplate) as RepeatDirective;
            const template = directive.templateBinding.evaluate(source, Fake.executionContext());
            expect(template).to.equal(itemTemplate);
        });

        it("creates a template binding when a function is provided", () => {
            const source = new ViewModel();
            const itemTemplate = html`test`;
            const directive = repeat<ViewModel>(x => x.items, () => itemTemplate) as RepeatDirective;
            const template = directive.templateBinding.evaluate(source, Fake.executionContext());
            expect(template).equal(itemTemplate);
        });
    });

    context("directive", () => {
        it("creates a RepeatBehavior", () => {
            const { nodeId } = createLocation();
            const directive = repeat(
                () => [],
                html`test`
            ) as RepeatDirective;
            directive.targetNodeId = nodeId;

            const behavior = directive.createBehavior();

            expect(behavior).to.be.instanceOf(RepeatBehavior);
        });
    });

    context("behavior", () => {
        const itemTemplate = html<Item>`${x => x.name}`;
        const altItemTemplate = html<Item>`*${x => x.name}`;
        const oneThroughTen = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
        const zeroThroughTen = [0].concat(oneThroughTen);
        const wrappedItemTemplate = html<Item>`<div>${x => x.name}</div>`;

        interface Item {
            name: string;
            items?: Item[];
        }

        function createArray(size: number) {
            const items: { name: string }[] = [];

            for (let i = 0; i < size; ++i) {
                items.push({ name: `item${i + 1}` });
            }

            return items;
        }

        class ViewModel {
            name = "root";
            @observable items: Item[];
            @observable template = itemTemplate;

            constructor(size: number, nested: boolean = false) {
                this.items = createArray(size);

                if (nested) {
                    this.items.forEach(x => (x.items = createArray(size)));
                }
            }
        }

        function createOutput(
            size: number,
            filter: (index: number) => boolean = () => true,
            prefix = "",
            wrapper = input => input,
            fromIndex: number = 0
        ) {
            let output = "";
            const delta = fromIndex > 0 ? fromIndex : 0
            for (let i = 0; i < size; ++i) {
                if (filter(i)) {
                    output += wrapper(`${prefix}item${i + 1 + delta}`);
                }
            }

            return output;
        }

        function createController(source: any, targets: ViewBehaviorTargets) {
            const unbindables: { unbind(controller: ViewController) }[] = [];

            return {
                isBound: false,
                context: Fake.executionContext(),
                onUnbind(object) {
                    unbindables.push(object);
                },
                source,
                targets,
                unbind() {
                    unbindables.forEach(x => x.unbind(this))
                }
            };
        }

        zeroThroughTen.forEach(size => {
            it(`renders a template for each item in array of size ${size}`, () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;

                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                expect(toHTML(parent)).to.equal(createOutput(size));
            });

            it(`renders a template for each item in array of size ${size} with recycle property set to false`, () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate,
                    {positioning: true, recycle: false}
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;

                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                expectViewPositionToBeCorrect(behavior);
                expect(toHTML(parent)).to.equal(createOutput(size));
            });
        });

        zeroThroughTen.forEach(size => {
            it(`renders empty when an array of size ${size} is replaced with an empty array`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    wrappedItemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const data = new ViewModel(size);
                const controller = createController(data, targets);

                behavior.bind(controller);

                expect(toHTML(parent)).to.equal(
                    createOutput(size, void 0, void 0, input => `<div>${input}</div>`)
                );

                data.items = [];

                await Updates.next();

                expect(toHTML(parent)).to.equal("");

                data.items = createArray(size);

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    createOutput(size, void 0, void 0, input => `<div>${input}</div>`)
                );
            });
        });

        zeroThroughTen.forEach(size => {
            it(`updates rendered HTML when a new item is pushed into an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);
                vm.items.push({ name: "newitem" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(`${createOutput(size)}newitem`);
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a single item is spliced from the end of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const index = size - 1;
                vm.items.splice(index, 1);

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size, x => x !== index)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a single item is spliced from the beginning of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.splice(0, 1);

                await Updates.next();

                expect(toHTML(parent)).to.equal(`${createOutput(size, x => x !== 0)}`);
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a single item is replaced from the end of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const index = size - 1;
                vm.items.splice(index, 1, { name: "newitem1" }, { name: "newitem2" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size, x => x !== index)}newitem1newitem2`
                );
            });

            it(`updates rendered HTML when a single item is replaced from the end of an array of size ${size} with recycle property set to false`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate,
                    {positioning: true, recycle: false}
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);
                expectViewPositionToBeCorrect(behavior);

                const index = size - 1;
                vm.items.splice(index, 1, { name: "newitem1" }, { name: "newitem2" });

                await Updates.next();

                expectViewPositionToBeCorrect(behavior);
                expect(toHTML(parent)).to.equal(
                    `${createOutput(size, x => x !== index)}newitem1newitem2`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a single item is spliced from the middle of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const mid = Math.floor(vm.items.length/2)
                vm.items.splice(mid, 1, { name: "newitem1" });
                await Updates.next();
                expect(toHTML(parent)).to.equal(`${createOutput(mid)}newitem1${createOutput(vm.items.slice(mid +1).length , void 0, void 0, void 0, mid +1 ) }`);
            });

            it(`updates rendered HTML when a single item is spliced from the middle of an array of size ${size} with recycle property set to false`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate,
                    {positioning: true, recycle: false}
                ) as RepeatDirective;

                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);
                expectViewPositionToBeCorrect(behavior);

                const mid = Math.floor(vm.items.length/2)
                vm.items.splice(mid, 1, { name: "newitem1" });
                await Updates.next();

                expectViewPositionToBeCorrect(behavior);
                expect(toHTML(parent)).to.equal(`${createOutput(mid)}newitem1${createOutput(vm.items.slice(mid +1).length , void 0, void 0, void 0, mid +1 ) }`);
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a 2 items are spliced from the middle of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const mid = Math.floor(vm.items.length/2)
                vm.items.splice(mid, 2, { name: "newitem1" }, { name: "newitem2" });
                await Updates.next();
                expect(toHTML(parent)).to.equal(`${createOutput(mid)}newitem1newitem2${createOutput(vm.items.slice(mid +2).length , void 0, void 0, void 0, mid +2 ) }`);
            });

            it(`updates rendered HTML when 2 items are spliced from the middle of an array of size ${size} with recycle property set to false`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate,
                    { recycle: false}
                ) as RepeatDirective;

                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const mid = Math.floor(vm.items.length/2)
                vm.items.splice(mid, 2, { name: "newitem1" }, { name: "newitem2" });
                await Updates.next();
                expect(toHTML(parent)).to.equal(`${createOutput(mid)}newitem1newitem2${createOutput(vm.items.slice(mid +2).length , void 0, void 0, void 0, mid +2 ) }`);
            });
            it(`updates rendered HTML when all items are spliced to replace entire array with an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate,
                    { positioning: true}
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                expectViewPositionToBeCorrect(behavior);

                vm.items.splice(0, vm.items.length, ...vm.items);
                await Updates.next();
                expectViewPositionToBeCorrect(behavior);
                expect(toHTML(parent)).to.equal(createOutput(size));

                vm.items.splice(0, vm.items.length, ...vm.items);
                await Updates.next();
                expectViewPositionToBeCorrect(behavior);
                expect(toHTML(parent)).to.equal(createOutput(size));
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates rendered HTML when a single item is replaced from the beginning of an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.splice(0, 1, { name: "newitem1" }, { name: "newitem2" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `newitem1newitem2${createOutput(size, x => x !== 0)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`updates all when the template changes for an array of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    x => vm.template
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                expect(toHTML(parent)).to.equal(createOutput(size));

                vm.template = altItemTemplate;

                await Updates.next();

                expect(toHTML(parent)).to.equal(createOutput(size, () => true, "*"));
            });
        });

        oneThroughTen.forEach(size => {
            it(`renders grandparent values from nested arrays of size ${size}`, async () => {
                const deepItemTemplate = html<Item>`
                    parent-${x => x.name}${repeat(
                        x => x.items!,
                        html<Item>`child-${x => x.name}root-${(x, c) => c.parentContext.parent.name}`
                    )}
                `;

                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    deepItemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size, true);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                const text = toHTML(parent);

                for (let i = 0; i < size; ++i) {
                    const str = `child-item${i + 1}root-root`;
                    expect(text.indexOf(str)).to.not.equal(-1);
                }
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back shift operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.shift();
                vm.items.unshift({ name: "shift" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shift${createOutput(size, index => index !== 0)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back shift operations with multiple unshift items for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.shift();
                vm.items.unshift({ name: "shift" }, { name: "shift" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shiftshift${createOutput(size, index => index !== 0)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back shift and unshift operations with multiple unshift items for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.shift();
                vm.items.unshift({ name: "shift1" }, { name: "shift2" });
                vm.items.shift();

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shift2${createOutput(size, index => index !== 0)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back shift and push operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.shift();
                vm.items.push({ name: "shift3" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size, index => index !== 0)}shift3`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back push and shift operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.push({ name: "shift3" });
                vm.items.shift();

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size, index => index !== 0)}shift3`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back push and pop operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.push({ name: "shift3" });
                vm.items.pop();

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back pop and push operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.pop();
                vm.items.push({ name: "shift3" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `${createOutput(size-1)}shift3`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back array modification operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);;

                vm.items.pop();
                vm.items.push({ name: "shift3" });
                vm.items.unshift({ name: "shift1" }, { name: "shift2" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shift1shift2${createOutput(size-1)}shift3`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back array modification 2 operations for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.push({ name: "shift3" });
                vm.items.pop();
                vm.items.unshift({ name: "shift1" }, { name: "shift2" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shift1shift2${createOutput(size)}`
                );
            });
        });

        oneThroughTen.forEach(size => {
            it(`handles back to back multiple shift operations with unshift with multiple items for arrays of size ${size}`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                vm.items.shift();
                vm.items.shift();
                vm.items.unshift({ name: "shift1" }, { name: "shift2" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(
                    `shift1shift2${createOutput(size -1, index => index !== 0, void 0, void 0, 1 ) }`
                );
            });
        });

        zeroThroughTen.forEach(size => {
            it(`updates rendered HTML when a new item is pushed into an array of size ${size} after it has been unbound and rebound`, async () => {
                const { parent, targets, nodeId } = createLocation();
                const directive = repeat<ViewModel>(
                    x => x.items,
                    itemTemplate
                ) as RepeatDirective;
                directive.targetNodeId = nodeId;
                const behavior = directive.createBehavior();
                const vm = new ViewModel(size);
                const controller = createController(vm, targets);

                behavior.bind(controller);

                await Updates.next();

                controller.unbind();

                await Updates.next();

                behavior.bind(controller);

                await Updates.next();

                vm.items.push({ name: "newitem" });

                await Updates.next();

                expect(toHTML(parent)).to.equal(`${createOutput(size)}newitem`);
            });
        });
    });
});