Files
clawdbot/vendor/a2ui/renderers/lit/dist/src/0.8/model.test.js
2025-12-17 13:20:27 +01:00

1238 lines
55 KiB
JavaScript

/*
Copyright 2025 Google LLC
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
https://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
import assert from "node:assert";
import { describe, it, beforeEach } from "node:test";
import { v0_8 } from "@a2ui/lit";
// Helper function to strip reactivity for clean comparisons.
const toPlainObject = (value) => {
if (value instanceof Map) {
return Object.fromEntries(Array.from(value.entries(), ([k, v]) => [k, toPlainObject(v)]));
}
if (Array.isArray(value)) {
return value.map(toPlainObject);
}
if (v0_8.Data.Guards.isObject(value) &&
value.constructor.name === "SignalObject") {
const obj = {};
for (const key in value) {
if (Object.prototype.hasOwnProperty.call(value, key)) {
obj[key] = toPlainObject(value[key]);
}
}
return obj;
}
return value;
};
describe("A2uiMessageProcessor", () => {
let processor = new v0_8.Data.A2uiMessageProcessor();
beforeEach(() => {
processor = new v0_8.Data.A2uiMessageProcessor();
});
describe("Basic Initialization and State", () => {
it("should start with no surfaces", () => {
assert.strictEqual(processor.getSurfaces().size, 0);
});
it("should clear surfaces when clearSurfaces is called", () => {
processor.processMessages([
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
assert.strictEqual(processor.getSurfaces().size, 1);
processor.clearSurfaces();
assert.strictEqual(processor.getSurfaces().size, 0);
});
});
describe("Message Processing", () => {
it("should handle `beginRendering` by creating a default surface", () => {
processor.processMessages([
{
beginRendering: {
root: "comp-a",
styles: { color: "blue" },
surfaceId: "@default",
},
},
]);
const surfaces = processor.getSurfaces();
assert.strictEqual(surfaces.size, 1);
const defaultSurface = surfaces.get("@default");
assert.ok(defaultSurface, "Default surface should exist");
assert.strictEqual(defaultSurface.rootComponentId, "comp-a");
assert.deepStrictEqual(defaultSurface.styles, { color: "blue" });
});
it("should handle `surfaceUpdate` by adding components", () => {
const messages = [
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "comp-a",
component: {
Text: { text: { literalString: "Hi" } },
},
},
],
},
},
];
processor.processMessages(messages);
const surface = processor.getSurfaces().get("@default");
if (!surface) {
assert.fail("No default surface");
}
assert.strictEqual(surface.components.size, 1);
assert.ok(surface.components.has("comp-a"));
});
it("should handle `deleteSurface`", () => {
processor.processMessages([
{
beginRendering: { root: "root", surfaceId: "to-delete" },
},
{ deleteSurface: { surfaceId: "to-delete" } },
]);
assert.strictEqual(processor.getSurfaces().has("to-delete"), false);
});
});
describe("Data Model Updates", () => {
it("should update data at a specified path", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/user",
contents: [{ key: "name", valueString: "Alice" }],
},
},
]);
const name = processor.getData({ dataContextPath: "/" }, "/user/name");
assert.strictEqual(name, "Alice");
});
it("should replace the entire data model when path is not provided", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{ key: "user", valueString: JSON.stringify({ name: "Bob" }) },
],
},
},
]);
const user = processor.getData({ dataContextPath: "/" }, "/user");
assert.deepStrictEqual(toPlainObject(user), { name: "Bob" });
});
it("should create nested structures when setting data", () => {
const component = { dataContextPath: "/" };
// Note: setData is a public method that does not use the key-value format
processor.setData(component, "/a/b/c", "value");
const data = processor.getData(component, "/a/b/c");
assert.strictEqual(data, "value");
});
it("should handle paths correctly", () => {
const path1 = processor.resolvePath("/a/b/c", "/value");
const path2 = processor.resolvePath("a/b/c", "/value/");
const path3 = processor.resolvePath("a/b/c", "/value");
assert.strictEqual(path1, "/a/b/c");
assert.strictEqual(path2, "/value/a/b/c");
assert.strictEqual(path3, "/value/a/b/c");
});
it("should correctly parse nested valueMap structures", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/data",
contents: [
{
key: "users", // /data/users
valueMap: [
{
key: "user1", // /data/users/user1
valueMap: [
{
key: "firstName",
valueString: "Alice",
},
{
key: "lastName",
valueString: "Doe",
},
],
},
{
key: "user2", // /data/users/user2
valueMap: [
{
key: "firstName",
valueString: "John",
},
{
key: "lastName",
valueString: "Doe",
},
],
},
],
},
],
},
},
]);
const info = processor.getData({ dataContextPath: "/" }, "/data/users");
// The expected result is a Map of Maps.
const expected = new Map([
[
"user1",
new Map([
["firstName", "Alice"],
["lastName", "Doe"],
]),
],
[
"user2",
new Map([
["firstName", "John"],
["lastName", "Doe"],
]),
],
]);
assert.deepEqual(info, expected);
});
it("should additively update a Map using numeric-string keys (like timestamps)", () => {
// 1. First, establish the /messages path as a Map.
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/messages",
contents: [
// Sending an empty key-value array creates an empty Map at the path.
],
},
},
]);
const key1 = "1700000000001";
const message1 = "Hello";
// 2. Add the first message.
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: `/messages/${key1}`,
contents: [
{
key: ".",
valueString: message1,
},
],
},
},
]);
let messagesData = processor.getData({ dataContextPath: "/" }, "/messages");
// Check that it's a Map and has the first item.
assertIsDataMap(messagesData);
assert.strictEqual(messagesData.size, 1);
assert.strictEqual(messagesData.get(key1), message1);
const key2 = "1700000000002";
const message2 = "World";
// 3. Add the second message. This is where the old logic would fail.
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: `/messages/${key2}`,
contents: [
{
key: ".",
valueString: message2,
},
],
},
},
]);
messagesData = processor.getData({ dataContextPath: "/" }, "/messages");
// 4. Check that the Map was additively updated and now has both items.
assertIsDataMap(messagesData);
assert.strictEqual(messagesData.size, 2, "Map should have 2 items total");
assert.strictEqual(messagesData.get(key1), message1, "First item correct");
assert.strictEqual(messagesData.get(key2), message2, "Second item correct");
});
});
describe("Component Tree Building", () => {
it("should build a simple parent-child tree", () => {
processor.processMessages([
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
Column: { children: { explicitList: ["child"] } },
},
},
{
id: "child",
component: {
Text: { text: { literalString: "Hello" } },
},
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
assert.strictEqual(plainTree.id, "root");
assert.strictEqual(plainTree.type, "Column");
assert.strictEqual(plainTree.properties.children.length, 1);
assert.strictEqual(plainTree.properties.children[0].id, "child");
assert.strictEqual(plainTree.properties.children[0].type, "Text");
});
it("should not treat enum-like strings as child component IDs", () => {
processor.processMessages([
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
Column: { children: { explicitList: ["body"] } },
},
},
{
id: "body",
component: {
Text: {
text: { literalString: "Hello" },
usageHint: "body",
},
},
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
assert.strictEqual(plainTree.id, "root");
assert.strictEqual(plainTree.properties.children[0].id, "body");
assert.strictEqual(plainTree.properties.children[0].type, "Text");
});
it("should throw an error on circular dependencies", () => {
// First, load the components
processor.processMessages([
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{ id: "a", component: { Card: { child: "b" } } },
{ id: "b", component: { Card: { child: "a" } } },
],
},
},
]);
// Now, try to render, which triggers the tree build
assert.throws(() => {
processor.processMessages([
{
beginRendering: {
root: "a",
surfaceId: "@default",
},
},
]);
}, new Error(`Circular dependency for component "a".`));
const tree = processor.getSurfaces().get("@default")?.componentTree;
assert.strictEqual(tree, null, "Tree should be null due to circular dependency");
});
it("should correctly expand a template with `dataBinding`", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "items",
valueString: JSON.stringify([{ name: "A" }, { name: "B" }]),
},
],
},
},
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
List: {
children: {
template: {
componentId: "item-template",
dataBinding: "/items",
},
},
},
},
},
{
id: "item-template",
component: { Text: { text: { path: "/name" } } },
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
assert.strictEqual(plainTree.properties.children.length, 2);
// Check first generated child.
const child1 = plainTree.properties.children[0];
assert.strictEqual(child1.id, "item-template:0");
assert.strictEqual(child1.type, "Text");
assert.strictEqual(child1.dataContextPath, "/items/0");
assert.deepStrictEqual(child1.properties.text, { path: "name" });
// Check second generated child.
const child2 = plainTree.properties.children[1];
assert.strictEqual(child2.id, "item-template:1");
assert.strictEqual(child2.type, "Text");
assert.strictEqual(child2.dataContextPath, "/items/1");
assert.deepStrictEqual(child2.properties.text, { path: "name" });
});
it("should rebuild the tree when data for a template arrives later", () => {
processor.processMessages([
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
List: {
children: {
template: {
componentId: "item-template",
dataBinding: "/items",
},
},
},
},
},
{
id: "item-template",
component: { Text: { text: { path: "/name" } } },
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
let tree = processor.getSurfaces().get("@default")?.componentTree;
assert.strictEqual(toPlainObject(tree).properties.children.length, 0, "Children should be empty before data arrives");
// Now, the data arrives.
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "items",
valueString: JSON.stringify([{ name: "A" }, { name: "B" }]),
},
],
},
},
]);
tree = processor.getSurfaces().get("@default")?.componentTree;
assert.strictEqual(toPlainObject(tree).properties.children.length, 2, "Children should be populated after data arrives");
});
it("should trim relative paths within a data context (./item)", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "items",
valueString: JSON.stringify([{ name: "A" }, { name: "B" }]),
},
],
},
},
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
List: {
children: {
template: {
componentId: "item-template",
dataBinding: "/items",
},
},
},
},
},
// These paths would are typical when a databinding is used.
{
id: "item-template",
component: { Text: { text: { path: "./item/name" } } },
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
const child1 = plainTree.properties.children[0];
const child2 = plainTree.properties.children[1];
// The processor should have trimmed `/item` and `./` from the path
// because we are inside a data context.
assert.deepEqual(child1.properties.text, { path: "name" });
assert.deepEqual(child2.properties.text, { path: "name" });
});
it("should trim relative paths within a data context (./name)", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "items",
valueString: JSON.stringify([{ name: "A" }, { name: "B" }]),
},
],
},
},
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
List: {
children: {
template: {
componentId: "item-template",
dataBinding: "/items",
},
},
},
},
},
// These paths would are typical when a databinding is used.
{
id: "item-template",
component: { Text: { text: { path: "./name" } } },
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
const child1 = plainTree.properties.children[0];
const child2 = plainTree.properties.children[1];
// The processor should have trimmed `./` from the path
// because we are inside a data context.
assert.deepEqual(child1.properties.text, { path: "name" });
assert.deepEqual(child2.properties.text, { path: "name" });
});
});
describe("Data Normalization and Parsing", () => {
it("should correctly handle and parse the key-value array data format at the root", () => {
const messages = [
{
dataModelUpdate: {
surfaceId: "test-surface",
path: "/",
contents: [
{ key: "title", valueString: "My Title" },
{
key: "items",
valueString: '[{"id": 1}, {"id": 2}]',
},
],
},
},
];
processor.processMessages(messages);
const component = { dataContextPath: "/" };
const title = processor.getData(component, "/title", "test-surface");
const items = processor.getData(component, "/items", "test-surface");
assert.strictEqual(title, "My Title");
assert.deepStrictEqual(toPlainObject(items), [{ id: 1 }, { id: 2 }]);
});
it("should fallback to a string if stringified JSON is invalid", () => {
const invalidJSON = '[{"id": 1}, {"id": 2}'; // Missing closing bracket
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [{ key: "badData", valueString: invalidJSON }],
},
},
]);
const component = { dataContextPath: "/" };
const badData = processor.getData(component, "/badData");
assert.strictEqual(badData, invalidJSON);
});
});
describe("Complex Template Scenarios", () => {
it("should correctly expand a template with dataBinding to a Map (from valueMap)", () => {
const messages = [
{
beginRendering: {
surfaceId: "default",
root: "root-column",
},
},
{
surfaceUpdate: {
surfaceId: "default",
components: [
{
id: "root-column",
component: {
Column: {
children: {
explicitList: ["title-heading", "item-list"],
},
},
},
},
{
id: "title-heading",
component: {
Text: {
text: {
literalString: "Top Restaurants",
},
},
usageHint: "h1",
},
},
{
id: "item-list",
component: {
List: {
direction: "vertical",
children: {
template: {
componentId: "item-card-template",
dataBinding: "/items",
},
},
},
},
},
{
id: "item-card-template",
component: {
Card: {
child: "card-layout",
},
},
},
{
id: "card-layout",
component: {
Row: {
children: {
explicitList: ["template-image", "card-details"],
},
},
},
},
{
id: "template-image",
weight: 1,
component: {
Image: {
url: {
path: "imageUrl",
},
},
},
},
{
id: "card-details",
weight: 2,
component: {
Column: {
children: {
explicitList: [
"template-name",
"template-rating",
"template-detail",
"template-link",
"template-book-button",
],
},
},
},
},
{
id: "template-name",
component: {
Text: {
text: {
path: "name",
},
},
usageHint: "h3",
},
},
{
id: "template-rating",
component: {
Text: {
text: {
path: "rating",
},
},
},
},
{
id: "template-detail",
component: {
Text: {
text: {
path: "detail",
},
},
},
},
{
id: "template-link",
component: {
Text: {
text: {
path: "infoLink",
},
},
},
},
{
id: "template-book-button",
component: {
Button: {
child: "book-now-text",
action: {
name: "book_restaurant",
context: [
{
key: "restaurantName",
value: {
path: "name",
},
},
{
key: "imageUrl",
value: {
path: "imageUrl",
},
},
{
key: "address",
value: {
path: "address",
},
},
],
},
},
},
},
{
id: "book-now-text",
component: {
Text: {
text: {
literalString: "Book Now",
},
},
},
},
],
},
},
{
dataModelUpdate: {
surfaceId: "default",
path: "/",
contents: [
{
key: "items",
valueMap: [
{
key: "item1",
valueMap: [
{
key: "name",
valueString: "Business 1",
},
{
key: "rating",
valueString: "★★★★☆",
},
{
key: "detail",
valueString: "Spicy and savory hand-pulled noodles.",
},
{
key: "infoLink",
valueString: "[More Info](https://www.example.com/)",
},
{
key: "imageUrl",
valueString: "http://www.example.com/static/shrimpchowmein.jpeg",
},
{
key: "address",
valueString: "Address 1",
},
],
},
{
key: "item2",
valueMap: [
{
key: "name",
valueString: "Business 2",
},
{
key: "rating",
valueString: "★★★★☆",
},
{
key: "detail",
valueString: "Authentic and real.",
},
{
key: "infoLink",
valueString: "[More Info](https://www.example.com/)",
},
{
key: "imageUrl",
valueString: "http://www.example.com/static/mapotofu.jpeg",
},
{
key: "address",
valueString: "Address 2",
},
],
},
{
key: "item3",
valueMap: [
{
key: "name",
valueString: "Business 3",
},
{
key: "rating",
valueString: "★★★★☆",
},
{
key: "detail",
valueString: "Modern food with a farm-to-table approach.",
},
{
key: "infoLink",
valueString: "[More Info](https://www.example.com/)",
},
{
key: "imageUrl",
valueString: "http://www.example.com/static/beefbroccoli.jpeg",
},
{
key: "address",
valueString: "Address 3",
},
],
},
{
key: "item4",
valueMap: [
{
key: "name",
valueString: "Business 4",
},
{
key: "rating",
valueString: "★★★★★",
},
{
key: "detail",
valueString: "Upscale dining.",
},
{
key: "infoLink",
valueString: "[More Info](https://www.example.com/)",
},
{
key: "imageUrl",
valueString: "http://www.example.com/static/springrolls.jpeg",
},
{
key: "address",
valueString: "Address 4",
},
],
},
{
key: "item5",
valueMap: [
{
key: "name",
valueString: "Business 5",
},
{
key: "rating",
valueString: "★★★★☆",
},
{
key: "detail",
valueString: "Famous for its noodles.",
},
{
key: "infoLink",
valueString: "[More Info](https://www.example.com/)",
},
{
key: "imageUrl",
valueString: "http://www.example.com/static/kungpao.jpeg",
},
{
key: "address",
valueString: "Address 5",
},
],
},
],
},
],
},
},
];
processor.processMessages(messages);
const tree = processor.getSurfaces().get("default")?.componentTree;
const plainTree = toPlainObject(tree);
// 1. Find the "item-list" component (the List)
const itemList = plainTree.properties.children[1];
assert.strictEqual(itemList.id, "item-list");
// 2. Check that it expanded 5 children from the Map
const templateChildren = itemList.properties.children;
assert.strictEqual(templateChildren.length, 5);
// 3. Check the first generated child for correct key-based ID and data context
const child1 = templateChildren[0];
assert.strictEqual(child1.id, "item-card-template:item1");
assert.strictEqual(child1.dataContextPath, "/items/item1");
// 4. Go deeper to check the data binding on a nested component
// Path: Card -> Row -> Column -> Heading
const child1NameHeading = child1.properties.child.properties.children[1].properties.children[0];
assert.strictEqual(child1NameHeading.id, "template-name:item1");
assert.strictEqual(child1NameHeading.dataContextPath, "/items/item1");
assert.deepStrictEqual(child1NameHeading.properties.text, {
path: "name",
});
// 5. Check the second generated child
const child2 = templateChildren[1];
assert.strictEqual(child2.id, "item-card-template:item2");
assert.strictEqual(child2.dataContextPath, "/items/item2");
});
it("should correctly expand nested templates with layered data contexts", () => {
const messages = [
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "days",
// The correct way to send an array of objects is as a stringified JSON.
valueString: JSON.stringify([
{
title: "Day 1",
activities: ["Morning Walk", "Museum Visit"],
},
{
title: "Day 2",
activities: ["Market Trip"],
},
]),
},
],
},
},
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
List: {
children: {
template: {
componentId: "day-list",
dataBinding: "/days",
},
},
},
},
},
{
id: "day-list",
component: {
Column: {
children: { explicitList: ["day-title", "activity-list"] },
},
},
},
{
id: "day-title",
component: {
Text: { text: { path: "title" }, usageHint: "h1" },
},
},
{
id: "activity-list",
component: {
List: {
children: {
template: {
componentId: "activity-text",
dataBinding: "activities",
},
},
},
},
},
{
id: "activity-text",
component: { Text: { text: { path: "." } } },
},
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
];
processor.processMessages(messages);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
// Assert Day 1 structure
const day1 = plainTree.properties.children[0];
assert.strictEqual(day1.dataContextPath, "/days/0");
const day1Activities = day1.properties.children[1].properties.children;
assert.strictEqual(day1Activities.length, 2);
assert.strictEqual(day1Activities[0].id, "activity-text:0:0");
assert.strictEqual(day1Activities[0].dataContextPath, "/days/0/activities/0");
assert.deepStrictEqual(day1.properties.children[0].properties.text, {
path: "title",
});
assert.deepStrictEqual(day1Activities[0].properties.text, { path: "." });
// Assert Day 2 structure
const day2 = plainTree.properties.children[1];
assert.strictEqual(day2.dataContextPath, "/days/1");
const day2Activities = day2.properties.children[1].properties.children;
assert.strictEqual(day2Activities.length, 1);
assert.strictEqual(day2Activities[0].id, "activity-text:1:0");
assert.strictEqual(day2Activities[0].dataContextPath, "/days/1/activities/0");
assert.deepStrictEqual(day2.properties.children[0].properties.text, {
path: "title",
});
assert.deepStrictEqual(day2Activities[0].properties.text, { path: "." });
});
it("should correctly bind to primitive values in an array using path: '.'", () => {
processor.processMessages([
{
dataModelUpdate: {
surfaceId: "@default",
path: "/",
contents: [
{
key: "tags",
valueString: JSON.stringify(["travel", "paris", "guide"]),
},
],
},
},
{
surfaceUpdate: {
surfaceId: "@default",
components: [
{
id: "root",
component: {
Row: {
children: {
template: { componentId: "tag", dataBinding: "/tags" },
},
},
},
},
{ id: "tag", component: { Text: { text: { path: "." } } } },
],
},
},
{
beginRendering: {
root: "root",
surfaceId: "@default",
},
},
]);
const tree = processor.getSurfaces().get("@default")?.componentTree;
const plainTree = toPlainObject(tree);
const children = plainTree.properties.children;
assert.strictEqual(children.length, 3);
assert.strictEqual(children[0].dataContextPath, "/tags/0");
assert.deepEqual(children[0].properties.text, { path: "." });
assert.strictEqual(children[1].dataContextPath, "/tags/1");
assert.deepEqual(children[1].properties.text, { path: "." });
});
});
describe("Multi-Surface Interaction", () => {
it("should keep data and components for different surfaces separate", () => {
processor.processMessages([
// Surface A
{
dataModelUpdate: {
surfaceId: "A",
path: "/",
contents: [{ key: "name", valueString: "Surface A Data" }],
},
},
{
surfaceUpdate: {
surfaceId: "A",
components: [
{
id: "comp-a",
component: { Text: { text: { path: "/name" } } },
},
],
},
},
{ beginRendering: { root: "comp-a", surfaceId: "A" } },
// Surface B
{
dataModelUpdate: {
surfaceId: "B",
path: "/",
contents: [{ key: "name", valueString: "Surface B Data" }],
},
},
{
surfaceUpdate: {
surfaceId: "B",
components: [
{
id: "comp-b",
component: { Text: { text: { path: "/name" } } },
},
],
},
},
{ beginRendering: { root: "comp-b", surfaceId: "B" } },
]);
const surfaces = processor.getSurfaces();
assert.strictEqual(surfaces.size, 2);
const surfaceA = surfaces.get("A");
const surfaceB = surfaces.get("B");
assert.ok(surfaceA && surfaceB, "Both surfaces should exist");
// Check Surface A
assert.ok(surfaceA, "Surface A exists.");
assert.strictEqual(surfaceA.components.size, 1);
assert.ok(surfaceA.components.has("comp-a"));
assert.deepStrictEqual(toPlainObject(surfaceA.dataModel), {
name: "Surface A Data",
});
assert.deepStrictEqual(toPlainObject(surfaceA.componentTree).properties.text, { path: "/name" });
// Check Surface B
assert.ok(surfaceB, "Surface B exists.");
assert.strictEqual(surfaceB.components.size, 1);
assert.ok(surfaceB.components.has("comp-b"));
assert.deepStrictEqual(toPlainObject(surfaceB.dataModel), {
name: "Surface B Data",
});
assert.deepStrictEqual(toPlainObject(surfaceB.componentTree).properties.text, { path: "/name" });
});
});
});
function assertIsDataMap(obj) {
assert.ok(obj instanceof Map, `Data should be a DataMap`);
}
//# sourceMappingURL=model.test.js.map