diff --git a/package-lock.json b/package-lock.json index b87fa304de5687e38c4ad39605ed52736d1e29e7..b30a2953ce38964d40d02a5982511ed55bccf0b7 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1315,20 +1315,6 @@ "@babel/types": "^7.3.0" } }, - "@types/cookie": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.3.3.tgz", - "integrity": "sha512-LKVP3cgXBT9RYj+t+9FDKwS5tdI+rPBXaNSkma7hvqy35lc7mAokC2zsqWJH0LaqIt3B962nuYI77hsJoT1gow==" - }, - "@types/hoist-non-react-statics": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.1.tgz", - "integrity": "sha512-iMIqiko6ooLrTh1joXodJK5X9xeEALT1kM5G3ZLhD3hszxBdIEd5C75U834D9mLcINgD4OyZf5uQXjkuYydWvA==", - "requires": { - "@types/react": "*", - "hoist-non-react-statics": "^3.3.0" - } - }, "@types/istanbul-lib-coverage": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.1.tgz", @@ -1356,30 +1342,11 @@ "resolved": "https://registry.npmjs.org/@types/node/-/node-12.0.3.tgz", "integrity": "sha512-zkOxCS/fA+3SsdA+9Yun0iANxzhQRiNwTvJSr6N95JhuJ/x27z9G2URx1Jpt3zYFfCGUXZGL5UDxt5eyLE7wgw==" }, - "@types/object-assign": { - "version": "4.0.30", - "resolved": "https://registry.npmjs.org/@types/object-assign/-/object-assign-4.0.30.tgz", - "integrity": "sha1-iUk3HVqZ9Dge4PHfCpt6GH4H5lI=" - }, - "@types/prop-types": { - "version": "15.7.1", - "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.1.tgz", - "integrity": "sha512-CFzn9idOEpHrgdw8JsoTkaDDyRWk1jrzIV8djzcgpq0y9tG4B4lFT+Nxh52DVpDXV+n4+NPNv7M1Dj5uMp6XFg==" - }, "@types/q": { "version": "1.5.2", "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.2.tgz", "integrity": "sha512-ce5d3q03Ex0sy4R14722Rmt6MT07Ua+k4FwDfdcToYJcMKNtRVQvJ6JCAPdAmAnbRb6CsX6aYb9m96NGod9uTw==" }, - "@types/react": { - "version": "16.8.19", - "resolved": "https://registry.npmjs.org/@types/react/-/react-16.8.19.tgz", - "integrity": "sha512-QzEzjrd1zFzY9cDlbIiFvdr+YUmefuuRYrPxmkwG0UQv5XF35gFIi7a95m1bNVcFU0VimxSZ5QVGSiBmlggQXQ==", - "requires": { - "@types/prop-types": "*", - "csstype": "^2.2.0" - } - }, "@types/stack-utils": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-1.0.1.tgz", @@ -1667,6 +1634,11 @@ "resolved": "https://registry.npmjs.org/address/-/address-1.0.3.tgz", "integrity": "sha512-z55ocwKBRLryBs394Sm3ushTtBeg6VAeuku7utSoSnsJKvKcnXFIyC6vh27n3rXyxSgkJBBCAvyOn7gSUcTYjg==" }, + "after": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/after/-/after-0.8.2.tgz", + "integrity": "sha1-/ts5T58OAqqXaOcCvaI7UF+ufh8=" + }, "ajv": { "version": "6.10.0", "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.0.tgz", @@ -1819,6 +1791,11 @@ "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=" }, + "arraybuffer.slice": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/arraybuffer.slice/-/arraybuffer.slice-0.0.7.tgz", + "integrity": "sha512-wGUIVQXuehL5TCqQun8OW81jGzAWycqzFF8lFp+GOM5BXLYj3bKNsYC4daB7n6XjCqxQA/qgTJ+8ANR3acjrog==" + }, "arrify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", @@ -2278,6 +2255,11 @@ "resolved": "https://registry.npmjs.org/babylon/-/babylon-6.18.0.tgz", "integrity": "sha512-q/UEjfGJ2Cm3oKV71DJz9d25TPnq5rhBVL2Q4fA5wcC3jcrdn7+SssEybFIxwAvvP+YCsCYNKughoF33GxgycQ==" }, + "backo2": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz", + "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc=" + }, "bail": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/bail/-/bail-1.0.4.tgz", @@ -2343,11 +2325,21 @@ } } }, + "base64-arraybuffer": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz", + "integrity": "sha1-c5JncZI7Whl0etZmqlzUv5xunOg=" + }, "base64-js": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.0.tgz", "integrity": "sha512-ccav/yGvoa80BQDljCxsmmQ3Xvx60/UpBIij5QN21W3wBi/hhIC9OoO+KLpu9IJTS9j4DRVJ3aDDF9cMSoa2lw==" }, + "base64id": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/base64id/-/base64id-1.0.0.tgz", + "integrity": "sha1-R2iMuZu2gE8OBtPnY7HDLlfY5rY=" + }, "batch": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", @@ -2361,6 +2353,14 @@ "tweetnacl": "^0.14.3" } }, + "better-assert": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/better-assert/-/better-assert-1.0.2.tgz", + "integrity": "sha1-QIZrnhueC1W0gYlDEeaPr/rrxSI=", + "requires": { + "callsite": "1.0.0" + } + }, "big.js": { "version": "5.2.2", "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", @@ -2371,6 +2371,11 @@ "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==" }, + "blob": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.5.tgz", + "integrity": "sha512-gaqbzQPqOoamawKg0LGVd7SzLgXS+JH61oWprSLH+P+abTczqJbhTR8CmJ2u9/bUYNmHTGJx/UEmn6doAvvuig==" + }, "bluebird": { "version": "3.5.5", "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.5.tgz", @@ -2678,6 +2683,11 @@ "caller-callsite": "^2.0.0" } }, + "callsite": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz", + "integrity": "sha1-KAOY5dZkvXQDi28JBRU+borxvCA=" + }, "callsites": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-2.0.0.tgz", @@ -3444,11 +3454,21 @@ "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=" }, + "component-bind": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/component-bind/-/component-bind-1.0.0.tgz", + "integrity": "sha1-AMYIq33Nk4l8AAllGx06jh5zu9E=" + }, "component-emitter": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==" }, + "component-inherit": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/component-inherit/-/component-inherit-0.0.3.tgz", + "integrity": "sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM=" + }, "compressible": { "version": "2.0.17", "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.17.tgz", @@ -3927,11 +3947,6 @@ "cssom": "0.3.x" } }, - "csstype": { - "version": "2.6.5", - "resolved": "https://registry.npmjs.org/csstype/-/csstype-2.6.5.tgz", - "integrity": "sha512-JsTaiksRsel5n7XwqPAfB0l3TFKdpjW/kgAELf9vrb5adGA7UCPLajKK5s3nFrcFm3Rkyp/Qkgl73ENc1UY3cA==" - }, "cyclist": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/cyclist/-/cyclist-0.2.2.tgz", @@ -4352,6 +4367,105 @@ "once": "^1.4.0" } }, + "engine.io": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-3.3.2.tgz", + "integrity": "sha512-AsaA9KG7cWPXWHp5FvHdDWY3AMWeZ8x+2pUVLcn71qE5AtAzgGbxuclOytygskw8XGmiQafTmnI9Bix3uihu2w==", + "requires": { + "accepts": "~1.3.4", + "base64id": "1.0.0", + "cookie": "0.3.1", + "debug": "~3.1.0", + "engine.io-parser": "~2.1.0", + "ws": "~6.1.0" + }, + "dependencies": { + "cookie": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", + "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "ws": { + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.1.4.tgz", + "integrity": "sha512-eqZfL+NE/YQc1/ZynhojeV8q+H050oR8AZ2uIev7RU10svA9ZnJUddHcOUZTJLinZ9yEfdA2kSATS2qZK5fhJA==", + "requires": { + "async-limiter": "~1.0.0" + } + } + } + }, + "engine.io-client": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.3.2.tgz", + "integrity": "sha512-y0CPINnhMvPuwtqXfsGuWE8BB66+B6wTtCofQDRecMQPYX3MYUZXFNKDhdrSe3EVjgOu4V3rxdeqN/Tr91IgbQ==", + "requires": { + "component-emitter": "1.2.1", + "component-inherit": "0.0.3", + "debug": "~3.1.0", + "engine.io-parser": "~2.1.1", + "has-cors": "1.1.0", + "indexof": "0.0.1", + "parseqs": "0.0.5", + "parseuri": "0.0.5", + "ws": "~6.1.0", + "xmlhttprequest-ssl": "~1.5.4", + "yeast": "0.1.2" + }, + "dependencies": { + "component-emitter": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "ws": { + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.1.4.tgz", + "integrity": "sha512-eqZfL+NE/YQc1/ZynhojeV8q+H050oR8AZ2uIev7RU10svA9ZnJUddHcOUZTJLinZ9yEfdA2kSATS2qZK5fhJA==", + "requires": { + "async-limiter": "~1.0.0" + } + } + } + }, + "engine.io-parser": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.1.3.tgz", + "integrity": "sha512-6HXPre2O4Houl7c4g7Ic/XzPnHBvaEmN90vtRO9uLmwtRqQmTOw0QMevL1TOfL2Cpu1VzsaTmMotQgMdkzGkVA==", + "requires": { + "after": "0.8.2", + "arraybuffer.slice": "~0.0.7", + "base64-arraybuffer": "0.1.5", + "blob": "0.0.5", + "has-binary2": "~1.0.2" + } + }, "enhanced-resolve": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.1.0.tgz", @@ -5661,6 +5775,26 @@ } } }, + "has-binary2": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-binary2/-/has-binary2-1.0.3.tgz", + "integrity": "sha512-G1LWKhDSvhGeAQ8mPVQlqNcOB2sJdwATtZKl2pDKKHfpf/rYj24lkinxf69blJbnsvtqqNU+L3SL50vzZhXOnw==", + "requires": { + "isarray": "2.0.1" + }, + "dependencies": { + "isarray": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", + "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=" + } + } + }, + "has-cors": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-cors/-/has-cors-1.1.0.tgz", + "integrity": "sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk=" + }, "has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", @@ -8444,6 +8578,11 @@ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" }, + "object-component": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/object-component/-/object-component-0.0.3.tgz", + "integrity": "sha1-8MaapQ78lbhmwYb0AKM3acsvEpE=" + }, "object-copy": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", @@ -8764,6 +8903,22 @@ "resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.0.tgz", "integrity": "sha512-fxNG2sQjHvlVAYmzBZS9YlDp6PTSSDwa98vkD4QgVDDCAo84z5X1t5XyJQ62ImdLXx5NdIIfihey6xpum9/gRQ==" }, + "parseqs": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.5.tgz", + "integrity": "sha1-1SCKNzjkZ2bikbouoXNoSSGouJ0=", + "requires": { + "better-assert": "~1.0.0" + } + }, + "parseuri": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.5.tgz", + "integrity": "sha1-gCBKUNTbt3m/3G6+J3jZDkvOMgo=", + "requires": { + "better-assert": "~1.0.0" + } + }, "parseurl": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", @@ -11167,6 +11322,100 @@ "kind-of": "^3.2.0" } }, + "socket.io": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-2.2.0.tgz", + "integrity": "sha512-wxXrIuZ8AILcn+f1B4ez4hJTPG24iNgxBBDaJfT6MsyOhVYiTXWexGoPkd87ktJG8kQEcL/NBvRi64+9k4Kc0w==", + "requires": { + "debug": "~4.1.0", + "engine.io": "~3.3.1", + "has-binary2": "~1.0.2", + "socket.io-adapter": "~1.1.0", + "socket.io-client": "2.2.0", + "socket.io-parser": "~3.3.0" + } + }, + "socket.io-adapter": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-1.1.1.tgz", + "integrity": "sha1-KoBeihTWNyEk3ZFZrUUC+MsH8Gs=" + }, + "socket.io-client": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-2.2.0.tgz", + "integrity": "sha512-56ZrkTDbdTLmBIyfFYesgOxsjcLnwAKoN4CiPyTVkMQj3zTUh0QAx3GbvIvLpFEOvQWu92yyWICxB0u7wkVbYA==", + "requires": { + "backo2": "1.0.2", + "base64-arraybuffer": "0.1.5", + "component-bind": "1.0.0", + "component-emitter": "1.2.1", + "debug": "~3.1.0", + "engine.io-client": "~3.3.1", + "has-binary2": "~1.0.2", + "has-cors": "1.1.0", + "indexof": "0.0.1", + "object-component": "0.0.3", + "parseqs": "0.0.5", + "parseuri": "0.0.5", + "socket.io-parser": "~3.3.0", + "to-array": "0.1.4" + }, + "dependencies": { + "component-emitter": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "socket.io-parser": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.3.0.tgz", + "integrity": "sha512-hczmV6bDgdaEbVqhAeVMM/jfUfzuEZHsQg6eOmLgJht6G3mPKMxYm75w2+qhAQZ+4X+1+ATZ+QFKeOZD5riHng==", + "requires": { + "component-emitter": "1.2.1", + "debug": "~3.1.0", + "isarray": "2.0.1" + }, + "dependencies": { + "component-emitter": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "isarray": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", + "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=" + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, "sockjs": { "version": "0.3.19", "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.19.tgz", @@ -11717,6 +11966,11 @@ "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.4.tgz", "integrity": "sha1-I2QN17QtAEM5ERQIIOXPRA5SHdE=" }, + "to-array": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/to-array/-/to-array-0.1.4.tgz", + "integrity": "sha1-F+bBH3PdTz10zaek/zI46a2b+JA=" + }, "to-arraybuffer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", @@ -11981,24 +12235,6 @@ "resolved": "https://registry.npmjs.org/unist-util-stringify-position/-/unist-util-stringify-position-1.1.2.tgz", "integrity": "sha512-pNCVrk64LZv1kElr0N1wPiHEUoXNVFERp+mlTg/s9R5Lwg87f9bM/3sQB99w+N9D/qnM9ar3+AKDBwo/gm/iQQ==" }, - "universal-cookie": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/universal-cookie/-/universal-cookie-4.0.0.tgz", - "integrity": "sha512-6JVx+3oGPjslGqFhQ8YSIBHmYTx8HbyAEH++2/b6SKNXsbsdQ7lU7wRG2bYcRB5JVCz8GYgQ+Ixew91hn3Dy9w==", - "requires": { - "@types/cookie": "^0.3.1", - "@types/object-assign": "^4.0.30", - "cookie": "^0.3.1", - "object-assign": "^4.1.0" - }, - "dependencies": { - "cookie": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", - "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" - } - } - }, "universalify": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", @@ -12807,6 +13043,11 @@ "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-1.3.1.tgz", "integrity": "sha512-tGkGJkN8XqCod7OT+EvGYK5Z4SfDQGD30zAa58OcnAa0RRWgzUEK72tkXhsX1FZd+rgnhRxFtmO+ihkp8LHSkw==" }, + "xmlhttprequest-ssl": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.5.tgz", + "integrity": "sha1-wodrBhaKrcQOV9l+gRkayPQ5iz4=" + }, "xregexp": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/xregexp/-/xregexp-4.0.0.tgz", @@ -12861,6 +13102,11 @@ "camelcase": "^5.0.0", "decamelize": "^1.2.0" } + }, + "yeast": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/yeast/-/yeast-0.1.2.tgz", + "integrity": "sha1-AI4G2AlDIMNy28L47XagymyKxBk=" } } } diff --git a/package.json b/package.json index 88fddfbef5a2435db648728f1b4e4d9e4d127730..f1ae9df1c88c4b64dcbc03f7f700cb9a6e613233 100644 --- a/package.json +++ b/package.json @@ -13,7 +13,9 @@ "react-leaflet": "^2.3.0", "react-leaflet-draw": "0.19.0", "react-scripts": "3.0.1", - "universal-cookie": "^4.0.0" + "universal-cookie": "^4.0.0", + "socket.io": "^2.2.0", + "socket.io-client": "^2.2.0" }, "scripts": { "start": "react-scripts start", diff --git a/src/App.js b/src/App.js index b2d8e87008ca309f10a6c6f85a8211ea6b313321..013fe4216ec2016aaee09be3ca6be530c5c1b3a9 100644 --- a/src/App.js +++ b/src/App.js @@ -3,6 +3,7 @@ import "../node_modules/leaflet-draw/dist/leaflet.draw.css"; import "./App.css"; import UserMap from "./components/UserMap"; import Header from "./components/Header"; +import ClientSocket from "./components/Socket"; class App extends Component { constructor() { @@ -14,11 +15,9 @@ class App extends Component { lng: 25.7597186, zoom: 13, mapUrl: "https://tiles.kartat.kapsi.fi/taustakartta/{z}/{x}/{y}.jpg", - currentGameId: null + currentGameId: null, + socketSignal: null }; - - this.handleLayerChange = this.handleLayerChange.bind(this); - this.handleGameChange = this.handleGameChange.bind(this); } // Toggles through the list and changes the mapUrl state handleLayerChange = () => { @@ -42,6 +41,22 @@ class App extends Component { }); }; + // setting the socket signal automatically fires shouldComponentUpdate function where socketSignal prop is present + // setting socketSignal to null immediately after to avoid multiple database fetches + getSocketSignal = type => { + console.log(type); + this.setState( + { + socketSignal: type + }, + () => { + this.setState({ + socketSignal: null + }); + } + ); + }; + render() { const initialPosition = [this.state.lat, this.state.lng]; return ( @@ -51,12 +66,18 @@ class App extends Component { zoom={this.state.zoom} mapUrl={this.state.mapUrl} currentGameId={this.state.currentGameId} + socketSignal={this.state.socketSignal} /> - , <Header handleLayerChange={this.handleLayerChange} handleGameChange={this.handleGameChange} /> + {this.state.currentGameId && ( + <ClientSocket + gameId={this.state.currentGameId} + getSocketSignal={this.getSocketSignal} + /> + )} </div> ); } diff --git a/src/components/DrawTools.js b/src/components/DrawTools.js index 3de6eddbc6bd7478d33e7b8c49daeb2885646aca..be1b7b35dedab72665c5c3221072c5781656b027 100644 --- a/src/components/DrawTools.js +++ b/src/components/DrawTools.js @@ -152,6 +152,7 @@ class DrawTools extends Component { } // end if (e.layerType === "textbox") this.makeGeoJSON(e.layerType, e.layer); + e.layer.remove(); }; // end _onCreated // turn layer to GeoJSON data @@ -159,7 +160,8 @@ class DrawTools extends Component { // setting the format in which the data will be sent let geoJSON = { data: layer.toGeoJSON(), - mapDrawingId: layer.options.id + mapDrawingId: layer.options.id, + drawingIsActive: true }; // setting properties @@ -179,7 +181,7 @@ class DrawTools extends Component { geoJSON.data.properties.color = layer.options.color; // send item to database, and receive an ID for the layer - this.props.sendGeoJSON(geoJSON, false); + this.props.sendGeoJSON(geoJSON); }; _onEditDeleteStart = () => { @@ -209,6 +211,7 @@ class DrawTools extends Component { } else { this.makeGeoJSON(null, layer); } + return true; }); }; @@ -223,14 +226,16 @@ class DrawTools extends Component { idsToDelete.map(layer => { let geoJSON = { data: layer.toGeoJSON(), - mapDrawingId: layer.options.id + mapDrawingId: layer.options.id, + drawingIsActive: false }; - this.props.sendGeoJSON(geoJSON, true); + this.props.sendGeoJSON(geoJSON); + return true; }); }; - shouldComponentUpdate() { + shouldComponentUpdate(nextProps, nextState) { // disable re-rendering when edit mode is active return !this.state.editModeActive; } @@ -326,9 +331,9 @@ class DrawTools extends Component { className="editable" interactive={true} > - <div class="editable"> + <div className="editable"> <div - contenteditable="true" + contentEditable="true" placeholder="Click out to save" > {text} @@ -388,6 +393,7 @@ class DrawTools extends Component { /> ); } + return null; })} </FeatureGroup> ); diff --git a/src/components/GameList.js b/src/components/GameList.js index aa8d330e37d218f20bba5598deb772e862abd96f..9a56d3626985c58608d92bbcde2b28720829a536 100644 --- a/src/components/GameList.js +++ b/src/components/GameList.js @@ -36,7 +36,7 @@ class GameList extends React.Component { ? games[0].id : undefined }); - // taking the initialized gameID to UserMap.js (GameList.js -> Header.js -> App.js -> UserMap.js) + // taking the initialized gameID to App.js (GameList.js -> GameSidebar.js -> Header.js -> App.js) this.props.handleGameChange(games[0].id); }) .catch(error => { @@ -45,15 +45,11 @@ class GameList extends React.Component { } handleChange = e => { - this.setState( - { - selectedGame: e.target.value - }, - () => { - // taking the changed gameID to UserMap.js (GameList.js -> Header.js -> App.js -> UserMap.js) - //this.props.handleGameChange(this.state.selectedGame); - } - ); + this.setState({ + selectedGame: e.target.value + }); + // taking the changed gameID to App.js (GameList.js -> GameSidebar.js -> Header.js -> App.js) + this.props.handleGameChange(e.target.value); }; handleEditClick = e => { diff --git a/src/components/GameSidebar.js b/src/components/GameSidebar.js index 0511cdc3e9762a1e027a7f180ff8112289e5d311..5d4bad34caef03d5895ceb3d62167a553d6676b4 100644 --- a/src/components/GameSidebar.js +++ b/src/components/GameSidebar.js @@ -16,7 +16,7 @@ export default class GameSidebar extends React.Component { render() { return ( <div className="game-sidebar"> - <GameList /> + <GameList handleGameChange={this.props.handleGameChange} /> {this.props.loggedIn && ( <button id="newGameButton" onClick={() => this.toggleView("newgame")}> New Game diff --git a/src/components/Header.js b/src/components/Header.js index d83e7aad5577a9a9a8537e8b11a2c8a4a93b929c..1faedbbcd01873cc0179bfe91c29e38d3ab50b0f 100644 --- a/src/components/Header.js +++ b/src/components/Header.js @@ -100,7 +100,10 @@ class Header extends React.Component { Tools </button> {this.state.sidebar && ( - <GameSidebar loggedIn={this.state.username ? true : false} /> + <GameSidebar + loggedIn={this.state.username ? true : false} + handleGameChange={this.props.handleGameChange} + /> )} </div> {this.state.form === "register" && ( diff --git a/src/components/Player.js b/src/components/Player.js index 195cc40acbdc959c08b58da9b290bf44577f9e9f..078614209e63df32a8b4bc13340cfd7fa8a65c8f 100644 --- a/src/components/Player.js +++ b/src/components/Player.js @@ -5,11 +5,12 @@ class Player extends Component { constructor(props) { super(props); this.state = { - players: null + players: null, + timer: null }; } - getPlayers() { + getPlayers = () => { fetch( `${process.env.REACT_APP_API_URL}/tracking/players/${ this.props.currentGameId @@ -26,45 +27,51 @@ class Player extends Component { // don't do anything if data is not an array, as it breaks the map function at render if (Array.isArray(data)) { this.setState({ - players: data + players: data, + timer: null // state for updating player positions every minute }); } }) .catch(error => { console.log(error); }); - } - - componentDidMount() { - // update components every 10 seconds - this.interval = setInterval(() => this.setState(this.getPlayers()), 5000); - } + }; shouldComponentUpdate(nextProps, nextState) { // do not update component until players have been fetched and game ID is available - if (this.state.players === null) { - this.getPlayers(); + if (this.props.currentGameId === null) { return false; - } else if (this.props.currentGameId === null) { + } + /* + if (this.props.socketSignal !== "tracking-update") { + return false; + } + */ + return true; + /* + if (nextProps.currentGameId === null) { + return false; + } else if (this.state.players === null) { + this.getPlayers(); return false; } else { return true; } + */ } - componentDidUpdate() { - // check if game ID has changed - if (this.state.currentGameId !== this.props.currentGameId) { - this.setState({ - currentGameId: this.props.currentGameId - }); + componentDidUpdate(prevProps, prevState) { + if (prevProps.socketSignal === "tracking-update") { + // start updating interval + if (prevState.timer === null) { + this.getPlayers(); + this.setState({ + timer: setInterval(this.getPlayers, 60000) + }); + } } } - componentWillUnmount() { - clearInterval(this.interval); - } - render() { return ( <div> diff --git a/src/components/Socket.js b/src/components/Socket.js new file mode 100644 index 0000000000000000000000000000000000000000..06cc2fd899f90ce897e5bbecdd598410aca02cb1 --- /dev/null +++ b/src/components/Socket.js @@ -0,0 +1,62 @@ +import React from "react"; +import io from "socket.io-client"; + +const socketUrl = process.env.REACT_APP_API_URL; + +export default class ClientSocket extends React.Component { + constructor(props) { + super(props); + + this.state = { + // stores the socket object for notifications + sock: null, + // stores updates sent by socket + update: null + }; + } + + // initiate the socket on component mount + async componentWillMount() { + console.log("hi socket"); + // need to explicitly update drawings and trackings when gameID first becomes available + if (this.props.gameId !== null) { + await this.props.getSocketSignal("drawing-update"); + await this.props.getSocketSignal("tracking-update"); + } + this.initSocket(); + } + + shouldComponentUpdate(nextProps, nextState) { + // re-initialize socket when gameId has changed + if (nextProps.gameId !== this.props.gameId) { + this.initSocket(); + return true; + } else { + return false; + } + } + + // disconnect the socket on component dismount + componentWillUnmount() { + console.log("bye socket"); + this.state.sock.disconnect(); + } + + initSocket = () => { + const socket = io(socketUrl); + + // set the socket to listen gameId-thread + socket.on(this.props.gameId, data => { + console.log(data); + this.props.getSocketSignal(data.type); + // check socket update type + this.setState({ update: data.type }); + }); + + this.setState({ sock: socket }); + }; + + render() { + return this.state.update; + } +} diff --git a/src/components/UserMap.js b/src/components/UserMap.js index d1e3f86927e264a3e337574b29597eaa0941b744..c5347fa78df07b4092296702fae1e70ee1704f8f 100644 --- a/src/components/UserMap.js +++ b/src/components/UserMap.js @@ -19,6 +19,7 @@ class UserMap extends Component { }; this.sendGeoJSON = this.sendGeoJSON.bind(this); + this.fetchGeoJSON = this.fetchGeoJSON.bind(this); this.setCurrentPosition = this.setCurrentPosition.bind(this); this.watchPositionId = null; } @@ -29,64 +30,41 @@ class UserMap extends Component { }); } - componentDidUpdate() { - // check if game ID has changed and fetch that game's drawings - if (this.state.currentGameId !== this.props.currentGameId) { - this.setState({ - currentGameId: this.props.currentGameId - }); - this.fetchGeoJSON(); + shouldComponentUpdate(nextProps, nextState) { + if (nextProps.currentGameId === null) { + return false; } + + return true; } - // Sends the players drawings to the backend (and database) - sendGeoJSON(layerToDatabase, isDeleted) { - // isDeleted is used to determine the drawing's drawingIsActive status - // otherwise the fetch functions are the same in both if and else. any smarter way to do this? - if (isDeleted === true) { - fetch( - `${process.env.REACT_APP_API_URL}/draw/mapdrawing/${ - this.props.currentGameId - }`, - { - method: "PUT", - headers: { - Authorization: "Bearer " + sessionStorage.getItem("token"), - Accept: "application/json", - "Content-Type": "application/json" - }, - body: JSON.stringify({ - type: "FeatureCollection", - drawingIsActive: false, - mapDrawingId: layerToDatabase.mapDrawingId, - data: layerToDatabase.data - }) - } - ); - } else { - fetch( - `${process.env.REACT_APP_API_URL}/draw/mapdrawing/${ - this.props.currentGameId - }`, - { - method: "PUT", - headers: { - Authorization: "Bearer " + sessionStorage.getItem("token"), - Accept: "application/json", - "Content-Type": "application/json" - }, - body: JSON.stringify({ - type: "FeatureCollection", - drawingIsActive: true, - mapDrawingId: layerToDatabase.mapDrawingId, - data: layerToDatabase.data - }) - } - ); + componentDidUpdate(prevProps, prevState) { + if (prevProps.socketSignal === "drawing-update") { + this.fetchGeoJSON(); } + } - // get the layers again to stop updating with old objects - this.fetchGeoJSON(); + // Sends the players drawings to the backend (and database) + sendGeoJSON(layerToDatabase) { + fetch( + `${process.env.REACT_APP_API_URL}/draw/mapdrawing/${ + this.props.currentGameId + }`, + { + method: "PUT", + headers: { + Authorization: "Bearer " + sessionStorage.getItem("token"), + Accept: "application/json", + "Content-Type": "application/json" + }, + body: JSON.stringify({ + type: "FeatureCollection", + drawingIsActive: layerToDatabase.drawingIsActive, + mapDrawingId: layerToDatabase.mapDrawingId, + data: layerToDatabase.data + }) + } + ); } // Get the drawings from the backend and add them to the state, so they can be drawn @@ -107,6 +85,7 @@ class UserMap extends Component { let newFeatures = []; data.map(item => { newFeatures.push(item); + return true; }); this.setState({ @@ -197,7 +176,10 @@ class UserMap extends Component { </Popup> </Marker> )} - <Player currentGameId={this.state.currentGameId} /> + <Player + currentGameId={this.props.currentGameId} + socketSignal={this.props.socketSignal} + /> </Map> ); }