diff --git a/package-lock.json b/package-lock.json index 447c01e..6075b0c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -20,6 +20,8 @@ "file-saver": "^2.0.5", "i18next": "^23.7.6", "i18next-browser-languagedetector": "^7.2.0", + "json-edit-react": "^1.15.4", + "jszip": "^3.10.1", "papaparse": "^5.4.1", "react": "^18.2.0", "react-azure-maps": "^1.0.0", @@ -11028,8 +11030,7 @@ "node_modules/core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==", - "dev": true + "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==" }, "node_modules/cosmiconfig": { "version": "7.1.0", @@ -15438,6 +15439,11 @@ "node": ">= 4" } }, + "node_modules/immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ==" + }, "node_modules/immer": { "version": "9.0.18", "resolved": "https://registry.npmjs.org/immer/-/immer-9.0.18.tgz", @@ -15519,8 +15525,7 @@ "node_modules/inherits": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, "node_modules/ini": { "version": "2.0.0", @@ -20379,6 +20384,18 @@ "node": ">=4" } }, + "node_modules/json-edit-react": { + "version": "1.15.4", + "resolved": "https://registry.npmjs.org/json-edit-react/-/json-edit-react-1.15.4.tgz", + "integrity": "sha512-DxuIcBJ0FsuNrYzODDMc/C5TNP3sGpLR1Iq6UFtbYTKWJhOAt6BVncQdNZlwJ7lYO0HY0pjNyAUiWQCX3Sbayw==", + "dependencies": { + "object-property-assigner": "^1.3.0", + "object-property-extractor": "^1.0.11" + }, + "peerDependencies": { + "react": ">=16.0.0" + } + }, "node_modules/json-parse-even-better-errors": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", @@ -20474,6 +20491,49 @@ "node": ">=4.0" } }, + "node_modules/jszip": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.10.1.tgz", + "integrity": "sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==", + "dependencies": { + "lie": "~3.3.0", + "pako": "~1.0.2", + "readable-stream": "~2.3.6", + "setimmediate": "^1.0.5" + } + }, + "node_modules/jszip/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, + "node_modules/jszip/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/jszip/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/jszip/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, "node_modules/keyborg": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/keyborg/-/keyborg-2.2.0.tgz", @@ -20552,6 +20612,14 @@ "node": ">= 0.8.0" } }, + "node_modules/lie": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", + "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", + "dependencies": { + "immediate": "~3.0.5" + } + }, "node_modules/lilconfig": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.0.6.tgz", @@ -21596,6 +21664,16 @@ "node": ">= 0.4" } }, + "node_modules/object-property-assigner": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object-property-assigner/-/object-property-assigner-1.3.0.tgz", + "integrity": "sha512-19A0RsC9rP9klCKHDPL/MeERxeopV9wyMNfP+eD2uKOafzLjF+OUEN4FoP6RAlCFHmerBPJ4ohNv/WrgaNpeIA==" + }, + "node_modules/object-property-extractor": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/object-property-extractor/-/object-property-extractor-1.0.11.tgz", + "integrity": "sha512-VnDQcyN0FTXZ0hMZS/CTb2QkIssZ9XKB8zlf5rnFh1HjFQX1P73EHawavSztBOiPDGqAPNXebv4agjhF9eACAw==" + }, "node_modules/object.assign": { "version": "4.1.4", "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", @@ -21953,6 +22031,11 @@ "node": ">=6" } }, + "node_modules/pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" + }, "node_modules/papaparse": { "version": "5.4.1", "resolved": "https://registry.npmjs.org/papaparse/-/papaparse-5.4.1.tgz", @@ -23973,8 +24056,7 @@ "node_modules/process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", - "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", - "dev": true + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" }, "node_modules/proj4": { "version": "2.8.1", @@ -28602,8 +28684,7 @@ "node_modules/util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "dev": true + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" }, "node_modules/util.promisify": { "version": "1.0.1", @@ -38057,8 +38138,7 @@ "core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==", - "dev": true + "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==" }, "cosmiconfig": { "version": "7.1.0", @@ -41358,6 +41438,11 @@ "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", "dev": true }, + "immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ==" + }, "immer": { "version": "9.0.18", "resolved": "https://registry.npmjs.org/immer/-/immer-9.0.18.tgz", @@ -41414,8 +41499,7 @@ "inherits": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, "ini": { "version": "2.0.0", @@ -45188,6 +45272,15 @@ "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", "dev": true }, + "json-edit-react": { + "version": "1.15.4", + "resolved": "https://registry.npmjs.org/json-edit-react/-/json-edit-react-1.15.4.tgz", + "integrity": "sha512-DxuIcBJ0FsuNrYzODDMc/C5TNP3sGpLR1Iq6UFtbYTKWJhOAt6BVncQdNZlwJ7lYO0HY0pjNyAUiWQCX3Sbayw==", + "requires": { + "object-property-assigner": "^1.3.0", + "object-property-extractor": "^1.0.11" + } + }, "json-parse-even-better-errors": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", @@ -45266,6 +45359,51 @@ "object.assign": "^4.1.3" } }, + "jszip": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.10.1.tgz", + "integrity": "sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==", + "requires": { + "lie": "~3.3.0", + "pako": "~1.0.2", + "readable-stream": "~2.3.6", + "setimmediate": "^1.0.5" + }, + "dependencies": { + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, + "readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "requires": { + "safe-buffer": "~5.1.0" + } + } + } + }, "keyborg": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/keyborg/-/keyborg-2.2.0.tgz", @@ -45326,6 +45464,14 @@ "type-check": "~0.4.0" } }, + "lie": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", + "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", + "requires": { + "immediate": "~3.0.5" + } + }, "lilconfig": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.0.6.tgz", @@ -46090,6 +46236,16 @@ "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==" }, + "object-property-assigner": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object-property-assigner/-/object-property-assigner-1.3.0.tgz", + "integrity": "sha512-19A0RsC9rP9klCKHDPL/MeERxeopV9wyMNfP+eD2uKOafzLjF+OUEN4FoP6RAlCFHmerBPJ4ohNv/WrgaNpeIA==" + }, + "object-property-extractor": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/object-property-extractor/-/object-property-extractor-1.0.11.tgz", + "integrity": "sha512-VnDQcyN0FTXZ0hMZS/CTb2QkIssZ9XKB8zlf5rnFh1HjFQX1P73EHawavSztBOiPDGqAPNXebv4agjhF9eACAw==" + }, "object.assign": { "version": "4.1.4", "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", @@ -46344,6 +46500,11 @@ "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", "dev": true }, + "pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" + }, "papaparse": { "version": "5.4.1", "resolved": "https://registry.npmjs.org/papaparse/-/papaparse-5.4.1.tgz", @@ -47649,8 +47810,7 @@ "process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", - "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", - "dev": true + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" }, "proj4": { "version": "2.8.1", @@ -51165,8 +51325,7 @@ "util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "dev": true + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" }, "util.promisify": { "version": "1.0.1", diff --git a/package.json b/package.json index 9f0fd49..c2f3130 100644 --- a/package.json +++ b/package.json @@ -15,6 +15,8 @@ "file-saver": "^2.0.5", "i18next": "^23.7.6", "i18next-browser-languagedetector": "^7.2.0", + "json-edit-react": "^1.15.4", + "jszip": "^3.10.1", "papaparse": "^5.4.1", "react": "^18.2.0", "react-azure-maps": "^1.0.0", diff --git a/src/pages/conversion/download-imdf.js b/src/pages/conversion/download-imdf.js index 7a3b863..ba991ab 100644 --- a/src/pages/conversion/download-imdf.js +++ b/src/pages/conversion/download-imdf.js @@ -1,11 +1,44 @@ -import { cx } from '@emotion/css'; -import { PrimaryButton } from '@fluentui/react'; -import { failedLogsButton, logsButton } from './style'; +import { cx } from '@emotion/css'; +import { PrimaryButton } from '@fluentui/react'; +import { logsButton } from './style'; +import React, { useState } from 'react'; +import JSZip from 'jszip'; +import { saveAs } from 'file-saver'; + +// This component downloads the updated IMDF zip file and is triggered by users clicking "Download IMDF" button +export const DownloadIMDF = ({ imdfPackageLocation, units, levels, footprint, building }) => { + const [isLoading, setIsLoading] = useState(false); + + // Fetches current zip file, updates the geojson files and downloads the updated zip + const handleUpdateZip = () => { + setIsLoading(true); + fetch(imdfPackageLocation) + .then(response => response.arrayBuffer()) + .then(data => { + const zip = new JSZip(); + return zip.loadAsync(data); + }) + .then(zip => { + zip.file('unit.geojson', JSON.stringify(units, null, 2)); + zip.file('level.geojson', JSON.stringify(levels, null, 2)); + zip.file('footprint.geojson', JSON.stringify(footprint, null, 2)); + zip.file('building.geojson', JSON.stringify(building, null, 2)); + return zip.generateAsync({ type: 'blob' }); + }) + .then(updatedZip => { + saveAs(updatedZip, 'updated_imdf_package.zip'); + setIsLoading(false); + }) + .catch(() => { + setIsLoading(false); + }); + }; -export const DownloadIMDF = ({ isFailed, link }) => { return ( - - Download IMDF - +
+ + {isLoading ? 'Downloading..' : 'Download IMDF'} + +
); }; diff --git a/src/pages/conversion/imdf-conversion.js b/src/pages/conversion/imdf-conversion.js index db513ed..6367e43 100644 --- a/src/pages/conversion/imdf-conversion.js +++ b/src/pages/conversion/imdf-conversion.js @@ -1,4 +1,5 @@ import { cx } from '@emotion/css'; +import { useEffect, useState } from 'react'; import { MessageBar, MessageBarType } from '@fluentui/react'; import { Icon } from '@fluentui/react/lib/Icon'; import { PATHS } from 'common'; @@ -13,9 +14,34 @@ import { actionButtonsLeft, actionButtonsWrapper, messageWrapper } from './imdf- import { ImdfDiagnostics } from './imdf-diagnostics'; import PlacesPreviewMap from './places-preview-map'; import StepButton from './step-button'; +import { processZip } from './places-preview-map/utils'; import { container, content, enabledStep, step as stepStyle, stepTitle, stepsContainer } from './style'; const ImdfConversion = () => { + const [units, setUnits] = useState({ features: [] }); + const [levels, setLevels] = useState({ features: [] }); + const [footprint, setFootprint] = useState({ features: [] }); + const [building, setBuilding] = useState({ features: [] }); + + const handleUnitsChange = (editedUnits) => { + if(!editedUnits.type) { + editedUnits.type = 'FeatureCollection'; + } + setUnits(editedUnits); + }; + + const handleLevelsChange = (editedLevels) => { + setLevels(editedLevels); + }; + + const handleFootprintChange = (editedFootprint) => { + setFootprint(editedFootprint); + }; + + const handleBuildingChange = (editedBuilding) => { + setBuilding(editedBuilding); + }; + const { t } = useTranslation(); const navigate = useCustomNavigate(); @@ -28,7 +54,25 @@ const ImdfConversion = () => { s.imdfPackageLocation, s.diagnosticPackageLocation, ]); + + useEffect(() => { + if (!imdfPackageLocation) return; + + processZip(imdfPackageLocation).then(files => { + const unitFile = files.find(file => file.filename === 'unit.geojson'); + const levelFile = files.find(file => file.filename === 'level.geojson'); + const buildingFile = files.find(file => file.filename === 'building.geojson'); + const footprintFile = files.find(file => file.filename === 'footprint.geojson'); + if (unitFile && levelFile && footprintFile) { + setUnits(unitFile.content); + setLevels(levelFile.content); + setBuilding(buildingFile.content); + setFootprint(footprintFile.content); + } + }); + }, [imdfPackageLocation]); + const { isRunningIMDFConversion, hasCompletedIMDFConversion, imdfConversionStatus, errorList } = useIMDFConversionStatus(); @@ -76,7 +120,7 @@ const ImdfConversion = () => {
{imdfConversionStatus === conversionStatuses.finishedSuccessfully && ( - + )}
@@ -84,7 +128,7 @@ const ImdfConversion = () => {
{imdfConversionStatus === conversionStatuses.finishedSuccessfully && ( - {({ height }) => } + {({ height }) => } )} diff --git a/src/pages/conversion/places-preview-map/imdf-model-helpers/index.js b/src/pages/conversion/places-preview-map/imdf-model-helpers/index.js index 3c3a29b..ef5b068 100644 --- a/src/pages/conversion/places-preview-map/imdf-model-helpers/index.js +++ b/src/pages/conversion/places-preview-map/imdf-model-helpers/index.js @@ -8,12 +8,12 @@ function drawingModeChanged(allLayers) { } // Displays information of a feature that is being drawn/edited, at time of click -function currentEditData(map, drawingManager) { +function currentEditData(map, drawingManager, setJsonData) { map.events.add('drawingstarted', drawingManager, (f) => { if(f?.data) { - var shapeId = f.data.id; - var geojsonData = drawingManager.getSource().getShapeById(shapeId).data; - document.getElementById('infoPanel-json').value = JSON.stringify(geojsonData, null, 2); + var shapeId = f.data.id; + var geojsonData = drawingManager.getSource().getShapeById(shapeId).data; + setJsonData(geojsonData); } }); } @@ -35,26 +35,119 @@ function groupAndSort(units, language, selectedLevel) { return groupedFeatures; } -function writeToInfoPanel(geojsonData) { - const { map, ...obj } = geojsonData; - document.getElementById('infoPanel-json').value = JSON.stringify(obj, null, 2); +// Changes the cursor to be a pointer when a clickable feature is hovered over +function grabToPointer(layerName, hoverLayer, map) { + map.events.add('mousemove', layerName, function (e) { + hoverLayer.setOptions({ filter: ['==', ['get', '_azureMapsShapeId'], e.shapes[0].getProperties()['_azureMapsShapeId']] }); + map.getCanvas().style.cursor = 'pointer'; + }); + + map.events.add('mouseleave', layerName, function (e) { + hoverLayer.setOptions({ filter: ['==', ['get', '_azureMapsShapeId'], ''] }); + map.getCanvas().style.cursor = 'grab'; + }); + + map.events.add(['mousedown', 'mouseup'], layerName, function (e) { + map.getCanvas().style.cursor = 'pointer'; + }); } -// Changes the cursor to be a pointer when a clickable feature is hovered over -function grabToPointer(layerName, map) { - map.events.add('mouseover', layerName, function () { - map.getCanvasContainer().style.cursor = 'pointer'; +// Changes cursor between grabbing and grabbing when map is clicked and dragged +function grabAndGrabbing(map) { + map.events.add('mousedown', () => { + map.getCanvas().style.cursor = 'grabbing'; }); - map.events.add('mouseout', layerName, function () { - map.getCanvasContainer().style.cursor = 'grab'; + map.events.add('mouseup', () => { + map.getCanvas().style.cursor = 'grab'; }); } +function updateSelectedColor(map, unitSelected) { + if(unitSelected) { + const lineLayer = map.layers.getLayerById('lineClickLayer'); + try { + lineLayer.setOptions({strokeColor: 'hsla(0, 0%, 0%, 0)'}); + } catch { + console.log('No line layer to change color of.') + } + const unitLayer = map.layers.getLayerById('unitClickChange'); + try { + unitLayer.setOptions({fillColor: 'rgba(75, 146, 210, 0.8)'}); + } catch { + console.log('No unit layer to change color of.') + } + } else { + const polygonLayer = map.layers.getLayerById('lineClickLayer'); + try { + polygonLayer.setOptions({strokeColor: 'rgba(75, 146, 210, 0.8)'}); + } catch { + console.log('No unit layer to change color of.') + } + const unitLayer = map.layers.getLayerById('unitClickChange'); + try { + unitLayer.setOptions({fillColor: 'hsla(0, 0%, 0%, 0)'}); + } catch { + console.log('No unit layer to change color of.') + } + } +} + +function updateLevels(levels, selectedLevel, newValue) { + levels.features = levels.features.map(item => + item.id === selectedLevel.id ? newValue : item + ); + + return levels; +} + +function setFields(feature, selectedLevel) { + if (!feature.data.properties.name) { + feature.data.properties.name = {}; + feature.data.properties.name.en = ''; + } + + if(!feature.data.properties.category) { + feature.data.properties.category = 'unspecified'; + } + + if(!feature.data.properties.level_id) { + feature.data.properties.level_id = selectedLevel.id; + } + + if(!feature.data.properties.display_point) { + feature.data.properties.display_point = {}; + feature.data.properties.display_point.type = 'Point'; + feature.data.properties.display_point.coordinates = []; + } + + if(!feature.data.properties.label) { + feature.data.properties.label = ''; + } + + if(feature.data.properties._azureMapsShapeId) { + let savedAzureId = feature.data.properties._azureMapsShapeId; + delete feature.data.properties._azureMapsShapeId; + feature.data.properties._azureMapsShapeId = savedAzureId; + } + + if ('bbox' in feature.data.geometry) { + delete feature.data.geometry.bbox; + } +} + +function deleteUnitPrevEdits(units, selectedLevel) { + units.features = units.features.filter(item => item.properties.level_id !== selectedLevel.id); +} + export { currentEditData, groupAndSort, drawingModeChanged, - writeToInfoPanel, - grabToPointer + grabToPointer, + updateLevels, + setFields, + deleteUnitPrevEdits, + grabAndGrabbing, + updateSelectedColor }; \ No newline at end of file diff --git a/src/pages/conversion/places-preview-map/indes.style.js b/src/pages/conversion/places-preview-map/indes.style.js index 0f78087..6179d5d 100644 --- a/src/pages/conversion/places-preview-map/indes.style.js +++ b/src/pages/conversion/places-preview-map/indes.style.js @@ -18,27 +18,21 @@ export const imdfPreviewMap = css` export const layerSelect = css ` position: relative; - top: 65%; + bottom: -55%; `; + export const textWrapper = css ` - flex: 1 1 4rem; + flex: 1 1 5rem; position: relative; display: inline; - margin-top: 1%; `; -export const textArea = css ` - width: 95%; - height: 95%; - white-space: nowrap; - resize: none; +export const toolbar = css ` + backgroundColor: #f0f0f0, + border: 2px solid #0078d4, + color: #0078d4, + cursor: not-allowed, + pointerEvents: none `; -export const saveButtonWrapper = css ` - display: flex; - flex-direction: row; - justify-content: flex-end; -`; - - diff --git a/src/pages/conversion/places-preview-map/index.js b/src/pages/conversion/places-preview-map/index.js index 1807348..d90d4ec 100644 --- a/src/pages/conversion/places-preview-map/index.js +++ b/src/pages/conversion/places-preview-map/index.js @@ -1,29 +1,32 @@ -import { Map, layer, source, control } from 'azure-maps-control'; +import { Map, layer, source, control, HtmlMarker } from 'azure-maps-control'; import { control as draw_control, drawing } from 'azure-maps-drawing-tools'; import { getDomain, useConversionStore, useLevelsStore, useUserStore } from 'common/store'; -import { useEffect, useMemo, useState } from 'react'; -import { DefaultButton } from '@fluentui/react'; -import { logsButton } from '../style'; -import { imdfPreviewMap, imdfPreviewMapWrapper, layerSelect, textWrapper, textArea, mapTextWrapper, saveButtonWrapper } from './indes.style'; +import { useEffect, useMemo, useState, useRef } from 'react'; +import { imdfPreviewMap, imdfPreviewMapWrapper, layerSelect, textWrapper, mapTextWrapper } from './indes.style'; import LevelSelector from './level-selector'; import LayerSelector from './layer-selector'; import MapNotification from './map-notification'; import { calculateBoundingBox, getFeatureLabel, getFillStyles, getLineStyles, getTextStyle, processZip } from './utils'; -import { currentEditData, groupAndSort, drawingModeChanged, writeToInfoPanel, grabToPointer } from './imdf-model-helpers'; +import { currentEditData, groupAndSort, drawingModeChanged, updateLevels, setFields, deleteUnitPrevEdits, grabAndGrabbing, grabToPointer, updateSelectedColor } from './imdf-model-helpers'; +import { JsonEditor } from 'json-edit-react' import 'azure-maps-drawing-tools/dist/atlas-drawing.min.css'; import 'azure-maps-control/dist/atlas.min.css'; -const PlacesPreviewMap = ({ style }) => { +const PlacesPreviewMap = ({ style, unitsChanged, levelsChanged, footprintChanged, buildingChanged }) => { const [geography, subscriptionKey] = useUserStore(s => [s.geography, s.subscriptionKey]); const [imdfPackageLocation] = useConversionStore(s => [s.imdfPackageLocation]); const [language] = useLevelsStore(s => [s.language]); - const [units, setUnits] = useState({ features: [] }); + const [units, setUnits] = useState({ features: [], type: 'FeatureCollection' }); const [levels, setLevels] = useState({ features: [] }); - - // const [building, setBuilding] = useState({ features: [] }); // building will be used eventually + const [building, setBuilding] = useState({ features: [] }); const [footprint, setFootprint] = useState({ features: [] }); + const [jsonData, setJsonData] = useState({}); + const newDataRef = useRef(false); + + // const [prevStates, setPrevStates] = useState([]); // Tracking previous changes for Undo feature + useEffect(() => { if (!imdfPackageLocation) return; @@ -36,13 +39,15 @@ const PlacesPreviewMap = ({ style }) => { if (unitFile && levelFile && buildingFile && footprintFile) { setUnits(unitFile.content); setLevels(levelFile.content); - // setBuilding(buildingFile.content); + setBuilding(buildingFile.content); setFootprint(footprintFile.content); } }); }, [imdfPackageLocation]); const [selectedLevelId, setSelectedLevelId] = useState(null); + const [selectedLayerId, setSelectedLayerId] = useState(null); + const [drawNotif, setDrawNotif] = useState(false); // Triggers drawing tip if true const selectedLevel = useMemo(() => { const level = levels.features.find(item => item.id === selectedLevelId) || levels.features[0] || {}; @@ -50,8 +55,6 @@ const PlacesPreviewMap = ({ style }) => { return level; }, [levels, selectedLevelId]); - const [selectedLayerId, setSelectedLayerId] = useState(null); - useEffect(() => { var drawingManager; @@ -61,7 +64,7 @@ const PlacesPreviewMap = ({ style }) => { language: 'en-US', domain: getDomain(geography), staticAssetsDomain: getDomain(geography), - style: 'blank', + style: 'dark', }); map.controls.add([new control.ZoomControl()], { @@ -70,6 +73,8 @@ const PlacesPreviewMap = ({ style }) => { map.events.add('ready', () => { var drawingToolbar; + setJsonData({}); + if(selectedLayerId === 'unitButton') { drawingToolbar = new draw_control.DrawingToolbar({ position: 'bottom-right', @@ -104,18 +109,17 @@ const PlacesPreviewMap = ({ style }) => { }); footprintInteractions(footprint, drawingManager, map); + } else if(selectedLayerId === 'buildingButton') { + buildingInteractions(building, map); } else { - drawingToolbar = new draw_control.DrawingToolbar({ - position: 'bottom-right', - style: 'light', - buttons: ['edit-geometry'] - }); - - drawingManager = new drawing.DrawingManager(map, { - toolbar: drawingToolbar - }); + fullViewInteractions(units, levels, map); + } - fullViewInteractions(units, levels, drawingManager, map); + if(selectedLayerId !== 'buildingButton') { + var layers = drawingManager.getLayers(); + map.events.add('mousedown', layers.polygonLayer, handleDeletion); + map.events.add('touchstart', layers.polygonLayer, handleDeletion); + map.events.add('click', layers.polygonLayer, handleDeletion); } }); @@ -128,107 +132,178 @@ const PlacesPreviewMap = ({ style }) => { features = map.layers.getRenderedShapes(e.position, 'levelClick'); else if(selectedLayerId === 'footprintButton') features = map.layers.getRenderedShapes(e.position, 'footprintClick'); + else if(selectedLayerId === 'buildingButton') { + + } else features = map.layers.getRenderedShapes(e.position, ['unitClick', 'levelClick']); - features.forEach(function (feature) { - writeToInfoPanel(feature.data); - }); + if(selectedLayerId !== 'buildingButton') { + features.forEach(function (feature) { + const newData = feature.data || {}; + setJsonData(newData); + }); + } }); - // Shows change in corresponding feature color when mouse hovers over that feature - function featureHover(layerName, polygonHoverLayer) { - map.events.add('mousemove', layerName, function (e) { - polygonHoverLayer.setOptions({ filter: ['==', ['get', '_azureMapsShapeId'], e.shapes[0].getProperties()['_azureMapsShapeId']] }); - }); + grabAndGrabbing(map); - map.events.add('mouseleave', layerName, function (e) { - polygonHoverLayer.setOptions({ filter: ['==', ['get', '_azureMapsShapeId'], ''] }); - }); - } + // Shows change in corresponding feature color when mouse hovers over OR clicks on that feature + let selectedFeatureID = null; + function featureHoverClick(layerName, hoverLayer, clickLayer, unitSelected=false) { + grabToPointer(layerName, hoverLayer, map); - // Entry point when "unit.geojson" is pressed; the following code should be refactored due to redundancy - function unitInteractions(units, drawingManager, map) { - var unitLayer, unitLines, polygonHoverLayer, unitSymbols; - var layersAdded = [unitLayer, unitLines, polygonHoverLayer, unitSymbols]; - const groupedFeatures = groupAndSort(units, language, selectedLevel); - const keys = Object.keys(groupedFeatures); - - keys.forEach(category => { - const features = groupedFeatures[category].features; - - const dataSource = new source.DataSource(); - map.sources.add(dataSource); - dataSource.add(features); - - unitLayer = new layer.PolygonLayer(dataSource, 'unitClick', getFillStyles('unit', category)); - unitLines = new layer.LineLayer(dataSource, null, getLineStyles('unit', category)); - polygonHoverLayer = new layer.PolygonLayer(dataSource, null, { - fillColor: 'rgba(150, 50, 255, 0.2)', - filter: ['==', ['get', 'id'], ''] - }); + map.events.add('click', layerName, function (e) { + const clickedFeatureID = e.shapes[0].getProperties()['_azureMapsShapeId']; + map.getCanvas().style.cursor = 'pointer'; - unitSymbols = new layer.SymbolLayer(dataSource, null, getTextStyle(category)); - - map.layers.add([unitLayer, polygonHoverLayer, unitLines, unitSymbols], 'roomPolygons'); - - grabToPointer([unitLayer, polygonHoverLayer], map); - featureHover(unitLayer, polygonHoverLayer); - - // map.layers.add(new layer.SymbolLayer(dataSource, null, getTextStyle(category)), 'roomLabels'); - - var drawingSource = drawingManager.getSource(); - drawingSource.add(features); - - let dmLayers = drawingManager.getLayers(); - dmLayers.polygonLayer.setOptions({ visible: false }); - dmLayers.polygonOutlineLayer.setOptions({ visible: false }); - layersAdded = [unitLayer, unitLines, polygonHoverLayer, unitSymbols]; - - map.events.add('drawingmodechanged', drawingManager, (e) => { - let dmLayers = drawingManager.getLayers(); - layersAdded = [unitLayer, unitLines, polygonHoverLayer, unitSymbols]; - - if (e === 'idle') { - dmLayers.polygonLayer.setOptions({ visible: false }); - dmLayers.polygonOutlineLayer.setOptions({ visible: false }); - map.layers.remove([unitLayer, unitLines, polygonHoverLayer]); - - unitLayer = new layer.PolygonLayer(drawingManager.getSource(), 'unitClick', getFillStyles('unit', category)); - unitLines = new layer.LineLayer(drawingManager.getSource(), null, getLineStyles('unit', category)); - polygonHoverLayer = new layer.PolygonLayer(drawingManager.getSource(), null, { - fillColor: 'rgba(150, 50, 255, 0.2)', - filter: ['==', ['get', 'id'], ''] - }); - unitSymbols = new layer.SymbolLayer(drawingManager.getSource(), null, getTextStyle(category)); - - map.layers.add([unitLayer, polygonHoverLayer, unitLines, unitSymbols], 'roomPolygons'); - - grabToPointer([unitLayer, polygonHoverLayer], map); - featureHover(unitLayer, polygonHoverLayer); - layersAdded = [unitLayer, unitLines, polygonHoverLayer, unitSymbols]; - } - else if (e === 'edit-geometry' || e === 'erase-geometry' || e === 'draw-polygon') { - drawingModeChanged(layersAdded); - dmLayers.polygonLayer.setOptions({ visible: true }); - dmLayers.polygonOutlineLayer.setOptions({ visible: true }); - } - else { - // This will eventually be a visible pop-up - console.log('Not a valid drawing toolbar option.'); - } - }); - - currentEditData(map, drawingManager); + if (selectedFeatureID !== clickedFeatureID) { + // If feature is clicked, change color of feature + selectedFeatureID = clickedFeatureID; + clickLayer.setOptions({ + filter: ['==', ['get', '_azureMapsShapeId'], clickedFeatureID] + }); + + // Handles the change in color of the feature when clicked (for full view) + updateSelectedColor(map, unitSelected) + + var features; + if(selectedLayerId === 'unitButton') + features = map.layers.getRenderedShapes(e.position, 'unitClick'); + else if(selectedLayerId === 'levelButton') + features = map.layers.getRenderedShapes(e.position, 'levelFill'); + else if(selectedLayerId === 'footprintButton') + features = map.layers.getRenderedShapes(e.position, 'footprintClick'); + else + features = map.layers.getRenderedShapes(e.position, ['unitClick', 'levelClick']); + + features.forEach(function (feature) { + const newData = feature.data || {}; + setJsonData(newData); + }); + } + else { + // If feature clicked is currently chosen OR it is another feature, change color of feature back to original + selectedFeatureID = null; + clickLayer.setOptions({ + filter: ['==', ['get', '_azureMapsShapeId'], ''] + }); + setJsonData({}); + } }); } + function handleDeletion(e) { + if (drawingManager.getOptions().mode === 'erase-geometry') { + if (window.confirm('Do you want to proceed with removing this feature?')) { + drawingManager.getSource().remove(e.shapes[0]); + } + } + } + + // Entry point when "unit.geojson" is pressed; the following code should be refactored due to redundancy + function unitInteractions(units, drawingManager, map) { + // Update the units state with the edited features (for updating zip) + unitsChanged(units); + + var unitLayer, unitLines, polygonHoverLayer, polygonClickLayer, unitSymbols; + var layersAdded = [unitLayer, unitLines, polygonHoverLayer, polygonClickLayer, unitSymbols]; + const groupedFeatures = groupAndSort(units, language, selectedLevel); + const keys = Object.keys(groupedFeatures); + + keys.forEach(category => { + var features = groupedFeatures[category].features; + const dataSource = new source.DataSource(); + map.sources.add(dataSource); + dataSource.add(features); + + unitLayer = new layer.PolygonLayer(dataSource, 'unitClick', getFillStyles('unit', category)); + unitLines = new layer.LineLayer(dataSource, null, getLineStyles('unit', category)); + polygonHoverLayer = new layer.PolygonLayer(dataSource, null, { + fillColor: 'rgba(135, 206, 250, 0.8)', + filter: ['==', ['get', 'id'], ''], + cursor: 'pointer !important', + }); + + polygonClickLayer = new layer.PolygonLayer(dataSource, 'unitClickChange', { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] , + cursor: 'pointer !important', + }); + + unitSymbols = new layer.SymbolLayer(dataSource, null, getTextStyle(category)); + map.layers.add([unitLayer, polygonHoverLayer, unitLines, polygonClickLayer, unitSymbols], 'roomPolygons'); + featureHoverClick(unitLayer, polygonHoverLayer, polygonClickLayer, true); + + var drawingSource = drawingManager.getSource(); + drawingSource.add(features); + + let dmLayers = drawingManager.getLayers(); + dmLayers.polygonLayer.setOptions({ visible: false }); + dmLayers.polygonOutlineLayer.setOptions({ visible: false }); + layersAdded = [unitLayer, unitLines, polygonHoverLayer, polygonClickLayer, unitSymbols]; + + // Saving previous states for undo functionality + // if(prevStates.length === 0 || units.features !== prevStates[prevStates.length - 1].features) { + // setPrevStates(prev => [...prev, {...units}]); + // } + + map.events.add('drawingmodechanged', drawingManager, (e) => { + let dmLayers = drawingManager.getLayers(); + layersAdded = [unitLayer, unitLines, polygonHoverLayer, polygonClickLayer, unitSymbols]; + + if (e === 'idle') { + setDrawNotif(false); + let updatedFeatures = drawingManager.getSource().shapes; + + deleteUnitPrevEdits(units, selectedLevel); + + const newFeatures = updatedFeatures.reduce((acc, feature) => { + if (isNaN(feature.data.properties.ordinal)) { + setFields(feature, selectedLevel); + acc.push(feature.data); + } + return acc; + }, []); + + setUnits(prevUnits => ({ + features: [...prevUnits.features, ...newFeatures], + })); + } + else if (e === 'edit-geometry' || e === 'erase-geometry' || e === 'draw-polygon') { + drawingModeChanged(layersAdded); + dmLayers.polygonLayer.setOptions({ visible: true }); + dmLayers.polygonOutlineLayer.setOptions({ visible: true }); + + if(e === 'draw-polygon') { + setDrawNotif(true); + map.events.add('mousedown', function (e) { + map.getCanvas().style.cursor = 'crosshair'; + }); + + map.events.add('mouseup', function (e) { + map.getCanvas().style.cursor = 'crosshair'; + }); + } + if(e === 'edit-geometry') { + currentEditData(map, drawingManager, setJsonData); + } + } + else { + // This will eventually be a visible pop-up + console.log('Not a valid drawing toolbar option.'); + } + }); + }); + } + // Entry point when "level.geojson" is pressed; the following code should be refactored due to redundancy function levelInteractions(levels, drawingManager, map) { // Retrieve information about the level currently chosen by user + levelsChanged(levels); const selectedLevelDetails = levels.features.filter(item => item.id === selectedLevel.id); - var lineLayer, lineHoverLayer; - var layersAdded = [lineLayer, lineHoverLayer]; + var lineLayer, lineHoverLayer, lineClickLayer, linePolygonLayer; + var layersAdded = [lineLayer, lineHoverLayer, lineClickLayer, linePolygonLayer]; const dataSource = new source.DataSource(); map.sources.add(dataSource); @@ -236,15 +311,20 @@ const PlacesPreviewMap = ({ style }) => { // Displays outline of level + change in color when cursor is hovering lineLayer = new layer.LineLayer(dataSource, 'levelClick', getLineStyles('level', 'walkway')); - lineHoverLayer = new layer.LineLayer(dataSource, null, { - fillColor: 'rgba(150, 50, 255, 0.2)', + linePolygonLayer = new layer.PolygonLayer(dataSource, 'levelFill', getFillStyles('level', 'unspecified')); + lineHoverLayer = new layer.PolygonLayer(dataSource, null, { + fillColor: 'rgba(135, 206, 250, 0.8)', filter: ['==', ['get', '_azureMapsShapeId'], ''] }); - map.layers.add([lineLayer, lineHoverLayer], 'walkwayPolygons'); - grabToPointer([lineLayer, lineHoverLayer], map); - layersAdded = [lineLayer, lineHoverLayer]; - featureHover(lineLayer, lineHoverLayer); + lineClickLayer = new layer.PolygonLayer(dataSource, 'lineClickLayer', { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] + }); + + map.layers.add([lineLayer, linePolygonLayer, lineHoverLayer, lineClickLayer], 'walkwayPolygons'); + layersAdded = [lineLayer, lineHoverLayer, lineClickLayer, linePolygonLayer]; + featureHoverClick(linePolygonLayer, lineHoverLayer, lineClickLayer); var drawingSource = drawingManager.getSource(); drawingSource.add(selectedLevelDetails); @@ -260,32 +340,53 @@ const PlacesPreviewMap = ({ style }) => { dmLayers.polygonOutlineLayer.setOptions({ visible: false }); var lineLayer = new layer.LineLayer(drawingManager.getSource(), 'levelClick', getLineStyles('level', 'walkway')); - lineHoverLayer = new layer.LineLayer(drawingManager.getSource(), null, { - fillColor: 'rgba(150, 50, 255, 0.2)', + linePolygonLayer = new layer.PolygonLayer(drawingManager.getSource(), 'levelFill', getFillStyles('level', 'unspecified')); + lineHoverLayer = new layer.PolygonLayer(drawingManager.getSource(), null, { + fillColor: 'rgba(135, 206, 250, 0.8)', filter: ['==', ['get', '_azureMapsShapeId'], ''] }); + + lineClickLayer = new layer.PolygonLayer(drawingManager.getSource(), 'lineClickLayer', { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] + }); + + map.layers.add([lineLayer, lineHoverLayer, lineClickLayer, linePolygonLayer], 'walkwayPolygons'); + layersAdded = [lineLayer, lineHoverLayer, lineClickLayer, linePolygonLayer]; + featureHoverClick(linePolygonLayer, lineHoverLayer, lineClickLayer); + + let updatedFeatures = drawingManager.getSource().shapes; + setLevels(prevLevels => updateLevels(prevLevels, selectedLevel, (updatedFeatures[0]).data)); - map.layers.add([lineLayer, lineHoverLayer], 'walkwayPolygons'); - grabToPointer([lineLayer, lineHoverLayer], map); - layersAdded = [lineLayer, lineHoverLayer]; - featureHover(lineLayer, lineHoverLayer); + // Update the levels state with the edited features (for updating zip) + levelsChanged(levels); } - else if (e === 'edit-geometry' || e === 'erase-geometry' || e === 'draw-polygon') { + else if (e === 'edit-geometry') { drawingModeChanged(layersAdded); dmLayers.polygonOutlineLayer.setOptions({ visible: true }); - dmLayers.polygonLayer.setOptions({ visible: false }); + dmLayers.polygonLayer.setOptions({ visible: true }); + + currentEditData(map, drawingManager, setJsonData); + + document.addEventListener('keydown', function (e) { + // Check if the delete or backspace key is pressed + if (e.key === 'Delete' || e.key === 'Backspace') { + drawingManager.setOptions({ mode: 'idle' }); + } + }); } else { // This will eventually be a visible pop-up console.log('Not a valid drawing toolbar option.'); } - }); + }); } // Entry point when "footprint.geojson" is pressed; the following code should be refactored due to redundancy function footprintInteractions(footprint, drawingManager, map) { - var footprintLayer, footprintLines, footprintHoverLayer; - var layersAdded = [footprintLayer, footprintLines, footprintHoverLayer]; + footprintChanged(footprint); + var footprintLayer, footprintLines, footprintHoverLayer, footprintClickLayer; + var layersAdded = [footprintLayer, footprintLines, footprintHoverLayer, footprintClickLayer]; const groupedFeatures = {}; const keys = Object.keys(groupedFeatures); @@ -298,14 +399,18 @@ const PlacesPreviewMap = ({ style }) => { footprintLines = new layer.LineLayer(dataSource, null, getLineStyles('footprint', 'walkway')); footprintLayer = new layer.PolygonLayer(dataSource, 'footprintClick', getFillStyles('footprint', keys.category)); footprintHoverLayer = new layer.PolygonLayer(dataSource, null, { - fillColor: 'rgba(150, 50, 255, 0.2)', + fillColor: 'rgba(135, 206, 250, 0.8)', filter: ['==', ['get', '_azureMapsShapeId'], ''] }); - map.layers.add([footprintLayer, footprintHoverLayer, footprintLines], 'roomPolygons'); - grabToPointer([footprintLayer, footprintHoverLayer], map); - featureHover(footprintLayer, footprintHoverLayer); - layersAdded = [footprintLayer, footprintLines, footprintHoverLayer]; + footprintClickLayer = new layer.PolygonLayer(dataSource, null, { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] + }); + + map.layers.add([footprintLayer, footprintHoverLayer, footprintLines, footprintClickLayer], 'roomPolygons'); + featureHoverClick(footprintLayer, footprintHoverLayer, footprintClickLayer); + layersAdded = [footprintLayer, footprintLines, footprintHoverLayer, footprintClickLayer]; var drawingSource = drawingManager.getSource(); @@ -324,39 +429,113 @@ const PlacesPreviewMap = ({ style }) => { footprintLines = new layer.LineLayer(drawingManager.getSource(), null, getLineStyles('footprint', 'walkway')); footprintLayer = new layer.PolygonLayer(drawingManager.getSource(), 'footprintClick', getFillStyles('footprint', keys.category)); footprintHoverLayer = new layer.PolygonLayer(drawingManager.getSource(), null, { - fillColor: 'rgba(150, 50, 255, 0.2)', + fillColor: 'rgba(135, 206, 250, 0.8)', filter: ['==', ['get', '_azureMapsShapeId'], ''] }); + footprintClickLayer = new layer.PolygonLayer(drawingManager.getSource(), null, { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] + }); + + map.layers.add([footprintLayer, footprintHoverLayer, footprintLines, footprintClickLayer], 'roomPolygons'); + featureHoverClick(footprintLayer, footprintHoverLayer, footprintClickLayer); + layersAdded = [footprintLayer, footprintLines, footprintHoverLayer, footprintClickLayer]; - map.layers.add([footprintLayer, footprintHoverLayer, footprintLines], 'roomPolygons'); - grabToPointer([footprintLayer, footprintHoverLayer], map); - featureHover(footprintLayer, footprintHoverLayer); - layersAdded = [footprintLayer, footprintLines, footprintHoverLayer]; + let updatedFeatures = drawingManager.getSource().shapes; + footprint.features[0] = updatedFeatures[0].data; + + // Update the footprint state with the edited features (for updating zip) + footprintChanged(footprint); } - else if (e === 'edit-geometry' || e === 'erase-geometry' || e === 'draw-polygon') { + else if (e === 'edit-geometry') { drawingModeChanged(layersAdded); dmLayers.polygonLayer.setOptions({ visible: true }); dmLayers.polygonOutlineLayer.setOptions({ visible: true }); + + currentEditData(map, drawingManager, setJsonData); } else { // This will eventually be a visible pop-up console.log('Not a valid drawing toolbar option.'); } - }); + }); + } + + function buildingInteractions(building, map) { + var buildingLayer = new HtmlMarker({ + position: building.features[0].properties.display_point.coordinates, + }); + map.events.add('click', buildingLayer, highlight); + map.markers.add(buildingLayer); + + function highlight(e) { + setJsonData(building.features[0]); + } + + map.setCamera({ + zoom: 12 + }); } // Entry point when "full view" is pressed; the following code may need to be changed to allow fill color of units while editing - // Needs to be given edit properties - function fullViewInteractions(units, levels, drawingManager, map) { - unitInteractions(units, drawingManager, map); - levelInteractions(levels, drawingManager, map); + function fullViewInteractions(units, levels, map) { + unitsChanged(units); + levelsChanged(levels); + + var unitLayer, unitLines, polygonHoverLayer, polygonClickLayer, unitSymbols, lineLayer, lineHoverLayer, lineClickLayer; + const groupedFeatures = groupAndSort(units, language, selectedLevel); + const keys = Object.keys(groupedFeatures); + + keys.forEach(category => { + var features = groupedFeatures[category].features; + const dataSource = new source.DataSource(); + map.sources.add(dataSource); + dataSource.add(features); + + unitLayer = new layer.PolygonLayer(dataSource, 'unitClick', getFillStyles('unit', category)); + unitLines = new layer.LineLayer(dataSource, null, getLineStyles('unit', category)); + polygonHoverLayer = new layer.PolygonLayer(dataSource, null, { + fillColor: 'rgba(135, 206, 250, 0.8)', + filter: ['==', ['get', 'id'], ''], + cursor: 'pointer !important', + }); + + polygonClickLayer = new layer.PolygonLayer(dataSource, 'unitClickChange', { + fillColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] , + cursor: 'pointer !important', + }); + + unitSymbols = new layer.SymbolLayer(dataSource, null, getTextStyle(category)); + map.layers.add([unitLayer, polygonHoverLayer, unitLines, polygonClickLayer, unitSymbols], 'roomPolygons'); + featureHoverClick(unitLayer, polygonHoverLayer, polygonClickLayer, true); + }); + + const selectedLevelDetails = levels.features.filter(item => item.id === selectedLevel.id); + const dataSource = new source.DataSource(); + map.sources.add(dataSource); + dataSource.add(selectedLevelDetails); + + // Displays outline of level + change in color when cursor is hovering + lineLayer = new layer.LineLayer(dataSource, 'levelClick', getLineStyles('level', 'walkway')); + lineHoverLayer = new layer.LineLayer(dataSource, null, { + strokeColor: 'rgba(135, 206, 250, 0.8)', + filter: ['==', ['get', '_azureMapsShapeId'], ''] + }); + lineClickLayer = new layer.LineLayer(dataSource, 'lineClickLayer', { + strokeColor: 'rgba(75, 146, 210, 0.8)', + filter: ['==', ['get', 'id'], ''] + }); + + map.layers.add([lineLayer, lineHoverLayer, lineClickLayer], 'walkwayPolygons'); + featureHoverClick(lineLayer, lineHoverLayer, lineClickLayer, false); } // Cleanup function to remove the map instance when component unmounts or reinitializes return () => { map.dispose(); }; - }, [units, levels, footprint, selectedLevel, selectedLayerId, subscriptionKey, geography, language]); + }, [ units, levels, footprint, building, selectedLevel, selectedLayerId, subscriptionKey, geography, language, imdfPackageLocation, unitsChanged, levelsChanged, footprintChanged ]); const handleLevelChange = levelId => { setSelectedLevelId(levelId); @@ -366,6 +545,56 @@ const PlacesPreviewMap = ({ style }) => { setSelectedLayerId(layerId); }; + const updateJsonData = (newData) => { + setJsonData(newData); + }; + + // Handles updates when a property is changed in the JSON editor + const handleUpdate = ({ newData, currentData, newValue, currentValue, name, path }) => { + if(newData.feature_type === 'building') { + // Building + building.features = [newData]; + buildingChanged(building); + } + else if (newData.properties.name && isNaN(newData.properties.ordinal)) { + // Unit + let editedIndex = units.features.findIndex(unit => unit.id === currentData.id); + if (editedIndex !== -1) { + // Replace the old data with the new data for specific feature, then save to zip + const updatedFeatures = [...units.features]; + updatedFeatures[editedIndex] = { + ...newData, + properties: { + ...newData.properties, + label: newData.properties.name.en, + }, + }; + + setUnits({ features: updatedFeatures }); // To update state to trigger map refresh + unitsChanged(units); // To update zip + newDataRef.current = true; + } + else { + console.log('Invalid property change.'); + } + } + else if (newData.properties.name) { + // Level + let editedIndex = levels.features.findIndex(level => level.id === currentData.id); + if (editedIndex !== -1) { + // Replace the old data with the new data for specific feature, then save to zip + levels.features[editedIndex] = newData; + levelsChanged(levels); + } + else { + console.log('Invalid property change.'); + } + } + + return true; + }; + + return (
@@ -376,7 +605,6 @@ const PlacesPreviewMap = ({ style }) => { options={levels.features.map(level => ({ key: level.id, text: getFeatureLabel(level, language) }))} /> -
{
Zoom in to see labels and icons. + {drawNotif && Click to draw a point. To connect the final lines of current drawing, press 'c'.}
-
- +
+ { + const allowedFields = [ + 'properties.name.en', + 'properties.category' + ]; + return !allowedFields.includes(path.join('.')); + }} + restrictTypeSelection={ ({ path, value }) => { + if (typeof value === 'string') + return ['string']; + else + return ['string', 'number', 'boolean', 'array', 'object']; + }} + rootFontSize={12} + indent={2} + theme="githubLight" + onUpdate={handleUpdate} + />
-
- -
- Save Changes -
- -
+
); }; diff --git a/src/pages/conversion/places-preview-map/layer-selector/index.js b/src/pages/conversion/places-preview-map/layer-selector/index.js index 62b1592..821da13 100644 --- a/src/pages/conversion/places-preview-map/layer-selector/index.js +++ b/src/pages/conversion/places-preview-map/layer-selector/index.js @@ -4,10 +4,11 @@ import { buttonStyle, layerSelectorWrapper, selectedButtonStyle } from './index. const layerButtons = [ // will need to add a "buildingButton" once basemap is loaded - {id: 'fullViewButton', text: 'full view'}, - {id: 'footprintButton', text: 'footprint.geojson'}, - {id: 'levelButton', text: 'level.geojson'}, {id: 'unitButton', text: 'unit.geojson'}, + {id: 'levelButton', text: 'level.geojson'}, + {id: 'footprintButton', text: 'footprint.geojson'}, + {id: 'buildingButton', text: 'building.geojson'}, + {id: 'fullViewButton', text: 'full view'}, ]; const LayerSelector = props => { diff --git a/src/pages/conversion/places-preview-map/layer-selector/index.style.js b/src/pages/conversion/places-preview-map/layer-selector/index.style.js index c66977a..29a46a3 100644 --- a/src/pages/conversion/places-preview-map/layer-selector/index.style.js +++ b/src/pages/conversion/places-preview-map/layer-selector/index.style.js @@ -32,6 +32,7 @@ export const buttonStyle = css` background-repeat: no-repeat; overflow: hidden; + &:hover { color: #31acce; } diff --git a/src/pages/conversion/places-preview-map/utils.js b/src/pages/conversion/places-preview-map/utils.js index ca37ff3..4e75a30 100644 --- a/src/pages/conversion/places-preview-map/utils.js +++ b/src/pages/conversion/places-preview-map/utils.js @@ -7,13 +7,10 @@ import stairs from './assets/stairs.svg'; import { categoryToFillColor, defaultFillColor, levelStyles, levelOnlyStyles, textStyles, footprintStyles, unitStyles } from './theme/floorPlanStyles'; export function getFillStyles(featureType, category) { - if (featureType !== 'unit') { - return {}; - } - var fillStyle = { fillColor: categoryToFillColor.has(category) ? categoryToFillColor.get(category) : defaultFillColor, fillOpacity: 1, + cursor: 'pointer !important', }; return fillStyle; @@ -41,6 +38,7 @@ export function getLineStyles(featureType, category) { strokeWidth: levelStyles.lineWidth, lineJoin: 'round', lineCap: 'round', + fillColor: 'hsla(0, 0%, 0%, 0)', }; }