1 | import { getNodeKeyForPreboot } from '../common/get-node-key';
|
2 | export function _window() {
|
3 | return {
|
4 | prebootData: window['prebootData'],
|
5 | getComputedStyle: window.getComputedStyle,
|
6 | document: document
|
7 | };
|
8 | }
|
9 | export class EventReplayer {
|
10 | constructor() {
|
11 | this.clientNodeCache = {};
|
12 | this.replayStarted = false;
|
13 | }
|
14 | /**
|
15 | * Window setting and getter to facilitate testing of window
|
16 | * in non-browser environments
|
17 | */
|
18 | setWindow(win) {
|
19 | this.win = win;
|
20 | }
|
21 | /**
|
22 | * Window setting and getter to facilitate testing of window
|
23 | * in non-browser environments
|
24 | */
|
25 | getWindow() {
|
26 | if (!this.win) {
|
27 | this.win = _window();
|
28 | }
|
29 | return this.win;
|
30 | }
|
31 | /**
|
32 | * Replay all events for all apps. this can only be run once.
|
33 | * if called multiple times, will only do something once
|
34 | */
|
35 | replayAll() {
|
36 | if (this.replayStarted) {
|
37 | return;
|
38 | }
|
39 | else {
|
40 | this.replayStarted = true;
|
41 | }
|
42 | // loop through each of the preboot apps
|
43 | const prebootData = this.getWindow().prebootData || {};
|
44 | const apps = prebootData.apps || [];
|
45 | apps.forEach(appData => this.replayForApp(appData));
|
46 | // once all events have been replayed and buffers switched, then we cleanup preboot
|
47 | this.cleanup(prebootData);
|
48 | }
|
49 | /**
|
50 | * Replay all events for one app (most of the time there is just one app)
|
51 | * @param appData
|
52 | */
|
53 | replayForApp(appData) {
|
54 | appData = (appData || {});
|
55 | // try catch around events b/c even if error occurs, we still move forward
|
56 | try {
|
57 | const events = appData.events || [];
|
58 | // replay all the events from the server view onto the client view
|
59 | events.forEach(event => this.replayEvent(appData, event));
|
60 | }
|
61 | catch (ex) {
|
62 | console.error(ex);
|
63 | }
|
64 | // if we are buffering, switch the buffers
|
65 | this.switchBuffer(appData);
|
66 | }
|
67 | /**
|
68 | * Replay one particular event
|
69 | * @param appData
|
70 | * @param prebootEvent
|
71 | */
|
72 | replayEvent(appData, prebootEvent) {
|
73 | appData = (appData || {});
|
74 | prebootEvent = (prebootEvent || {});
|
75 | const event = prebootEvent.event;
|
76 | const serverNode = prebootEvent.node || {};
|
77 | const nodeKey = prebootEvent.nodeKey;
|
78 | const clientNode = this.findClientNode({
|
79 | root: appData.root,
|
80 | node: serverNode,
|
81 | nodeKey: nodeKey
|
82 | });
|
83 | // if client node can't be found, log a warning
|
84 | if (!clientNode) {
|
85 | console.warn(`Trying to dispatch event ${event.type} to node ${nodeKey}
|
86 | but could not find client node. Server node is: ${serverNode}`);
|
87 | return;
|
88 | }
|
89 | // now dispatch events and whatnot to the client node
|
90 | clientNode.checked = serverNode.checked;
|
91 | clientNode.selected = serverNode.selected;
|
92 | clientNode.value = serverNode.value;
|
93 | clientNode.dispatchEvent(event);
|
94 | }
|
95 | /**
|
96 | * Switch the buffer for one particular app (i.e. display the client
|
97 | * view and destroy the server view)
|
98 | * @param appData
|
99 | */
|
100 | switchBuffer(appData) {
|
101 | appData = (appData || {});
|
102 | const root = (appData.root || {});
|
103 | const serverView = root.serverNode;
|
104 | const clientView = root.clientNode;
|
105 | // if no client view or the server view is the body or client
|
106 | // and server view are the same, then don't do anything and return
|
107 | if (!clientView || !serverView || serverView === clientView || serverView.nodeName === 'BODY') {
|
108 | return;
|
109 | }
|
110 | // do a try-catch just in case something messed up
|
111 | try {
|
112 | // get the server view display mode
|
113 | const gcs = this.getWindow().getComputedStyle;
|
114 | const display = gcs(serverView).getPropertyValue('display') || 'block';
|
115 | // first remove the server view
|
116 | serverView.remove ? serverView.remove() : (serverView.style.display = 'none');
|
117 | // now add the client view
|
118 | clientView.style.display = display;
|
119 | }
|
120 | catch (ex) {
|
121 | console.error(ex);
|
122 | }
|
123 | }
|
124 | /**
|
125 | * Finally, set focus, remove all the event listeners and remove
|
126 | * any freeze screen that may be there
|
127 | * @param prebootData
|
128 | */
|
129 | cleanup(prebootData) {
|
130 | prebootData = prebootData || {};
|
131 | const listeners = prebootData.listeners || [];
|
132 | // set focus on the active node AFTER a small delay to ensure buffer
|
133 | // switched
|
134 | const activeNode = prebootData.activeNode;
|
135 | if (activeNode != null) {
|
136 | setTimeout(() => this.setFocus(activeNode), 1);
|
137 | }
|
138 | // remove all event listeners
|
139 | for (const listener of listeners) {
|
140 | listener.node.removeEventListener(listener.eventName, listener.handler);
|
141 | }
|
142 | // remove the freeze overlay if it exists
|
143 | const doc = this.getWindow().document;
|
144 | const prebootOverlay = doc.getElementById('prebootOverlay');
|
145 | if (prebootOverlay) {
|
146 | prebootOverlay.remove ?
|
147 | prebootOverlay.remove() : prebootOverlay.parentNode !== null ?
|
148 | prebootOverlay.parentNode.removeChild(prebootOverlay) :
|
149 | prebootOverlay.style.display = 'none';
|
150 | }
|
151 | // clear out the data stored for each app
|
152 | prebootData.apps = [];
|
153 | this.clientNodeCache = {};
|
154 | // send event to document that signals preboot complete
|
155 | // constructor is not supported by older browsers ( i.e. IE9-11 )
|
156 | // in these browsers, the type of CustomEvent will be "object"
|
157 | if (typeof CustomEvent === 'function') {
|
158 | const completeEvent = new CustomEvent('PrebootComplete');
|
159 | doc.dispatchEvent(completeEvent);
|
160 | }
|
161 | else {
|
162 | console.warn(`Could not dispatch PrebootComplete event.
|
163 | You can fix this by including a polyfill for CustomEvent.`);
|
164 | }
|
165 | }
|
166 | setFocus(activeNode) {
|
167 | // only do something if there is an active node
|
168 | if (!activeNode || !activeNode.node || !activeNode.nodeKey) {
|
169 | return;
|
170 | }
|
171 | // find the client node in the new client view
|
172 | const clientNode = this.findClientNode(activeNode);
|
173 | if (clientNode) {
|
174 | // set focus on the client node
|
175 | clientNode.focus();
|
176 | // set selection if a modern browser (i.e. IE9+, etc.)
|
177 | const selection = activeNode.selection;
|
178 | if (clientNode.setSelectionRange && selection) {
|
179 | try {
|
180 | clientNode
|
181 | .setSelectionRange(selection.start, selection.end, selection.direction);
|
182 | }
|
183 | catch (ex) { }
|
184 | }
|
185 | }
|
186 | }
|
187 | /**
|
188 | * Given a node from the server rendered view, find the equivalent
|
189 | * node in the client rendered view. We do this by the following approach:
|
190 | * 1. take the name of the server node tag (ex. div or h1 or input)
|
191 | * 2. add either id (ex. div#myid) or class names (ex. div.class1.class2)
|
192 | * 3. use that value as a selector to get all the matching client nodes
|
193 | * 4. loop through all client nodes found and for each generate a key value
|
194 | * 5. compare the client key to the server key; once there is a match,
|
195 | * we have our client node
|
196 | *
|
197 | * NOTE: this only works when the client view is almost exactly the same as
|
198 | * the server view. we will need an improvement here in the future to account
|
199 | * for situations where the client view is different in structure from the
|
200 | * server view
|
201 | */
|
202 | findClientNode(serverNodeContext) {
|
203 | serverNodeContext = (serverNodeContext || {});
|
204 | const serverNode = serverNodeContext.node;
|
205 | const root = serverNodeContext.root;
|
206 | // if no server or client root, don't do anything
|
207 | if (!root || !root.serverNode || !root.clientNode) {
|
208 | return null;
|
209 | }
|
210 | // we use the string of the node to compare to the client node & as key in
|
211 | // cache
|
212 | const serverNodeKey = serverNodeContext.nodeKey || getNodeKeyForPreboot(serverNodeContext);
|
213 | // if client node already in cache, return it
|
214 | if (this.clientNodeCache[serverNodeKey]) {
|
215 | return this.clientNodeCache[serverNodeKey];
|
216 | }
|
217 | // get the selector for client nodes
|
218 | const className = (serverNode.className || '').replace('ng-binding', '').trim();
|
219 | let selector = serverNode.tagName;
|
220 | if (serverNode.id) {
|
221 | selector += `#${serverNode.id}`;
|
222 | }
|
223 | else if (className) {
|
224 | selector += `.${className.replace(/ /g, '.')}`;
|
225 | }
|
226 | // select all possible client nodes and look through them to try and find a
|
227 | // match
|
228 | const rootClientNode = root.clientNode;
|
229 | let clientNodes = rootClientNode.querySelectorAll(selector);
|
230 | // if nothing found, then just try the tag name as a final option
|
231 | if (!clientNodes.length) {
|
232 | console.log(`nothing found for ${selector} so using ${serverNode.tagName}`);
|
233 | clientNodes = rootClientNode.querySelectorAll(serverNode.tagName);
|
234 | }
|
235 | const length = clientNodes.length;
|
236 | for (let i = 0; i < length; i++) {
|
237 | const clientNode = clientNodes.item(i);
|
238 | // get the key for the client node
|
239 | const clientNodeKey = getNodeKeyForPreboot({
|
240 | root: root,
|
241 | node: clientNode
|
242 | });
|
243 | // if the client node key is exact match for the server node key, then we
|
244 | // found the client node
|
245 | if (clientNodeKey === serverNodeKey) {
|
246 | this.clientNodeCache[serverNodeKey] = clientNode;
|
247 | return clientNode;
|
248 | }
|
249 | }
|
250 | // if we get here and there is one clientNode, use it as a fallback
|
251 | if (clientNodes.length === 1) {
|
252 | this.clientNodeCache[serverNodeKey] = clientNodes[0];
|
253 | return clientNodes[0];
|
254 | }
|
255 | // if we get here it means we couldn't find the client node so give the user
|
256 | // a warning
|
257 | console.warn(`No matching client node found for ${serverNodeKey}.
|
258 | You can fix this by assigning this element a unique id attribute.`);
|
259 | return null;
|
260 | }
|
261 | }
|
262 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"event.replayer.js","sourceRoot":"../../src/lib/","sources":["api/event.replayer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport {\n  NodeContext,\n  PrebootAppData,\n  PrebootData,\n  PrebootEvent,\n  PrebootWindow,\n  ServerClientRoot,\n} from '../common/preboot.interfaces';\nimport {getNodeKeyForPreboot} from '../common/get-node-key';\n\nexport function _window(): PrebootWindow {\n  return {\n    prebootData: (window as any)['prebootData'],\n    getComputedStyle: window.getComputedStyle,\n    document: document\n  };\n}\n\nexport class EventReplayer {\n  clientNodeCache: { [key: string]: Element } = {};\n  replayStarted = false;\n  win: PrebootWindow;\n\n  /**\n   * Window setting and getter to facilitate testing of window\n   * in non-browser environments\n   */\n  setWindow(win: PrebootWindow) {\n    this.win = win;\n  }\n\n  /**\n   * Window setting and getter to facilitate testing of window\n   * in non-browser environments\n   */\n  getWindow() {\n    if (!this.win) {\n      this.win = _window();\n    }\n    return this.win;\n  }\n\n  /**\n   * Replay all events for all apps. this can only be run once.\n   * if called multiple times, will only do something once\n   */\n  replayAll() {\n    if (this.replayStarted) {\n      return;\n    } else {\n      this.replayStarted = true;\n    }\n\n    // loop through each of the preboot apps\n    const prebootData = this.getWindow().prebootData || {};\n    const apps = prebootData.apps || [];\n    apps.forEach(appData => this.replayForApp(appData));\n\n    // once all events have been replayed and buffers switched, then we cleanup preboot\n    this.cleanup(prebootData);\n  }\n\n  /**\n   * Replay all events for one app (most of the time there is just one app)\n   * @param appData\n   */\n  replayForApp(appData: PrebootAppData) {\n    appData = <PrebootAppData>(appData || {});\n\n    // try catch around events b/c even if error occurs, we still move forward\n    try {\n      const events = appData.events || [];\n\n      // replay all the events from the server view onto the client view\n      events.forEach(event => this.replayEvent(appData, event));\n    } catch (ex) {\n      console.error(ex);\n    }\n\n    // if we are buffering, switch the buffers\n    this.switchBuffer(appData);\n  }\n\n  /**\n   * Replay one particular event\n   * @param appData\n   * @param prebootEvent\n   */\n  replayEvent(appData: PrebootAppData, prebootEvent: PrebootEvent) {\n    appData = <PrebootAppData>(appData || {});\n    prebootEvent = <PrebootEvent>(prebootEvent || {});\n\n    const event = prebootEvent.event as Event;\n    const serverNode = prebootEvent.node || {};\n    const nodeKey = prebootEvent.nodeKey;\n    const clientNode = this.findClientNode({\n      root: appData.root,\n      node: serverNode,\n      nodeKey: nodeKey\n    });\n\n    // if client node can't be found, log a warning\n    if (!clientNode) {\n      console.warn(\n        `Trying to dispatch event ${event.type} to node ${nodeKey}\n        but could not find client node. Server node is: ${serverNode}`\n      );\n      return;\n    }\n\n    // now dispatch events and whatnot to the client node\n    (clientNode as HTMLInputElement).checked = serverNode.checked;\n    (clientNode as HTMLOptionElement).selected = serverNode.selected;\n    (clientNode as HTMLOptionElement).value = serverNode.value;\n    clientNode.dispatchEvent(event);\n  }\n\n  /**\n   * Switch the buffer for one particular app (i.e. display the client\n   * view and destroy the server view)\n   * @param appData\n   */\n  switchBuffer(appData: PrebootAppData) {\n    appData = <PrebootAppData>(appData || {});\n\n    const root = <ServerClientRoot>(appData.root || {});\n    const serverView = root.serverNode;\n    const clientView = root.clientNode;\n\n    // if no client view or the server view is the body or client\n    // and server view are the same, then don't do anything and return\n    if (!clientView || !serverView || serverView === clientView || serverView.nodeName === 'BODY') {\n      return;\n    }\n\n    // do a try-catch just in case something messed up\n    try {\n      // get the server view display mode\n      const gcs = this.getWindow().getComputedStyle;\n      const display = gcs(serverView).getPropertyValue('display') || 'block';\n\n      // first remove the server view\n      serverView.remove ? serverView.remove() : (serverView.style.display = 'none');\n\n      // now add the client view\n      clientView.style.display = display;\n    } catch (ex) {\n      console.error(ex);\n    }\n  }\n\n  /**\n   * Finally, set focus, remove all the event listeners and remove\n   * any freeze screen that may be there\n   * @param prebootData\n   */\n  cleanup(prebootData: PrebootData) {\n    prebootData = prebootData || {};\n\n    const listeners = prebootData.listeners || [];\n\n    // set focus on the active node AFTER a small delay to ensure buffer\n    // switched\n    const activeNode = prebootData.activeNode;\n    if (activeNode != null) {\n      setTimeout(() => this.setFocus(activeNode), 1);\n    }\n\n    // remove all event listeners\n    for (const listener of listeners) {\n      listener.node.removeEventListener(listener.eventName, listener.handler);\n    }\n\n    // remove the freeze overlay if it exists\n    const doc = this.getWindow().document;\n    const prebootOverlay = doc.getElementById('prebootOverlay');\n    if (prebootOverlay) {\n      prebootOverlay.remove ?\n        prebootOverlay.remove() : prebootOverlay.parentNode !== null ?\n        prebootOverlay.parentNode.removeChild(prebootOverlay) :\n        prebootOverlay.style.display = 'none';\n    }\n\n    // clear out the data stored for each app\n    prebootData.apps = [];\n    this.clientNodeCache = {};\n\n    // send event to document that signals preboot complete\n    // constructor is not supported by older browsers ( i.e. IE9-11 )\n    // in these browsers, the type of CustomEvent will be \"object\"\n    if (typeof CustomEvent === 'function') {\n      const completeEvent = new CustomEvent('PrebootComplete');\n      doc.dispatchEvent(completeEvent);\n    } else {\n      console.warn(`Could not dispatch PrebootComplete event.\n       You can fix this by including a polyfill for CustomEvent.`);\n    }\n  }\n\n  setFocus(activeNode: NodeContext) {\n    // only do something if there is an active node\n    if (!activeNode || !activeNode.node || !activeNode.nodeKey) {\n      return;\n    }\n\n    // find the client node in the new client view\n    const clientNode = this.findClientNode(activeNode);\n    if (clientNode) {\n      // set focus on the client node\n      clientNode.focus();\n\n      // set selection if a modern browser (i.e. IE9+, etc.)\n      const selection = activeNode.selection;\n      if ((clientNode as HTMLInputElement).setSelectionRange && selection) {\n        try {\n          (clientNode as HTMLInputElement)\n            .setSelectionRange(selection.start, selection.end, selection.direction);\n        } catch (ex) {}\n      }\n    }\n  }\n\n  /**\n   * Given a node from the server rendered view, find the equivalent\n   * node in the client rendered view. We do this by the following approach:\n   *      1. take the name of the server node tag (ex. div or h1 or input)\n   *      2. add either id (ex. div#myid) or class names (ex. div.class1.class2)\n   *      3. use that value as a selector to get all the matching client nodes\n   *      4. loop through all client nodes found and for each generate a key value\n   *      5. compare the client key to the server key; once there is a match,\n   *          we have our client node\n   *\n   * NOTE: this only works when the client view is almost exactly the same as\n   * the server view. we will need an improvement here in the future to account\n   * for situations where the client view is different in structure from the\n   * server view\n   */\n  findClientNode(serverNodeContext: NodeContext): HTMLElement | null {\n    serverNodeContext = <NodeContext>(serverNodeContext || {});\n\n    const serverNode = serverNodeContext.node;\n    const root = serverNodeContext.root;\n\n    // if no server or client root, don't do anything\n    if (!root || !root.serverNode || !root.clientNode) {\n      return null;\n    }\n\n    // we use the string of the node to compare to the client node & as key in\n    // cache\n    const serverNodeKey = serverNodeContext.nodeKey || getNodeKeyForPreboot(serverNodeContext);\n\n    // if client node already in cache, return it\n    if (this.clientNodeCache[serverNodeKey]) {\n      return this.clientNodeCache[serverNodeKey] as HTMLElement;\n    }\n\n    // get the selector for client nodes\n    const className = (serverNode.className || '').replace('ng-binding', '').trim();\n    let selector = serverNode.tagName;\n\n    if (serverNode.id) {\n      selector += `#${serverNode.id}`;\n    } else if (className) {\n      selector += `.${className.replace(/ /g, '.')}`;\n    }\n\n    // select all possible client nodes and look through them to try and find a\n    // match\n    const rootClientNode = root.clientNode;\n    let clientNodes = rootClientNode.querySelectorAll(selector);\n\n    // if nothing found, then just try the tag name as a final option\n    if (!clientNodes.length) {\n      console.log(`nothing found for ${selector} so using ${serverNode.tagName}`);\n      clientNodes = rootClientNode.querySelectorAll(serverNode.tagName);\n    }\n\n    const length = clientNodes.length;\n    for (let i = 0; i < length; i++) {\n      const clientNode = clientNodes.item(i);\n\n      // get the key for the client node\n      const clientNodeKey = getNodeKeyForPreboot({\n        root: root,\n        node: clientNode\n      });\n\n      // if the client node key is exact match for the server node key, then we\n      // found the client node\n      if (clientNodeKey === serverNodeKey) {\n        this.clientNodeCache[serverNodeKey] = clientNode;\n        return clientNode as HTMLElement;\n      }\n    }\n\n    // if we get here and there is one clientNode, use it as a fallback\n    if (clientNodes.length === 1) {\n      this.clientNodeCache[serverNodeKey] = clientNodes[0];\n      return clientNodes[0] as HTMLElement;\n    }\n\n    // if we get here it means we couldn't find the client node so give the user\n    // a warning\n    console.warn(\n      `No matching client node found for ${serverNodeKey}.\n       You can fix this by assigning this element a unique id attribute.`\n    );\n    return null;\n  }\n}\n"]} |
\ | No newline at end of file |