UNPKG

35.9 kBJavaScriptView Raw
1import { getNodeKeyForPreboot } from '../common/get-node-key';
2/**
3 * Called right away to initialize preboot
4 *
5 * @param opts All the preboot options
6 * @param win
7 */
8export function initAll(opts, win) {
9 const theWindow = (win || window);
10 // Add the preboot options to the preboot data and then add the data to
11 // the window so it can be used later by the client.
12 // Only set new options if they're not already set - we may have multiple app roots
13 // and each of them invokes the init function separately.
14 const data = (theWindow.prebootData = {
15 opts: opts,
16 apps: [],
17 listeners: []
18 });
19 return () => start(data, theWindow);
20}
21/**
22 * Start up preboot by going through each app and assigning the appropriate
23 * handlers. Normally this wouldn't be called directly, but we have set it up so
24 * that it can for older versions of Universal.
25 *
26 * @param prebootData Global preboot data object that contains options and will
27 * have events
28 * @param win Optional param to pass in mock window for testing purposes
29 */
30export function start(prebootData, win) {
31 const theWindow = (win || window);
32 const _document = (theWindow.document || {});
33 // Remove the current script from the DOM so that child indexes match
34 // between the client & the server. The script is already running so it
35 // doesn't affect it.
36 const currentScript = _document.currentScript ||
37 // Support: IE 9-11 only
38 // IE doesn't support document.currentScript. Since the script is invoked
39 // synchronously, though, the current running script is just the last one
40 // currently in the document.
41 [].slice.call(_document.getElementsByTagName('script'), -1)[0];
42 if (!currentScript) {
43 console.error('Preboot initialization failed, no currentScript has been detected.');
44 return;
45 }
46 let serverNode = currentScript.parentNode;
47 if (!serverNode) {
48 console.error('Preboot initialization failed, the script is detached');
49 return;
50 }
51 serverNode.removeChild(currentScript);
52 const opts = prebootData.opts || {};
53 let eventSelectors = opts.eventSelectors || [];
54 // get the root info
55 const appRoot = prebootData.opts ? getAppRoot(_document, prebootData.opts, serverNode) : null;
56 // we track all events for each app in the prebootData object which is on
57 // the global scope; each `start` invocation adds data for one app only.
58 const appData = { root: appRoot, events: [] };
59 if (prebootData.apps) {
60 prebootData.apps.push(appData);
61 }
62 eventSelectors = eventSelectors.map(eventSelector => {
63 if (!eventSelector.hasOwnProperty('replay')) {
64 eventSelector.replay = true;
65 }
66 return eventSelector;
67 });
68 // loop through all the eventSelectors and create event handlers
69 eventSelectors.forEach(eventSelector => handleEvents(_document, prebootData, appData, eventSelector));
70}
71/**
72 * Create an overlay div and add it to the DOM so it can be used
73 * if a freeze event occurs
74 *
75 * @param _document The global document object (passed in for testing purposes)
76 * @returns Element The overlay node is returned
77 */
78export function createOverlay(_document) {
79 let overlay = _document.createElement('div');
80 overlay.setAttribute('id', 'prebootOverlay');
81 overlay.setAttribute('style', 'display:none;position:absolute;left:0;' +
82 'top:0;width:100%;height:100%;z-index:999999;background:black;opacity:.3');
83 _document.documentElement.appendChild(overlay);
84 return overlay;
85}
86/**
87 * Get references to the current app root node based on input options. Users can
88 * initialize preboot either by specifying appRoot which is just one or more
89 * selectors for apps. This section option is useful for people that are doing their own
90 * buffering (i.e. they have their own client and server view)
91 *
92 * @param _document The global document object used to attach the overlay
93 * @param opts Options passed in by the user to init()
94 * @param serverNode The server node serving as application root
95 * @returns ServerClientRoot An array of root info for the current app
96 */
97export function getAppRoot(_document, opts, serverNode) {
98 const root = { serverNode };
99 // if we are doing buffering, we need to create the buffer for the client
100 // else the client root is the same as the server
101 root.clientNode = opts.buffer ? createBuffer(root) : root.serverNode;
102 // create an overlay if not disabled ,that can be used later if a freeze event occurs
103 if (!opts.disableOverlay) {
104 root.overlay = createOverlay(_document);
105 }
106 return root;
107}
108/**
109 * Under given server root, for given selector, record events
110 *
111 * @param _document
112 * @param prebootData
113 * @param appData
114 * @param eventSelector
115 */
116export function handleEvents(_document, prebootData, appData, eventSelector) {
117 const serverRoot = appData.root.serverNode;
118 // don't do anything if no server root
119 if (!serverRoot) {
120 return;
121 }
122 // Attach delegated event listeners for each event selector.
123 // We need to use delegated events as only the top level server node
124 // exists at this point.
125 eventSelector.events.forEach((eventName) => {
126 // get the appropriate handler and add it as an event listener
127 const handler = createListenHandler(_document, prebootData, eventSelector, appData);
128 // attach the handler in the capture phase so that it fires even if
129 // one of the handlers below calls stopPropagation()
130 serverRoot.addEventListener(eventName, handler, true);
131 // need to keep track of listeners so we can do node.removeEventListener()
132 // when preboot done
133 if (prebootData.listeners) {
134 prebootData.listeners.push({
135 node: serverRoot,
136 eventName,
137 handler
138 });
139 }
140 });
141}
142/**
143 * Create handler for events that we will record
144 */
145export function createListenHandler(_document, prebootData, eventSelector, appData) {
146 const CARET_EVENTS = ['keyup', 'keydown', 'focusin', 'mouseup', 'mousedown'];
147 const CARET_NODES = ['INPUT', 'TEXTAREA'];
148 // Support: IE 9-11 only
149 // IE uses a prefixed `matches` version
150 const matches = _document.documentElement.matches ||
151 _document.documentElement.msMatchesSelector;
152 const opts = prebootData.opts;
153 return function (event) {
154 const node = event.target;
155 // a delegated handlers on document is used so we need to check if
156 // event target matches a desired selector
157 if (!matches.call(node, eventSelector.selector)) {
158 return;
159 }
160 const root = appData.root;
161 const eventName = event.type;
162 // if no node or no event name, just return
163 if (!node || !eventName) {
164 return;
165 }
166 // if key codes set for eventSelector, then don't do anything if event
167 // doesn't include key
168 const keyCodes = eventSelector.keyCodes;
169 if (keyCodes && keyCodes.length) {
170 const matchingKeyCodes = keyCodes.filter(keyCode => event.which === keyCode);
171 // if there are not matches (i.e. key entered NOT one of the key codes)
172 // then don't do anything
173 if (!matchingKeyCodes.length) {
174 return;
175 }
176 }
177 // if for a given set of events we are preventing default, do that
178 if (eventSelector.preventDefault) {
179 event.preventDefault();
180 }
181 // if an action handler passed in, use that
182 if (eventSelector.action) {
183 eventSelector.action(node, event);
184 }
185 // get the node key for a given node
186 const nodeKey = getNodeKeyForPreboot({ root: root, node: node });
187 // record active node
188 if (CARET_EVENTS.indexOf(eventName) >= 0) {
189 // if it's an caret node, get the selection for the active node
190 const isCaretNode = CARET_NODES.indexOf(node.tagName ? node.tagName : '') >= 0;
191 prebootData.activeNode = {
192 root: root,
193 node: node,
194 nodeKey: nodeKey,
195 selection: isCaretNode ? getSelection(node) : undefined
196 };
197 }
198 else if (eventName !== 'change' && eventName !== 'focusout') {
199 prebootData.activeNode = undefined;
200 }
201 // if overlay is not disabled and we are freezing the UI
202 if (opts && !opts.disableOverlay && eventSelector.freeze) {
203 const overlay = root.overlay;
204 // show the overlay
205 overlay.style.display = 'block';
206 // hide the overlay after 10 seconds just in case preboot.complete() never
207 // called
208 setTimeout(() => {
209 overlay.style.display = 'none';
210 }, 10000);
211 }
212 // we will record events for later replay unless explicitly marked as
213 // doNotReplay
214 if (eventSelector.replay) {
215 appData.events.push({
216 node,
217 nodeKey,
218 event,
219 name: eventName
220 });
221 }
222 };
223}
224/**
225 * Get the selection data that is later used to set the cursor after client view
226 * is active
227 */
228export function getSelection(node) {
229 node = node || {};
230 const nodeValue = node.value || '';
231 const selection = {
232 start: nodeValue.length,
233 end: nodeValue.length,
234 direction: 'forward'
235 };
236 // if browser support selectionStart on node (Chrome, FireFox, IE9+)
237 try {
238 if (node.selectionStart || node.selectionStart === 0) {
239 selection.start = node.selectionStart;
240 selection.end = node.selectionEnd ? node.selectionEnd : 0;
241 selection.direction = node.selectionDirection ?
242 node.selectionDirection : 'none';
243 }
244 }
245 catch (ex) { }
246 return selection;
247}
248/**
249 * Create buffer for a given node
250 *
251 * @param root All the data related to a particular app
252 * @returns Returns the root client node.
253 */
254export function createBuffer(root) {
255 const serverNode = root.serverNode;
256 // if no rootServerNode OR the selector is on the entire html doc or the body
257 // OR no parentNode, don't buffer
258 if (!serverNode || !serverNode.parentNode ||
259 serverNode === document.documentElement || serverNode === document.body) {
260 return serverNode;
261 }
262 // create shallow clone of server root
263 const rootClientNode = serverNode.cloneNode(false);
264 // we want the client to write to a hidden div until the time for switching
265 // the buffers
266 rootClientNode.style.display = 'none';
267 // insert the client node before the server and return it
268 serverNode.parentNode.insertBefore(rootClientNode, serverNode);
269 // mark server node as not to be touched by AngularJS - needed for ngUpgrade
270 serverNode.setAttribute('ng-non-bindable', '');
271 // return the rootClientNode
272 return rootClientNode;
273}
274//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"event.recorder.js","sourceRoot":"../../src/lib/","sources":["api/event.recorder.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  EventSelector,\n  PrebootOptions,\n  PrebootAppData,\n  PrebootData,\n  DomEvent,\n  PrebootWindow,\n  ServerClientRoot,\n  PrebootSelection,\n  PrebootSelectionDirection,\n} from '../common/preboot.interfaces';\nimport {getNodeKeyForPreboot} from '../common/get-node-key';\n\n/**\n * Called right away to initialize preboot\n *\n * @param opts All the preboot options\n * @param win\n */\nexport function initAll(opts: PrebootOptions, win?: PrebootWindow) {\n  const theWindow = <PrebootWindow>(win || window);\n\n  // Add the preboot options to the preboot data and then add the data to\n  // the window so it can be used later by the client.\n  // Only set new options if they're not already set - we may have multiple app roots\n  // and each of them invokes the init function separately.\n  const data = (theWindow.prebootData = <PrebootData>{\n    opts: opts,\n    apps: [],\n    listeners: []\n  });\n\n  return () => start(data, theWindow);\n}\n\n/**\n * Start up preboot by going through each app and assigning the appropriate\n * handlers. Normally this wouldn't be called directly, but we have set it up so\n * that it can for older versions of Universal.\n *\n * @param prebootData Global preboot data object that contains options and will\n * have events\n * @param win Optional param to pass in mock window for testing purposes\n */\nexport function start(prebootData: PrebootData, win?: PrebootWindow) {\n  const theWindow = <PrebootWindow>(win || window);\n  const _document = <Document>(theWindow.document || {});\n\n  // Remove the current script from the DOM so that child indexes match\n  // between the client & the server. The script is already running so it\n  // doesn't affect it.\n  const currentScript = _document.currentScript ||\n    // Support: IE 9-11 only\n    // IE doesn't support document.currentScript. Since the script is invoked\n    // synchronously, though, the current running script is just the last one\n    // currently in the document.\n    [].slice.call(_document.getElementsByTagName('script'), -1)[0];\n\n  if (!currentScript) {\n    console.error('Preboot initialization failed, no currentScript has been detected.');\n    return;\n  }\n\n  let serverNode = currentScript.parentNode;\n  if (!serverNode) {\n    console.error('Preboot initialization failed, the script is detached');\n    return;\n  }\n\n  serverNode.removeChild(currentScript);\n\n  const opts = prebootData.opts || ({} as PrebootOptions);\n  let eventSelectors = opts.eventSelectors || [];\n\n  // get the root info\n  const appRoot = prebootData.opts ? getAppRoot(_document, prebootData.opts, serverNode) : null;\n\n  // we track all events for each app in the prebootData object which is on\n  // the global scope; each `start` invocation adds data for one app only.\n  const appData = <PrebootAppData>{ root: appRoot, events: [] };\n  if (prebootData.apps) {\n    prebootData.apps.push(appData);\n  }\n\n  eventSelectors = eventSelectors.map(eventSelector => {\n    if (!eventSelector.hasOwnProperty('replay')) {\n      eventSelector.replay = true;\n    }\n    return eventSelector;\n  });\n\n  // loop through all the eventSelectors and create event handlers\n  eventSelectors.forEach(eventSelector =>\n    handleEvents(_document, prebootData, appData, eventSelector));\n}\n\n/**\n * Create an overlay div and add it to the DOM so it can be used\n * if a freeze event occurs\n *\n * @param _document The global document object (passed in for testing purposes)\n * @returns Element The overlay node is returned\n */\nexport function createOverlay(_document: Document): HTMLElement | undefined {\n  let overlay = _document.createElement('div');\n  overlay.setAttribute('id', 'prebootOverlay');\n  overlay.setAttribute(\n    'style',\n    'display:none;position:absolute;left:0;' +\n    'top:0;width:100%;height:100%;z-index:999999;background:black;opacity:.3'\n  );\n  _document.documentElement.appendChild(overlay);\n\n  return overlay;\n}\n\n/**\n * Get references to the current app root node based on input options. Users can\n * initialize preboot either by specifying appRoot which is just one or more\n * selectors for apps. This section option is useful for people that are doing their own\n * buffering (i.e. they have their own client and server view)\n *\n * @param _document The global document object used to attach the overlay\n * @param opts Options passed in by the user to init()\n * @param serverNode The server node serving as application root\n * @returns ServerClientRoot An array of root info for the current app\n */\nexport function getAppRoot(\n  _document: Document,\n  opts: PrebootOptions,\n  serverNode: HTMLElement\n): ServerClientRoot {\n  const root: ServerClientRoot = {serverNode};\n\n  // if we are doing buffering, we need to create the buffer for the client\n  // else the client root is the same as the server\n  root.clientNode = opts.buffer ? createBuffer(root) : root.serverNode;\n\n  // create an overlay if not disabled ,that can be used later if a freeze event occurs\n  if (!opts.disableOverlay) {\n    root.overlay = createOverlay(_document);\n  }\n\n  return root;\n}\n\n/**\n * Under given server root, for given selector, record events\n *\n * @param _document\n * @param prebootData\n * @param appData\n * @param eventSelector\n */\nexport function handleEvents(_document: Document,\n                             prebootData: PrebootData,\n                             appData: PrebootAppData,\n                             eventSelector: EventSelector) {\n  const serverRoot = appData.root.serverNode;\n\n  // don't do anything if no server root\n  if (!serverRoot) {\n    return;\n  }\n\n  // Attach delegated event listeners for each event selector.\n  // We need to use delegated events as only the top level server node\n  // exists at this point.\n  eventSelector.events.forEach((eventName: string) => {\n    // get the appropriate handler and add it as an event listener\n    const handler = createListenHandler(_document, prebootData, eventSelector, appData);\n    // attach the handler in the capture phase so that it fires even if\n    // one of the handlers below calls stopPropagation()\n    serverRoot.addEventListener(eventName, handler, true);\n\n    // need to keep track of listeners so we can do node.removeEventListener()\n    // when preboot done\n    if (prebootData.listeners) {\n      prebootData.listeners.push({\n        node: serverRoot,\n        eventName,\n        handler\n      });\n    }\n  });\n}\n\n/**\n * Create handler for events that we will record\n */\nexport function createListenHandler(\n  _document: Document,\n  prebootData: PrebootData,\n  eventSelector: EventSelector,\n  appData: PrebootAppData\n): EventListener {\n  const CARET_EVENTS = ['keyup', 'keydown', 'focusin', 'mouseup', 'mousedown'];\n  const CARET_NODES = ['INPUT', 'TEXTAREA'];\n\n  // Support: IE 9-11 only\n  // IE uses a prefixed `matches` version\n  const matches = _document.documentElement.matches ||\n    (_document.documentElement as any).msMatchesSelector;\n  const opts = prebootData.opts;\n\n  return function(event: DomEvent) {\n    const node: Element = event.target;\n\n    // a delegated handlers on document is used so we need to check if\n    // event target matches a desired selector\n    if (!matches.call(node, eventSelector.selector)) {\n      return;\n    }\n\n    const root = appData.root;\n    const eventName = event.type;\n\n    // if no node or no event name, just return\n    if (!node || !eventName) {\n      return;\n    }\n\n    // if key codes set for eventSelector, then don't do anything if event\n    // doesn't include key\n    const keyCodes = eventSelector.keyCodes;\n    if (keyCodes && keyCodes.length) {\n      const matchingKeyCodes = keyCodes.filter(keyCode => event.which === keyCode);\n\n      // if there are not matches (i.e. key entered NOT one of the key codes)\n      // then don't do anything\n      if (!matchingKeyCodes.length) {\n        return;\n      }\n    }\n\n    // if for a given set of events we are preventing default, do that\n    if (eventSelector.preventDefault) {\n      event.preventDefault();\n    }\n\n    // if an action handler passed in, use that\n    if (eventSelector.action) {\n      eventSelector.action(node, event);\n    }\n\n    // get the node key for a given node\n    const nodeKey = getNodeKeyForPreboot({ root: root, node: node });\n\n    // record active node\n    if (CARET_EVENTS.indexOf(eventName) >= 0) {\n      // if it's an caret node, get the selection for the active node\n      const isCaretNode = CARET_NODES.indexOf(node.tagName ? node.tagName : '') >= 0;\n\n      prebootData.activeNode = {\n        root: root,\n        node: node,\n        nodeKey: nodeKey,\n        selection: isCaretNode ? getSelection(node as HTMLInputElement) : undefined\n      };\n    } else if (eventName !== 'change' && eventName !== 'focusout') {\n      prebootData.activeNode = undefined;\n    }\n\n    // if overlay is not disabled and we are freezing the UI\n    if (opts && !opts.disableOverlay && eventSelector.freeze) {\n      const overlay = root.overlay as HTMLElement;\n\n      // show the overlay\n      overlay.style.display = 'block';\n\n      // hide the overlay after 10 seconds just in case preboot.complete() never\n      // called\n      setTimeout(() => {\n        overlay.style.display = 'none';\n      }, 10000);\n    }\n\n    // we will record events for later replay unless explicitly marked as\n    // doNotReplay\n    if (eventSelector.replay) {\n      appData.events.push({\n        node,\n        nodeKey,\n        event,\n        name: eventName\n      });\n    }\n  };\n}\n\n/**\n * Get the selection data that is later used to set the cursor after client view\n * is active\n */\nexport function getSelection(node: HTMLInputElement): PrebootSelection {\n  node = node || {} as HTMLInputElement;\n\n  const nodeValue = node.value || '';\n  const selection: PrebootSelection = {\n    start: nodeValue.length,\n    end: nodeValue.length,\n    direction: 'forward'\n  };\n\n  // if browser support selectionStart on node (Chrome, FireFox, IE9+)\n  try {\n    if (node.selectionStart || node.selectionStart === 0) {\n      selection.start = node.selectionStart;\n      selection.end = node.selectionEnd ? node.selectionEnd : 0;\n      selection.direction = node.selectionDirection ?\n        node.selectionDirection as PrebootSelectionDirection : 'none';\n    }\n  } catch (ex) {}\n\n  return selection;\n}\n\n/**\n * Create buffer for a given node\n *\n * @param root All the data related to a particular app\n * @returns Returns the root client node.\n */\nexport function createBuffer(root: ServerClientRoot): HTMLElement {\n  const serverNode = root.serverNode;\n\n  // if no rootServerNode OR the selector is on the entire html doc or the body\n  // OR no parentNode, don't buffer\n  if (!serverNode || !serverNode.parentNode ||\n    serverNode === document.documentElement || serverNode === document.body) {\n    return serverNode as HTMLElement;\n  }\n\n  // create shallow clone of server root\n  const rootClientNode = serverNode.cloneNode(false) as HTMLElement;\n  // we want the client to write to a hidden div until the time for switching\n  // the buffers\n  rootClientNode.style.display = 'none';\n\n  // insert the client node before the server and return it\n  serverNode.parentNode.insertBefore(rootClientNode, serverNode);\n\n  // mark server node as not to be touched by AngularJS - needed for ngUpgrade\n  serverNode.setAttribute('ng-non-bindable', '');\n\n  // return the rootClientNode\n  return rootClientNode;\n}\n"]}
\No newline at end of file