123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452 |
- import Iterator from 'es6-iterator';
- import BackgroundTimer from 'react-native-background-timer';
- import 'url-polyfill'; // Polyfill for URL constructor
-
- import { Platform } from '../../react';
-
- // XXX The library lib-jitsi-meet utilizes window.localStorage at the time of
- // this writing and, consequently, the browser-related polyfills implemented
- // here by the feature base/lib-jitsi-meet for the purposes of the library
- // lib-jitsi-meet are incomplete without the Web Storage API! Should the library
- // lib-jitsi-meet (and its dependencies) stop utilizing window.localStorage,
- // the following import may be removed:
- import '../../storage';
-
- /**
- * Gets the first common prototype of two specified Objects (treating the
- * objects themselves as prototypes as well).
- *
- * @param {Object} a - The first prototype chain to climb in search of a common
- * prototype.
- * @param {Object} b - The second prototype chain to climb in search of a common
- * prototype.
- * @returns {Object|undefined} - The first common prototype of a and b.
- */
- function _getCommonPrototype(a, b) {
- // Allow the arguments to be prototypes themselves.
- if (a === b) {
- return a;
- }
-
- let p;
-
- if (((p = Object.getPrototypeOf(a)) && (p = _getCommonPrototype(b, p)))
- || ((p = Object.getPrototypeOf(b))
- && (p = _getCommonPrototype(a, p)))) {
- return p;
- }
-
- return undefined;
- }
-
- /**
- * Implements an absolute minimum of the common logic of Document.querySelector
- * and Element.querySelector. Implements the most simple of selectors necessary
- * to satisfy the call sites at the time of this writing i.e. select by tagName.
- *
- * @param {Node} node - The Node which is the root of the tree to query.
- * @param {string} selectors - The group of CSS selectors to match on.
- * @returns {Element} - The first Element which is a descendant of the specified
- * node and matches the specified group of selectors.
- */
- function _querySelector(node, selectors) {
- let element = null;
-
- node && _visitNode(node, n => {
- if (n.nodeType === 1 /* ELEMENT_NODE */
- && n.nodeName === selectors) {
- element = n;
-
- return true;
- }
-
- return false;
- });
-
- return element;
- }
-
- /**
- * Visits each Node in the tree of a specific root Node (using depth-first
- * traversal) and invokes a specific callback until the callback returns true.
- *
- * @param {Node} node - The root Node which represents the tree of Nodes to
- * visit.
- * @param {Function} callback - The callback to invoke with each visited Node.
- * @returns {boolean} - True if the specified callback returned true for a Node
- * (at which point the visiting stopped); otherwise, false.
- */
- function _visitNode(node, callback) {
- if (callback(node)) {
- return true;
- }
-
- /* eslint-disable no-param-reassign, no-extra-parens */
-
- if ((node = node.firstChild)) {
- do {
- if (_visitNode(node, callback)) {
- return true;
- }
- } while ((node = node.nextSibling));
- }
-
- /* eslint-enable no-param-reassign, no-extra-parens */
-
- return false;
- }
-
- (global => {
- const { DOMParser } = require('xmldom');
-
- // DOMParser
- //
- // Required by:
- // - lib-jitsi-meet requires this if using WebSockets
- global.DOMParser = DOMParser;
-
- // addEventListener
- //
- // Required by:
- // - jQuery
- if (typeof global.addEventListener === 'undefined') {
- // eslint-disable-next-line no-empty-function
- global.addEventListener = () => {};
- }
-
- // Array.prototype[@@iterator]
- //
- // Required by:
- // - for...of statement use(s) in lib-jitsi-meet
- const arrayPrototype = Array.prototype;
-
- if (typeof arrayPrototype['@@iterator'] === 'undefined') {
- arrayPrototype['@@iterator'] = function() {
- return new Iterator(this);
- };
- }
-
- // document
- //
- // Required by:
- // - jQuery
- // - lib-jitsi-meet/modules/RTC/adapter.screenshare.js
- // - Strophe
- if (typeof global.document === 'undefined') {
- const document
- = new DOMParser().parseFromString(
- '<html><head></head><body></body></html>',
- 'text/xml');
-
- // document.addEventListener
- //
- // Required by:
- // - jQuery
- if (typeof document.addEventListener === 'undefined') {
- // eslint-disable-next-line no-empty-function
- document.addEventListener = () => {};
- }
-
- // document.cookie
- //
- // Required by:
- // - herment
- if (typeof document.cookie === 'undefined') {
- document.cookie = '';
- }
-
- // document.implementation
- //
- // Required by:
- // - jQuery
- if (typeof document.implementation === 'undefined') {
- document.implementation = {};
- }
-
- // document.implementation.createHTMLDocument
- //
- // Required by:
- // - jQuery
- if (typeof document.implementation.createHTMLDocument === 'undefined') {
- document.implementation.createHTMLDocument = function(title = '') {
- const htmlDocument
- = new DOMParser().parseFromString(
- `<html>
- <head><title>${title}</title></head>
- <body></body>
- </html>`,
- 'text/xml');
-
- Object.defineProperty(htmlDocument, 'body', {
- get() {
- return htmlDocument.getElementsByTagName('body')[0];
- }
- });
-
- return htmlDocument;
- };
- }
-
- // Element.querySelector
- //
- // Required by:
- // - lib-jitsi-meet/modules/xmpp
- const elementPrototype
- = Object.getPrototypeOf(document.documentElement);
-
- if (elementPrototype) {
- if (typeof elementPrototype.querySelector === 'undefined') {
- elementPrototype.querySelector = function(selectors) {
- return _querySelector(this, selectors);
- };
- }
-
- // Element.remove
- //
- // Required by:
- // - lib-jitsi-meet ChatRoom#onPresence parsing
- if (typeof elementPrototype.remove === 'undefined') {
- elementPrototype.remove = function() {
- if (this.parentNode !== null) {
- this.parentNode.removeChild(this);
- }
- };
- }
-
- // Element.innerHTML
- //
- // Required by:
- // - jQuery's .append method
- if (!elementPrototype.hasOwnProperty('innerHTML')) {
- Object.defineProperty(elementPrototype, 'innerHTML', {
- get() {
- return this.childNodes.toString();
- },
-
- set(innerHTML) {
- // MDN says: removes all of element's children, parses
- // the content string and assigns the resulting nodes as
- // children of the element.
-
- // Remove all of element's children.
- this.textContent = '';
-
- // Parse the content string.
- const d
- = new DOMParser().parseFromString(
- `<div>${innerHTML}</div>`,
- 'text/xml');
-
- // Assign the resulting nodes as children of the
- // element.
- const documentElement = d.documentElement;
- let child;
-
- // eslint-disable-next-line no-cond-assign
- while (child = documentElement.firstChild) {
- this.appendChild(child);
- }
- }
- });
- }
-
- // Element.children
- //
- // Required by:
- // - lib-jitsi-meet ChatRoom#onPresence parsing
- if (!elementPrototype.hasOwnProperty('children')) {
- Object.defineProperty(elementPrototype, 'children', {
- get() {
- const nodes = this.childNodes;
- const children = [];
- let i = 0;
- let node = nodes[i];
-
- while (node) {
- if (node.nodeType === 1) {
- children.push(node);
- }
- i += 1;
- node = nodes[i];
- }
-
- return children;
- }
- });
- }
- }
-
- // FIXME There is a weird infinite loop related to console.log and
- // Document and/or Element at the time of this writing. Work around it
- // by patching Node and/or overriding console.log.
- const documentPrototype = Object.getPrototypeOf(document);
- const nodePrototype
- = _getCommonPrototype(documentPrototype, elementPrototype);
-
- if (nodePrototype
-
- // XXX The intention was to find Node from which Document and
- // Element extend. If for whatever reason we've reached Object,
- // then it doesn't sound like what expected.
- && nodePrototype !== Object.getPrototypeOf({})) {
- // Override console.log.
- const { console } = global;
-
- if (console) {
- const loggerLevels = require('jitsi-meet-logger').levels;
-
- Object.keys(loggerLevels).forEach(key => {
- const level = loggerLevels[key];
- const consoleLog = console[level];
-
- /* eslint-disable prefer-rest-params */
-
- if (typeof consoleLog === 'function') {
- console[level] = function(...args) {
- // XXX If console's disableYellowBox is truthy, then
- // react-native will not automatically display the
- // yellow box for the warn level. However, it will
- // still display the red box for the error level.
- // But I disable the yellow box when I don't want to
- // have react-native automatically show me the
- // console's output just like in the Release build
- // configuration. Because I didn't find a way to
- // disable the red box, downgrade the error level to
- // warn. The red box will still be displayed but not
- // for the error level.
- if (console.disableYellowBox && level === 'error') {
- console.warn(...args);
-
- return;
- }
-
- const { length } = args;
-
- for (let i = 0; i < length; ++i) {
- let arg = args[i];
-
- if (arg
- && typeof arg !== 'string'
-
- // Limit the console.log override to
- // Node (instances).
- && nodePrototype.isPrototypeOf(arg)) {
- const toString = arg.toString;
-
- if (toString) {
- arg = toString.call(arg);
- }
- }
- args[i] = arg;
- }
-
- consoleLog.apply(this, args);
- };
- }
-
- /* eslint-enable prefer-rest-params */
- });
- }
- }
-
- global.document = document;
- }
-
- // location
- if (typeof global.location === 'undefined') {
- global.location = {
- href: '',
-
- // Required by:
- // - lib-jitsi-meet/modules/xmpp/xmpp.js
- search: ''
- };
- }
-
- const { navigator } = global;
-
- if (navigator) {
- // platform
- //
- // Required by:
- // - lib-jitsi-meet/modules/RTC/adapter.screenshare.js
- if (typeof navigator.platform === 'undefined') {
- navigator.platform = '';
- }
-
- // plugins
- //
- // Required by:
- // - lib-jitsi-meet/modules/RTC/adapter.screenshare.js
- if (typeof navigator.plugins === 'undefined') {
- navigator.plugins = [];
- }
-
- // userAgent
- //
- // Required by:
- // - lib-jitsi-meet/modules/RTC/adapter.screenshare.js
- // - lib-jitsi-meet/modules/browser/BrowserDetection.js
- let userAgent = navigator.userAgent || '';
-
- // react-native/version
- const { name, version } = require('react-native/package.json');
- let rn = name || 'react-native';
-
- version && (rn += `/${version}`);
- if (userAgent.indexOf(rn) === -1) {
- userAgent = userAgent ? `${rn} ${userAgent}` : rn;
- }
-
- // (OS version)
- const os = `(${Platform.OS} ${Platform.Version})`;
-
- if (userAgent.indexOf(os) === -1) {
- userAgent = userAgent ? `${userAgent} ${os}` : os;
- }
-
- navigator.userAgent = userAgent;
- }
-
- // WebRTC
- require('./polyfills-webrtc');
- require('react-native-callstats/csio-polyfill');
-
- // XMLHttpRequest
- if (global.XMLHttpRequest) {
- const { prototype } = global.XMLHttpRequest;
-
- // XMLHttpRequest.responseXML
- //
- // Required by:
- // - Strophe
- if (prototype && !prototype.hasOwnProperty('responseXML')) {
- Object.defineProperty(prototype, 'responseXML', {
- get() {
- const { responseText } = this;
-
- return (
- responseText
- && new DOMParser().parseFromString(
- responseText,
- 'text/xml'));
- }
- });
- }
- }
-
- // Timers
- //
- // React Native's timers won't run while the app is in the background, this
- // is a known limitation. Replace them with a background-friendly
- // alternative.
- //
- // Required by:
- // - lib-jitsi-meet
- // - Strophe
- global.clearTimeout = BackgroundTimer.clearTimeout.bind(BackgroundTimer);
- global.clearInterval = BackgroundTimer.clearInterval.bind(BackgroundTimer);
- global.setInterval = BackgroundTimer.setInterval.bind(BackgroundTimer);
- global.setTimeout = (fn, ms = 0) => BackgroundTimer.setTimeout(fn, ms);
-
- })(global || window || this); // eslint-disable-line no-invalid-this
|