diff --git a/.github/workflows/pull-request.yml b/.github/workflows/pull-request.yml index 32f9cee912..3846371ed5 100644 --- a/.github/workflows/pull-request.yml +++ b/.github/workflows/pull-request.yml @@ -30,7 +30,7 @@ jobs: - name: Set up Node.js uses: actions/setup-node@v4 with: - node-version: '20.x' + node-version: '22.x' - name: Install Dependencies run: npm install @@ -176,7 +176,7 @@ jobs: - name: Set up Node.js uses: actions/setup-node@v4 with: - node-version: '20.x' + node-version: '22.x' - name: Install Dependencies run: npm install @@ -221,7 +221,7 @@ jobs: - name: Set up Node.js uses: actions/setup-node@v4 with: - node-version: '20.x' + node-version: '22.x' - name: resolve dependency run: npm install -g @graphql-inspector/cli diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index bc86d1b7d8..67e49556b4 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -24,7 +24,7 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - node-version: [20.x] + node-version: [22.x] steps: - uses: actions/checkout@v4 - uses: actions/setup-node@v4 diff --git a/.node-version b/.node-version index 790e1105f2..751f4c9f38 100644 --- a/.node-version +++ b/.node-version @@ -1 +1 @@ -v20.10.0 +v22.7.0 diff --git a/package-lock.json b/package-lock.json index 5460740ca3..a21fda97b5 100644 --- a/package-lock.json +++ b/package-lock.json @@ -15,12 +15,12 @@ "@dicebear/core": "^8.0.2", "@emotion/react": "^11.13.0", "@emotion/styled": "^11.13.0", - "@mui/icons-material": "^5.16.1", + "@mui/icons-material": "^5.16.7", "@mui/material": "^5.16.7", "@mui/private-theming": "^5.15.12", "@mui/system": "^5.14.12", "@mui/x-charts": "^7.8.0", - "@mui/x-data-grid": "^7.11.0", + "@mui/x-data-grid": "^7.15.0", "@mui/x-date-pickers": "^7.11.1", "@pdfme/generator": "^1.2.6", "bootstrap": "^5.3.3", @@ -38,12 +38,12 @@ "js-cookie": "^3.0.1", "markdown-toc": "^1.2.0", "prettier": "^3.3.2", - "react": "^17.0.2", + "react": "^18.3.1", "react-app-rewired": "^2.2.1", "react-beautiful-dnd": "^13.1.1", "react-bootstrap": "^2.10.4", "react-datepicker": "^7.3.0", - "react-dom": "^17.0.2", + "react-dom": "^18.3.1", "react-google-recaptcha": "^3.1.0", "react-i18next": "^12.3.1", "react-icons": "^5.2.1", @@ -52,7 +52,7 @@ "react-redux": "^7.2.5", "react-router-dom": "^6.26.0", "react-scripts": "5.0.1", - "react-toastify": "^9.0.3", + "react-toastify": "^10.0.5", "react-tooltip": "^5.27.1", "redux": "^4.1.1", "redux-thunk": "^2.3.0", @@ -63,19 +63,19 @@ }, "devDependencies": { "@babel/plugin-proposal-private-property-in-object": "^7.21.11", - "@testing-library/jest-dom": "^6.4.2", - "@testing-library/react": "^11.1.0", + "@testing-library/jest-dom": "^6.5.0", + "@testing-library/react": "^16.0.0", "@testing-library/user-event": "^12.1.10", "@types/inquirer": "^9.0.7", "@types/jest": "^26.0.24", "@types/js-cookie": "^3.0.6", - "@types/node": "^20.12.12", + "@types/node": "^22.5.2", "@types/node-fetch": "^2.6.10", - "@types/react": "^17.0.14", + "@types/react": "^18.3.3", "@types/react-beautiful-dnd": "^13.1.8", "@types/react-bootstrap": "^0.32.32", "@types/react-datepicker": "^4.1.4", - "@types/react-dom": "^17.0.9", + "@types/react-dom": "^18.3.0", "@types/react-google-recaptcha": "^2.1.5", "@types/react-router-dom": "^5.1.8", "@types/sanitize-html": "^2.13.0", @@ -2290,9 +2290,9 @@ "integrity": "sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==" }, "node_modules/@babel/runtime": { - "version": "7.25.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.0.tgz", - "integrity": "sha512-7dRy4DwXwtzBrPbZflqxnvfxLF8kdZXPkhymtDeFoFqE6ldzjQFgYTtYIFARcLEYDrqfBfYcZt1WqFxRoyC9Rw==", + "version": "7.25.6", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.6.tgz", + "integrity": "sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ==", "dependencies": { "regenerator-runtime": "^0.14.0" }, @@ -2300,19 +2300,6 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/runtime-corejs3": { - "version": "7.22.6", - "resolved": "https://registry.npmjs.org/@babel/runtime-corejs3/-/runtime-corejs3-7.22.6.tgz", - "integrity": "sha512-M+37LLIRBTEVjktoJjbw4KVhupF0U/3PYUCbBwgAd9k17hoKhRu1n935QiG7Tuxv0LJOMrb2vuKEeYUlv0iyiw==", - "dev": true, - "dependencies": { - "core-js-pure": "^3.30.2", - "regenerator-runtime": "^0.13.11" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/runtime/node_modules/regenerator-runtime": { "version": "0.14.0", "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.0.tgz", @@ -3670,12 +3657,12 @@ } }, "node_modules/@floating-ui/react": { - "version": "0.26.19", - "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.19.tgz", - "integrity": "sha512-Jk6zITdjjIvjO/VdQFvpRaD3qPwOHH6AoDHxjhpy+oK4KFgaSP871HYWUAPdnLmx1gQ+w/pB312co3tVml+BXA==", + "version": "0.26.22", + "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.22.tgz", + "integrity": "sha512-LNv4azPt8SpT4WW7Kku5JNVjLk2GcS0bGGjFTAgqOONRFo9r/aaGHHPpdiIuQbB1t8shmWyWqTTUDmZ9fcNshg==", "dependencies": { "@floating-ui/react-dom": "^2.1.1", - "@floating-ui/utils": "^0.2.4", + "@floating-ui/utils": "^0.2.7", "tabbable": "^6.0.0" }, "peerDependencies": { @@ -3696,9 +3683,9 @@ } }, "node_modules/@floating-ui/utils": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.4.tgz", - "integrity": "sha512-dWO2pw8hhi+WrXq1YJy2yCuWoL20PddgGaqTgVe4cOS9Q6qklXCiA1tJEqX6BEwRNSCP84/afac9hd4MS+zEUA==" + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.7.tgz", + "integrity": "sha512-X8R8Oj771YRl/w+c1HqAC1szL8zWQRwFvgDwT129k9ACdBoud/+/rX9V0qiMl6LWUdP9voC2nDVZYPMQQsb6eA==" }, "node_modules/@gar/promisify": { "version": "1.1.3", @@ -4232,14 +4219,6 @@ } } }, - "node_modules/@mui/base/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/core-downloads-tracker": { "version": "5.16.7", "resolved": "https://registry.npmjs.org/@mui/core-downloads-tracker/-/core-downloads-tracker-5.16.7.tgz", @@ -4250,9 +4229,9 @@ } }, "node_modules/@mui/icons-material": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-5.16.1.tgz", - "integrity": "sha512-ogQPweYba4+5XZykilwxn2/oS78uwoQ0BVBpOhhCJo0ooZsqTTsalhzP2qD/RdGqMQ8xyXPz1sYM2djTruVVVA==", + "version": "5.16.7", + "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-5.16.7.tgz", + "integrity": "sha512-UrGwDJCXEszbDI7yV047BYU5A28eGJ79keTCP4cc74WyncuVrnurlmIRxaHL8YK+LI1Kzq+/JM52IAkNnv4u+Q==", "dependencies": { "@babel/runtime": "^7.23.9" }, @@ -4318,14 +4297,6 @@ } } }, - "node_modules/@mui/material/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/private-theming": { "version": "5.16.6", "resolved": "https://registry.npmjs.org/@mui/private-theming/-/private-theming-5.16.6.tgz", @@ -4389,10 +4360,10 @@ "integrity": "sha512-Jncvs/r/d/itkxh7O7opOunTqbbSSzMTHzZkNLM+FjAOg+cYAZHrPDlYe1ZGKUYORwwb2XexlWnpZp0kZ4AHuA==", "dependencies": { "@babel/runtime": "^7.23.9", - "@mui/private-theming": "^5.16.6", - "@mui/styled-engine": "^5.16.6", + "@mui/private-theming": "^5.16.4", + "@mui/styled-engine": "^5.16.4", "@mui/types": "^7.2.15", - "@mui/utils": "^5.16.6", + "@mui/utils": "^5.16.4", "clsx": "^2.1.0", "csstype": "^3.1.3", "prop-types": "^15.8.1" @@ -4422,14 +4393,6 @@ } } }, - "node_modules/@mui/system/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/types": { "version": "7.2.15", "resolved": "https://registry.npmjs.org/@mui/types/-/types-7.2.15.tgz", @@ -4472,14 +4435,6 @@ } } }, - "node_modules/@mui/utils/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/x-charts": { "version": "7.8.0", "resolved": "https://registry.npmjs.org/@mui/x-charts/-/x-charts-7.8.0.tgz", @@ -4518,23 +4473,14 @@ } } }, - "node_modules/@mui/x-charts/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/x-data-grid": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@mui/x-data-grid/-/x-data-grid-7.11.0.tgz", - "integrity": "sha512-dXaIw3Noxc4d6xenS7J+zMPORG9ptkTW7B81P6QFovILSEuI/qebQhijy/IkqRvcCsuZYLL3nA89bp+EDI503Q==", - "dependencies": { - "@babel/runtime": "^7.24.8", - "@mui/system": "^5.16.2", - "@mui/utils": "^5.16.2", - "@mui/x-internals": "7.11.0", + "version": "7.15.0", + "resolved": "https://registry.npmjs.org/@mui/x-data-grid/-/x-data-grid-7.15.0.tgz", + "integrity": "sha512-S7FLp+SUz/fo+JAQcM4npnL70tGnL4qcYYMkjK4tPkPLrTnBWlm47D5OoGNzpPO1CnOAryFR5h0rdwbeHZmAWg==", + "dependencies": { + "@babel/runtime": "^7.25.4", + "@mui/utils": "^5.16.6", + "@mui/x-internals": "7.15.0", "clsx": "^2.1.1", "prop-types": "^15.8.1", "reselect": "^4.1.8" @@ -4547,7 +4493,10 @@ "url": "https://opencollective.com/mui-org" }, "peerDependencies": { - "@mui/material": "^5.15.14", + "@emotion/react": "^11.9.0", + "@emotion/styled": "^11.8.1", + "@mui/material": "^5.15.14 || ^6.0.0", + "@mui/system": "^5.15.14 || ^6.0.0", "react": "^17.0.0 || ^18.0.0", "react-dom": "^17.0.0 || ^18.0.0" } @@ -4565,9 +4514,10 @@ "resolved": "https://registry.npmjs.org/@mui/x-date-pickers/-/x-date-pickers-7.12.1.tgz", "integrity": "sha512-Zj8kt3SCQbJp1qhMi+A3I4KqB8i5OY2Q11mdOEathFhqN/SQm1sUjIa1G09cGP1dPDgK1a6KM6qJGNtcw/nuWA==", "dependencies": { - "@babel/runtime": "^7.25.0", - "@mui/system": "^5.16.5", - "@mui/utils": "^5.16.5", + "@babel/runtime": "^7.24.6", + "@mui/base": "^5.0.0-beta.40", + "@mui/system": "^5.15.15", + "@mui/utils": "^5.15.14", "@types/react-transition-group": "^4.4.10", "clsx": "^2.1.1", "prop-types": "^15.8.1", @@ -4624,21 +4574,13 @@ } } }, - "node_modules/@mui/x-date-pickers/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/@mui/x-internals": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.11.0.tgz", - "integrity": "sha512-GqCYylKiB4cLH9tK4JweJlT2JvPjnpXjS3TEIqtHB4BcSsezhdRrMGzHOO5zCJqkasqTirJh2t6X16Qw1llr4Q==", + "version": "7.15.0", + "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.15.0.tgz", + "integrity": "sha512-Q/IZvZhHpe64Ost1mRbdp6ML8KQQBprwwgzqo6pZbrCaWMPB2gk2jcUwdCwnLsc+gutaEPVhZ8N7it8VZcHtbg==", "dependencies": { - "@babel/runtime": "^7.24.8", - "@mui/utils": "^5.16.2" + "@babel/runtime": "^7.25.4", + "@mui/utils": "^5.16.6" }, "engines": { "node": ">=14.0.0" @@ -4928,25 +4870,25 @@ } }, "node_modules/@react-spring/animated": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/animated/-/animated-9.7.3.tgz", - "integrity": "sha512-5CWeNJt9pNgyvuSzQH+uy2pvTg8Y4/OisoscZIR8/ZNLIOI+CatFBhGZpDGTF/OzdNFsAoGk3wiUYTwoJ0YIvw==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/animated/-/animated-9.7.4.tgz", + "integrity": "sha512-7As+8Pty2QlemJ9O5ecsuPKjmO0NKvmVkRR1n6mEotFgWar8FKuQt2xgxz3RTgxcccghpx1YdS1FCdElQNexmQ==", "dependencies": { - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0" } }, "node_modules/@react-spring/core": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/core/-/core-9.7.3.tgz", - "integrity": "sha512-IqFdPVf3ZOC1Cx7+M0cXf4odNLxDC+n7IN3MDcVCTIOSBfqEcBebSv+vlY5AhM0zw05PDbjKrNmBpzv/AqpjnQ==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/core/-/core-9.7.4.tgz", + "integrity": "sha512-GzjA44niEJBFUe9jN3zubRDDDP2E4tBlhNlSIkTChiNf9p4ZQlgXBg50qbXfSXHQPHak/ExYxwhipKVsQ/sUTw==", "dependencies": { - "@react-spring/animated": "~9.7.3", - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/animated": "~9.7.4", + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" }, "funding": { "type": "opencollective", @@ -4957,35 +4899,36 @@ } }, "node_modules/@react-spring/rafz": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/rafz/-/rafz-9.7.3.tgz", - "integrity": "sha512-9vzW1zJPcC4nS3aCV+GgcsK/WLaB520Iyvm55ARHfM5AuyBqycjvh1wbmWmgCyJuX4VPoWigzemq1CaaeRSHhQ==" + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/rafz/-/rafz-9.7.4.tgz", + "integrity": "sha512-mqDI6rW0Ca8IdryOMiXRhMtVGiEGLIO89vIOyFQXRIwwIMX30HLya24g9z4olDvFyeDW3+kibiKwtZnA4xhldA==" }, "node_modules/@react-spring/shared": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/shared/-/shared-9.7.3.tgz", - "integrity": "sha512-NEopD+9S5xYyQ0pGtioacLhL2luflh6HACSSDUZOwLHoxA5eku1UPuqcJqjwSD6luKjjLfiLOspxo43FUHKKSA==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/shared/-/shared-9.7.4.tgz", + "integrity": "sha512-bEPI7cQp94dOtCFSEYpxvLxj0+xQfB5r9Ru1h8OMycsIq7zFZon1G0sHrBLaLQIWeMCllc4tVDYRTLIRv70C8w==", "dependencies": { - "@react-spring/types": "~9.7.3" + "@react-spring/rafz": "~9.7.4", + "@react-spring/types": "~9.7.4" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0" } }, "node_modules/@react-spring/types": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/types/-/types-9.7.3.tgz", - "integrity": "sha512-Kpx/fQ/ZFX31OtlqVEFfgaD1ACzul4NksrvIgYfIFq9JpDHFwQkMVZ10tbo0FU/grje4rcL4EIrjekl3kYwgWw==" + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/types/-/types-9.7.4.tgz", + "integrity": "sha512-iQVztO09ZVfsletMiY+DpT/JRiBntdsdJ4uqk3UJFhrhS8mIC9ZOZbmfGSRs/kdbNPQkVyzucceDicQ/3Mlj9g==" }, "node_modules/@react-spring/web": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/web/-/web-9.7.3.tgz", - "integrity": "sha512-BXt6BpS9aJL/QdVqEIX9YoUy8CE6TJrU0mNCqSoxdXlIeNcEBWOfIyE6B14ENNsyQKS3wOWkiJfco0tCr/9tUg==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/web/-/web-9.7.4.tgz", + "integrity": "sha512-UMvCZp7I5HCVIleSa4BwbNxynqvj+mJjG2m20VO2yPoi2pnCYANy58flvz9v/YcXTAvsmL655FV3pm5fbr6akA==", "dependencies": { - "@react-spring/animated": "~9.7.3", - "@react-spring/core": "~9.7.3", - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/animated": "~9.7.4", + "@react-spring/core": "~9.7.4", + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0", @@ -5645,35 +5588,23 @@ } }, "node_modules/@testing-library/dom": { - "version": "7.31.2", - "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-7.31.2.tgz", - "integrity": "sha512-3UqjCpey6HiTZT92vODYLPxTBWlM8ZOOjr3LX5F37/VRipW2M1kX6I/Cm4VXzteZqfGfagg8yXywpcOgQBlNsQ==", + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.0.tgz", + "integrity": "sha512-pemlzrSESWbdAloYml3bAJMEfNh1Z7EduzqPKprCH5S341frlpYnUEW0H72dLxa6IsYr+mPno20GiSm+h9dEdQ==", "dev": true, + "peer": true, "dependencies": { "@babel/code-frame": "^7.10.4", "@babel/runtime": "^7.12.5", - "@types/aria-query": "^4.2.0", - "aria-query": "^4.2.2", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", "chalk": "^4.1.0", - "dom-accessibility-api": "^0.5.6", - "lz-string": "^1.4.4", - "pretty-format": "^26.6.2" + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "pretty-format": "^27.0.2" }, "engines": { - "node": ">=10" - } - }, - "node_modules/@testing-library/dom/node_modules/aria-query": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-4.2.2.tgz", - "integrity": "sha512-o/HelwhuKpTj/frsOsbNLNgnNGVIFsVP/SW2BSF14gVl7kAfMOJ6/8wUAUvG1R1NHKrfG+2sHZTu0yauT1qBrA==", - "dev": true, - "dependencies": { - "@babel/runtime": "^7.10.2", - "@babel/runtime-corejs3": "^7.10.2" - }, - "engines": { - "node": ">=6.0" + "node": ">=18" } }, "node_modules/@testing-library/dom/node_modules/chalk": { @@ -5681,6 +5612,7 @@ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, + "peer": true, "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" @@ -5692,15 +5624,48 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, + "node_modules/@testing-library/dom/node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "peer": true, + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@testing-library/dom/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "peer": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@testing-library/dom/node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "peer": true + }, "node_modules/@testing-library/jest-dom": { - "version": "6.4.8", - "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.4.8.tgz", - "integrity": "sha512-JD0G+Zc38f5MBHA4NgxQMR5XtO5Jx9g86jqturNTt2WUfRmLDIY7iKkWHDCCTiDuFMre6nxAD5wHw9W5kI4rGw==", + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.5.0.tgz", + "integrity": "sha512-xGGHpBXYSHUUr6XsKBfs85TWlYKpTc37cSBBVrXcib2MkHLboWlkClhWF37JKlDb9KEq3dHs+f2xR7XJEWGBxA==", "dev": true, - "license": "MIT", "dependencies": { "@adobe/css-tools": "^4.4.0", - "@babel/runtime": "^7.9.2", "aria-query": "^5.0.0", "chalk": "^3.0.0", "css.escape": "^1.5.1", @@ -5722,20 +5687,30 @@ "license": "MIT" }, "node_modules/@testing-library/react": { - "version": "11.2.7", - "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-11.2.7.tgz", - "integrity": "sha512-tzRNp7pzd5QmbtXNG/mhdcl7Awfu/Iz1RaVHY75zTdOkmHCuzMhRL83gWHSgOAcjS3CCbyfwUHMZgRJb4kAfpA==", + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-16.0.0.tgz", + "integrity": "sha512-guuxUKRWQ+FgNX0h0NS0FIq3Q3uLtWVpBzcLOggmfMoUpgBnzBzvLLd4fbm6yS8ydJd94cIfY4yP9qUQjM2KwQ==", "dev": true, "dependencies": { - "@babel/runtime": "^7.12.5", - "@testing-library/dom": "^7.28.1" + "@babel/runtime": "^7.12.5" }, "engines": { - "node": ">=10" + "node": ">=18" }, "peerDependencies": { - "react": "*", - "react-dom": "*" + "@testing-library/dom": "^10.0.0", + "@types/react": "^18.0.0", + "@types/react-dom": "^18.0.0", + "react": "^18.0.0", + "react-dom": "^18.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } } }, "node_modules/@testing-library/user-event": { @@ -5771,10 +5746,11 @@ } }, "node_modules/@types/aria-query": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-4.2.2.tgz", - "integrity": "sha512-HnYpAE1Y6kRyKM/XkEuiRQhTHvkzMBurTHnpFLYLBGPIylZNPs9jJcuOOYWxPLJCSEtmZT0Y8rHDokKN7rRTig==", - "dev": true + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "peer": true }, "node_modules/@types/babel__core": { "version": "7.20.1", @@ -6012,11 +5988,11 @@ "peer": true }, "node_modules/@types/node": { - "version": "20.12.12", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.12.tgz", - "integrity": "sha512-eWLDGF/FOSPtAvEqeRAQ4C8LSA7M1I7i0ky1I8U7kD1J5ITyW3AsRhQrKVoWf5pFKZ2kILsEGJhsI9r93PYnOw==", + "version": "22.5.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.2.tgz", + "integrity": "sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==", "dependencies": { - "undici-types": "~5.26.4" + "undici-types": "~6.19.2" } }, "node_modules/@types/node-fetch": { @@ -6081,12 +6057,11 @@ "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==" }, "node_modules/@types/react": { - "version": "17.0.62", - "resolved": "https://registry.npmjs.org/@types/react/-/react-17.0.62.tgz", - "integrity": "sha512-eANCyz9DG8p/Vdhr0ZKST8JV12PhH2ACCDYlFw6DIO+D+ca+uP4jtEDEpVqXZrh/uZdXQGwk7whJa3ah5DtyLw==", + "version": "18.3.3", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.3.tgz", + "integrity": "sha512-hti/R0pS0q1/xx+TsI73XIqk26eBsISZ2R0wUijXIngRK9R/e7Xw/cXVxQK7R5JjW+SV4zGcn5hXjudkN/pLIw==", "dependencies": { "@types/prop-types": "*", - "@types/scheduler": "*", "csstype": "^3.0.2" } }, @@ -6121,12 +6096,12 @@ } }, "node_modules/@types/react-dom": { - "version": "17.0.20", - "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-17.0.20.tgz", - "integrity": "sha512-4pzIjSxDueZZ90F52mU3aPoogkHIoSIDG+oQ+wQK7Cy2B9S+MvOqY0uEA/qawKz381qrEDkvpwyt8Bm31I8sbA==", + "version": "18.3.0", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.0.tgz", + "integrity": "sha512-EhwApuTmMBmXuFOikhQLIBUn6uFg81SwLMOAUgodJF14SOBOCMdU04gDoYi0WOJJHD144TL32z4yDqCW3dnkQg==", "dev": true, "dependencies": { - "@types/react": "^17" + "@types/react": "*" } }, "node_modules/@types/react-google-recaptcha": { @@ -6200,11 +6175,6 @@ "htmlparser2": "^8.0.0" } }, - "node_modules/@types/scheduler": { - "version": "0.16.3", - "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.3.tgz", - "integrity": "sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==" - }, "node_modules/@types/semver": { "version": "7.5.0", "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.0.tgz", @@ -6648,9 +6618,9 @@ } }, "node_modules/@wry/equality": { - "version": "0.5.6", - "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.5.6.tgz", - "integrity": "sha512-D46sfMTngaYlrH+OspKf8mIJETntFnf6Hsjb0V41jAXJ7Bx2kB8Rv8RCUujuVWYttFtHkUNp7g+FwxNQAr6mXA==", + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.5.7.tgz", + "integrity": "sha512-BRFORjsTuQv5gxcXsuDXx6oGRhuVsEGwZy6LOzRRfgu+eSfxbhUQ9L9YtSEIuIjY/o7g3iWFjrc5eSY1GXP2Dw==", "dependencies": { "tslib": "^2.3.0" }, @@ -8403,9 +8373,9 @@ } }, "node_modules/clsx": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", - "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==", + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", "engines": { "node": ">=6" } @@ -9885,7 +9855,8 @@ "version": "0.5.16", "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", - "dev": true + "dev": true, + "peer": true }, "node_modules/dom-converter": { "version": "0.2.0", @@ -16844,6 +16815,7 @@ "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", "dev": true, + "peer": true, "bin": { "lz-string": "bin/bin.js" } @@ -20238,12 +20210,11 @@ } }, "node_modules/react": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react/-/react-17.0.2.tgz", - "integrity": "sha512-gnhPt75i/dq/z3/6q/0asP78D0u592D5L1pd7M8P+dck6Fu/jJeL6iVVK23fptSUZj8Vjf++7wXA8UNclGQcbA==", + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", + "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", "dependencies": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" }, "engines": { "node": ">=0.10.0" @@ -20362,14 +20333,6 @@ "react-dom": "^16.9.0 || ^17 || ^18" } }, - "node_modules/react-datepicker/node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "engines": { - "node": ">=6" - } - }, "node_modules/react-datepicker/node_modules/date-fns": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-3.6.0.tgz", @@ -20486,16 +20449,15 @@ } }, "node_modules/react-dom": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-17.0.2.tgz", - "integrity": "sha512-s4h96KtLDUQlsENhMn1ar8t2bEa+q/YAtj8pPPdIjPDGBDIVNsrD9aXNWqspUe6AzKCIG0C1HZZLqLV7qpOBGA==", + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", + "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", "dependencies": { "loose-envify": "^1.1.0", - "object-assign": "^4.1.1", - "scheduler": "^0.20.2" + "scheduler": "^0.23.2" }, "peerDependencies": { - "react": "17.0.2" + "react": "^18.3.1" } }, "node_modules/react-error-overlay": { @@ -20580,9 +20542,9 @@ } }, "node_modules/react-onclickoutside": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/react-onclickoutside/-/react-onclickoutside-6.13.0.tgz", - "integrity": "sha512-ty8So6tcUpIb+ZE+1HAhbLROvAIJYyJe/1vRrrcmW+jLsaM+/powDRqxzo6hSh9CuRZGSL1Q8mvcF5WRD93a0A==", + "version": "6.13.1", + "resolved": "https://registry.npmjs.org/react-onclickoutside/-/react-onclickoutside-6.13.1.tgz", + "integrity": "sha512-LdrrxK/Yh9zbBQdFbMTXPp3dTSN9B+9YJQucdDu3JNKRrbdU+H+/TVONJoWtOwy4II8Sqf1y/DTI6w/vGPYW0w==", "funding": { "type": "individual", "url": "https://github.com/Pomax/react-onclickoutside/blob/master/FUNDING.md" @@ -20823,15 +20785,15 @@ } }, "node_modules/react-toastify": { - "version": "9.1.3", - "resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-9.1.3.tgz", - "integrity": "sha512-fPfb8ghtn/XMxw3LkxQBk3IyagNpF/LIKjOBflbexr2AWxAH1MJgvnESwEwBn9liLFXgTKWgBSdZpw9m4OTHTg==", + "version": "10.0.5", + "resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-10.0.5.tgz", + "integrity": "sha512-mNKt2jBXJg4O7pSdbNUfDdTsK9FIdikfsIE/yUCxbAEXl4HMyJaivrVFcn3Elvt5xvCQYhUZm+hqTIu1UXM3Pw==", "dependencies": { - "clsx": "^1.1.1" + "clsx": "^2.1.0" }, "peerDependencies": { - "react": ">=16", - "react-dom": ">=16" + "react": ">=18", + "react-dom": ">=18" } }, "node_modules/react-tooltip": { @@ -22004,12 +21966,11 @@ } }, "node_modules/scheduler": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.20.2.tgz", - "integrity": "sha512-2eWfGgAqqWFGqtdMmcL5zCMK1U8KlXv8SQFGglL3CEtd0aDVDWgeF/YoCmvln55m5zSk3J/20hTaSBeSObsQDQ==", + "version": "0.23.2", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", + "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", "dependencies": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" } }, "node_modules/schema-utils": { @@ -24068,9 +24029,9 @@ } }, "node_modules/undici-types": { - "version": "5.26.5", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", - "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==" + "version": "6.19.8", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==" }, "node_modules/unicode-canonical-property-names-ecmascript": { "version": "2.0.0", @@ -26755,9 +26716,9 @@ "integrity": "sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==" }, "@babel/runtime": { - "version": "7.25.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.0.tgz", - "integrity": "sha512-7dRy4DwXwtzBrPbZflqxnvfxLF8kdZXPkhymtDeFoFqE6ldzjQFgYTtYIFARcLEYDrqfBfYcZt1WqFxRoyC9Rw==", + "version": "7.25.6", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.6.tgz", + "integrity": "sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ==", "requires": { "regenerator-runtime": "^0.14.0" }, @@ -26769,16 +26730,6 @@ } } }, - "@babel/runtime-corejs3": { - "version": "7.22.6", - "resolved": "https://registry.npmjs.org/@babel/runtime-corejs3/-/runtime-corejs3-7.22.6.tgz", - "integrity": "sha512-M+37LLIRBTEVjktoJjbw4KVhupF0U/3PYUCbBwgAd9k17hoKhRu1n935QiG7Tuxv0LJOMrb2vuKEeYUlv0iyiw==", - "dev": true, - "requires": { - "core-js-pure": "^3.30.2", - "regenerator-runtime": "^0.13.11" - } - }, "@babel/template": { "version": "7.23.9", "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.23.9.tgz", @@ -27539,12 +27490,12 @@ } }, "@floating-ui/react": { - "version": "0.26.19", - "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.19.tgz", - "integrity": "sha512-Jk6zITdjjIvjO/VdQFvpRaD3qPwOHH6AoDHxjhpy+oK4KFgaSP871HYWUAPdnLmx1gQ+w/pB312co3tVml+BXA==", + "version": "0.26.22", + "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.22.tgz", + "integrity": "sha512-LNv4azPt8SpT4WW7Kku5JNVjLk2GcS0bGGjFTAgqOONRFo9r/aaGHHPpdiIuQbB1t8shmWyWqTTUDmZ9fcNshg==", "requires": { "@floating-ui/react-dom": "^2.1.1", - "@floating-ui/utils": "^0.2.4", + "@floating-ui/utils": "^0.2.7", "tabbable": "^6.0.0" } }, @@ -27557,9 +27508,9 @@ } }, "@floating-ui/utils": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.4.tgz", - "integrity": "sha512-dWO2pw8hhi+WrXq1YJy2yCuWoL20PddgGaqTgVe4cOS9Q6qklXCiA1tJEqX6BEwRNSCP84/afac9hd4MS+zEUA==" + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.7.tgz", + "integrity": "sha512-X8R8Oj771YRl/w+c1HqAC1szL8zWQRwFvgDwT129k9ACdBoud/+/rX9V0qiMl6LWUdP9voC2nDVZYPMQQsb6eA==" }, "@gar/promisify": { "version": "1.1.3", @@ -27969,13 +27920,6 @@ "@popperjs/core": "^2.11.8", "clsx": "^2.1.0", "prop-types": "^15.8.1" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/core-downloads-tracker": { @@ -27984,9 +27928,9 @@ "integrity": "sha512-RtsCt4Geed2/v74sbihWzzRs+HsIQCfclHeORh5Ynu2fS4icIKozcSubwuG7vtzq2uW3fOR1zITSP84TNt2GoQ==" }, "@mui/icons-material": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-5.16.1.tgz", - "integrity": "sha512-ogQPweYba4+5XZykilwxn2/oS78uwoQ0BVBpOhhCJo0ooZsqTTsalhzP2qD/RdGqMQ8xyXPz1sYM2djTruVVVA==", + "version": "5.16.7", + "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-5.16.7.tgz", + "integrity": "sha512-UrGwDJCXEszbDI7yV047BYU5A28eGJ79keTCP4cc74WyncuVrnurlmIRxaHL8YK+LI1Kzq+/JM52IAkNnv4u+Q==", "requires": { "@babel/runtime": "^7.23.9" } @@ -28008,13 +27952,6 @@ "prop-types": "^15.8.1", "react-is": "^18.3.1", "react-transition-group": "^4.4.5" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/private-theming": { @@ -28044,20 +27981,13 @@ "integrity": "sha512-Jncvs/r/d/itkxh7O7opOunTqbbSSzMTHzZkNLM+FjAOg+cYAZHrPDlYe1ZGKUYORwwb2XexlWnpZp0kZ4AHuA==", "requires": { "@babel/runtime": "^7.23.9", - "@mui/private-theming": "^5.16.6", - "@mui/styled-engine": "^5.16.6", + "@mui/private-theming": "^5.16.4", + "@mui/styled-engine": "^5.16.4", "@mui/types": "^7.2.15", - "@mui/utils": "^5.16.6", + "@mui/utils": "^5.16.4", "clsx": "^2.1.0", "csstype": "^3.1.3", "prop-types": "^15.8.1" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/types": { @@ -28077,13 +28007,6 @@ "clsx": "^2.1.1", "prop-types": "^15.8.1", "react-is": "^18.3.1" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/x-charts": { @@ -28104,24 +28027,16 @@ "d3-scale": "^4.0.2", "d3-shape": "^3.2.0", "prop-types": "^15.8.1" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/x-data-grid": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@mui/x-data-grid/-/x-data-grid-7.11.0.tgz", - "integrity": "sha512-dXaIw3Noxc4d6xenS7J+zMPORG9ptkTW7B81P6QFovILSEuI/qebQhijy/IkqRvcCsuZYLL3nA89bp+EDI503Q==", - "requires": { - "@babel/runtime": "^7.24.8", - "@mui/system": "^5.16.2", - "@mui/utils": "^5.16.2", - "@mui/x-internals": "7.11.0", + "version": "7.15.0", + "resolved": "https://registry.npmjs.org/@mui/x-data-grid/-/x-data-grid-7.15.0.tgz", + "integrity": "sha512-S7FLp+SUz/fo+JAQcM4npnL70tGnL4qcYYMkjK4tPkPLrTnBWlm47D5OoGNzpPO1CnOAryFR5h0rdwbeHZmAWg==", + "requires": { + "@babel/runtime": "^7.25.4", + "@mui/utils": "^5.16.6", + "@mui/x-internals": "7.15.0", "clsx": "^2.1.1", "prop-types": "^15.8.1", "reselect": "^4.1.8" @@ -28139,29 +28054,23 @@ "resolved": "https://registry.npmjs.org/@mui/x-date-pickers/-/x-date-pickers-7.12.1.tgz", "integrity": "sha512-Zj8kt3SCQbJp1qhMi+A3I4KqB8i5OY2Q11mdOEathFhqN/SQm1sUjIa1G09cGP1dPDgK1a6KM6qJGNtcw/nuWA==", "requires": { - "@babel/runtime": "^7.25.0", - "@mui/system": "^5.16.5", - "@mui/utils": "^5.16.5", + "@babel/runtime": "^7.24.6", + "@mui/base": "^5.0.0-beta.40", + "@mui/system": "^5.15.15", + "@mui/utils": "^5.15.14", "@types/react-transition-group": "^4.4.10", "clsx": "^2.1.1", "prop-types": "^15.8.1", "react-transition-group": "^4.4.5" - }, - "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - } } }, "@mui/x-internals": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.11.0.tgz", - "integrity": "sha512-GqCYylKiB4cLH9tK4JweJlT2JvPjnpXjS3TEIqtHB4BcSsezhdRrMGzHOO5zCJqkasqTirJh2t6X16Qw1llr4Q==", + "version": "7.15.0", + "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.15.0.tgz", + "integrity": "sha512-Q/IZvZhHpe64Ost1mRbdp6ML8KQQBprwwgzqo6pZbrCaWMPB2gk2jcUwdCwnLsc+gutaEPVhZ8N7it8VZcHtbg==", "requires": { - "@babel/runtime": "^7.24.8", - "@mui/utils": "^5.16.2" + "@babel/runtime": "^7.25.4", + "@mui/utils": "^5.16.6" } }, "@nicolo-ribaudo/eslint-scope-5-internals": { @@ -28353,51 +28262,52 @@ } }, "@react-spring/animated": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/animated/-/animated-9.7.3.tgz", - "integrity": "sha512-5CWeNJt9pNgyvuSzQH+uy2pvTg8Y4/OisoscZIR8/ZNLIOI+CatFBhGZpDGTF/OzdNFsAoGk3wiUYTwoJ0YIvw==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/animated/-/animated-9.7.4.tgz", + "integrity": "sha512-7As+8Pty2QlemJ9O5ecsuPKjmO0NKvmVkRR1n6mEotFgWar8FKuQt2xgxz3RTgxcccghpx1YdS1FCdElQNexmQ==", "requires": { - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" } }, "@react-spring/core": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/core/-/core-9.7.3.tgz", - "integrity": "sha512-IqFdPVf3ZOC1Cx7+M0cXf4odNLxDC+n7IN3MDcVCTIOSBfqEcBebSv+vlY5AhM0zw05PDbjKrNmBpzv/AqpjnQ==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/core/-/core-9.7.4.tgz", + "integrity": "sha512-GzjA44niEJBFUe9jN3zubRDDDP2E4tBlhNlSIkTChiNf9p4ZQlgXBg50qbXfSXHQPHak/ExYxwhipKVsQ/sUTw==", "requires": { - "@react-spring/animated": "~9.7.3", - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/animated": "~9.7.4", + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" } }, "@react-spring/rafz": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/rafz/-/rafz-9.7.3.tgz", - "integrity": "sha512-9vzW1zJPcC4nS3aCV+GgcsK/WLaB520Iyvm55ARHfM5AuyBqycjvh1wbmWmgCyJuX4VPoWigzemq1CaaeRSHhQ==" + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/rafz/-/rafz-9.7.4.tgz", + "integrity": "sha512-mqDI6rW0Ca8IdryOMiXRhMtVGiEGLIO89vIOyFQXRIwwIMX30HLya24g9z4olDvFyeDW3+kibiKwtZnA4xhldA==" }, "@react-spring/shared": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/shared/-/shared-9.7.3.tgz", - "integrity": "sha512-NEopD+9S5xYyQ0pGtioacLhL2luflh6HACSSDUZOwLHoxA5eku1UPuqcJqjwSD6luKjjLfiLOspxo43FUHKKSA==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/shared/-/shared-9.7.4.tgz", + "integrity": "sha512-bEPI7cQp94dOtCFSEYpxvLxj0+xQfB5r9Ru1h8OMycsIq7zFZon1G0sHrBLaLQIWeMCllc4tVDYRTLIRv70C8w==", "requires": { - "@react-spring/types": "~9.7.3" + "@react-spring/rafz": "~9.7.4", + "@react-spring/types": "~9.7.4" } }, "@react-spring/types": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/types/-/types-9.7.3.tgz", - "integrity": "sha512-Kpx/fQ/ZFX31OtlqVEFfgaD1ACzul4NksrvIgYfIFq9JpDHFwQkMVZ10tbo0FU/grje4rcL4EIrjekl3kYwgWw==" + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/types/-/types-9.7.4.tgz", + "integrity": "sha512-iQVztO09ZVfsletMiY+DpT/JRiBntdsdJ4uqk3UJFhrhS8mIC9ZOZbmfGSRs/kdbNPQkVyzucceDicQ/3Mlj9g==" }, "@react-spring/web": { - "version": "9.7.3", - "resolved": "https://registry.npmjs.org/@react-spring/web/-/web-9.7.3.tgz", - "integrity": "sha512-BXt6BpS9aJL/QdVqEIX9YoUy8CE6TJrU0mNCqSoxdXlIeNcEBWOfIyE6B14ENNsyQKS3wOWkiJfco0tCr/9tUg==", + "version": "9.7.4", + "resolved": "https://registry.npmjs.org/@react-spring/web/-/web-9.7.4.tgz", + "integrity": "sha512-UMvCZp7I5HCVIleSa4BwbNxynqvj+mJjG2m20VO2yPoi2pnCYANy58flvz9v/YcXTAvsmL655FV3pm5fbr6akA==", "requires": { - "@react-spring/animated": "~9.7.3", - "@react-spring/core": "~9.7.3", - "@react-spring/shared": "~9.7.3", - "@react-spring/types": "~9.7.3" + "@react-spring/animated": "~9.7.4", + "@react-spring/core": "~9.7.4", + "@react-spring/shared": "~9.7.4", + "@react-spring/types": "~9.7.4" } }, "@remix-run/router": { @@ -28796,51 +28706,70 @@ } }, "@testing-library/dom": { - "version": "7.31.2", - "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-7.31.2.tgz", - "integrity": "sha512-3UqjCpey6HiTZT92vODYLPxTBWlM8ZOOjr3LX5F37/VRipW2M1kX6I/Cm4VXzteZqfGfagg8yXywpcOgQBlNsQ==", + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.0.tgz", + "integrity": "sha512-pemlzrSESWbdAloYml3bAJMEfNh1Z7EduzqPKprCH5S341frlpYnUEW0H72dLxa6IsYr+mPno20GiSm+h9dEdQ==", "dev": true, + "peer": true, "requires": { "@babel/code-frame": "^7.10.4", "@babel/runtime": "^7.12.5", - "@types/aria-query": "^4.2.0", - "aria-query": "^4.2.2", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", "chalk": "^4.1.0", - "dom-accessibility-api": "^0.5.6", - "lz-string": "^1.4.4", - "pretty-format": "^26.6.2" + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "pretty-format": "^27.0.2" }, "dependencies": { - "aria-query": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-4.2.2.tgz", - "integrity": "sha512-o/HelwhuKpTj/frsOsbNLNgnNGVIFsVP/SW2BSF14gVl7kAfMOJ6/8wUAUvG1R1NHKrfG+2sHZTu0yauT1qBrA==", - "dev": true, - "requires": { - "@babel/runtime": "^7.10.2", - "@babel/runtime-corejs3": "^7.10.2" - } - }, "chalk": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, + "peer": true, "requires": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } + }, + "pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "peer": true, + "requires": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "dependencies": { + "ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "peer": true + } + } + }, + "react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "peer": true } } }, "@testing-library/jest-dom": { - "version": "6.4.8", - "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.4.8.tgz", - "integrity": "sha512-JD0G+Zc38f5MBHA4NgxQMR5XtO5Jx9g86jqturNTt2WUfRmLDIY7iKkWHDCCTiDuFMre6nxAD5wHw9W5kI4rGw==", + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.5.0.tgz", + "integrity": "sha512-xGGHpBXYSHUUr6XsKBfs85TWlYKpTc37cSBBVrXcib2MkHLboWlkClhWF37JKlDb9KEq3dHs+f2xR7XJEWGBxA==", "dev": true, "requires": { "@adobe/css-tools": "^4.4.0", - "@babel/runtime": "^7.9.2", "aria-query": "^5.0.0", "chalk": "^3.0.0", "css.escape": "^1.5.1", @@ -28858,13 +28787,12 @@ } }, "@testing-library/react": { - "version": "11.2.7", - "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-11.2.7.tgz", - "integrity": "sha512-tzRNp7pzd5QmbtXNG/mhdcl7Awfu/Iz1RaVHY75zTdOkmHCuzMhRL83gWHSgOAcjS3CCbyfwUHMZgRJb4kAfpA==", + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-16.0.0.tgz", + "integrity": "sha512-guuxUKRWQ+FgNX0h0NS0FIq3Q3uLtWVpBzcLOggmfMoUpgBnzBzvLLd4fbm6yS8ydJd94cIfY4yP9qUQjM2KwQ==", "dev": true, "requires": { - "@babel/runtime": "^7.12.5", - "@testing-library/dom": "^7.28.1" + "@babel/runtime": "^7.12.5" } }, "@testing-library/user-event": { @@ -28887,10 +28815,11 @@ "integrity": "sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==" }, "@types/aria-query": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-4.2.2.tgz", - "integrity": "sha512-HnYpAE1Y6kRyKM/XkEuiRQhTHvkzMBurTHnpFLYLBGPIylZNPs9jJcuOOYWxPLJCSEtmZT0Y8rHDokKN7rRTig==", - "dev": true + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "peer": true }, "@types/babel__core": { "version": "7.20.1", @@ -29128,11 +29057,11 @@ "peer": true }, "@types/node": { - "version": "20.12.12", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.12.tgz", - "integrity": "sha512-eWLDGF/FOSPtAvEqeRAQ4C8LSA7M1I7i0ky1I8U7kD1J5ITyW3AsRhQrKVoWf5pFKZ2kILsEGJhsI9r93PYnOw==", + "version": "22.5.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.2.tgz", + "integrity": "sha512-acJsPTEqYqulZS/Yp/S3GgeE6GZ0qYODUR8aVr/DkhHQ8l9nd4j5x1/ZJy9/gHrRlFMqkO6i0I3E27Alu4jjPg==", "requires": { - "undici-types": "~5.26.4" + "undici-types": "~6.19.2" } }, "@types/node-fetch": { @@ -29196,12 +29125,11 @@ "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==" }, "@types/react": { - "version": "17.0.62", - "resolved": "https://registry.npmjs.org/@types/react/-/react-17.0.62.tgz", - "integrity": "sha512-eANCyz9DG8p/Vdhr0ZKST8JV12PhH2ACCDYlFw6DIO+D+ca+uP4jtEDEpVqXZrh/uZdXQGwk7whJa3ah5DtyLw==", + "version": "18.3.3", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.3.tgz", + "integrity": "sha512-hti/R0pS0q1/xx+TsI73XIqk26eBsISZ2R0wUijXIngRK9R/e7Xw/cXVxQK7R5JjW+SV4zGcn5hXjudkN/pLIw==", "requires": { "@types/prop-types": "*", - "@types/scheduler": "*", "csstype": "^3.0.2" } }, @@ -29236,12 +29164,12 @@ } }, "@types/react-dom": { - "version": "17.0.20", - "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-17.0.20.tgz", - "integrity": "sha512-4pzIjSxDueZZ90F52mU3aPoogkHIoSIDG+oQ+wQK7Cy2B9S+MvOqY0uEA/qawKz381qrEDkvpwyt8Bm31I8sbA==", + "version": "18.3.0", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.0.tgz", + "integrity": "sha512-EhwApuTmMBmXuFOikhQLIBUn6uFg81SwLMOAUgodJF14SOBOCMdU04gDoYi0WOJJHD144TL32z4yDqCW3dnkQg==", "dev": true, "requires": { - "@types/react": "^17" + "@types/react": "*" } }, "@types/react-google-recaptcha": { @@ -29315,11 +29243,6 @@ "htmlparser2": "^8.0.0" } }, - "@types/scheduler": { - "version": "0.16.3", - "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.3.tgz", - "integrity": "sha512-5cJ8CB4yAx7BH1oMvdU0Jh9lrEXyPkar6F9G/ERswkCuvP4KQZfZkSjcMbAICCpQTN4OuZn8tz0HiKv9TGZgrQ==" - }, "@types/semver": { "version": "7.5.0", "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.0.tgz", @@ -29658,9 +29581,9 @@ } }, "@wry/equality": { - "version": "0.5.6", - "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.5.6.tgz", - "integrity": "sha512-D46sfMTngaYlrH+OspKf8mIJETntFnf6Hsjb0V41jAXJ7Bx2kB8Rv8RCUujuVWYttFtHkUNp7g+FwxNQAr6mXA==", + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@wry/equality/-/equality-0.5.7.tgz", + "integrity": "sha512-BRFORjsTuQv5gxcXsuDXx6oGRhuVsEGwZy6LOzRRfgu+eSfxbhUQ9L9YtSEIuIjY/o7g3iWFjrc5eSY1GXP2Dw==", "requires": { "tslib": "^2.3.0" } @@ -30908,9 +30831,9 @@ } }, "clsx": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", - "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==" + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" }, "co": { "version": "4.6.0", @@ -32008,7 +31931,8 @@ "version": "0.5.16", "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", - "dev": true + "dev": true, + "peer": true }, "dom-converter": { "version": "0.2.0", @@ -37107,7 +37031,8 @@ "version": "1.5.0", "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", - "dev": true + "dev": true, + "peer": true }, "magic-string": { "version": "0.25.9", @@ -39451,12 +39376,11 @@ } }, "react": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react/-/react-17.0.2.tgz", - "integrity": "sha512-gnhPt75i/dq/z3/6q/0asP78D0u592D5L1pd7M8P+dck6Fu/jJeL6iVVK23fptSUZj8Vjf++7wXA8UNclGQcbA==", + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", + "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", "requires": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" } }, "react-app-polyfill": { @@ -39541,11 +39465,6 @@ "react-onclickoutside": "^6.13.0" }, "dependencies": { - "clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" - }, "date-fns": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-3.6.0.tgz", @@ -39629,13 +39548,12 @@ } }, "react-dom": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-17.0.2.tgz", - "integrity": "sha512-s4h96KtLDUQlsENhMn1ar8t2bEa+q/YAtj8pPPdIjPDGBDIVNsrD9aXNWqspUe6AzKCIG0C1HZZLqLV7qpOBGA==", + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", + "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", "requires": { "loose-envify": "^1.1.0", - "object-assign": "^4.1.1", - "scheduler": "^0.20.2" + "scheduler": "^0.23.2" } }, "react-error-overlay": { @@ -39697,9 +39615,9 @@ "integrity": "sha512-C5DAvJkfzR2JK9YixZ3oyF9x6R4LW6nzTpIXrl9Oujxi4uqP9SzVVCjl+JLM3tSdqdjAx/oWZK3dTVBSR73Q+w==" }, "react-onclickoutside": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/react-onclickoutside/-/react-onclickoutside-6.13.0.tgz", - "integrity": "sha512-ty8So6tcUpIb+ZE+1HAhbLROvAIJYyJe/1vRrrcmW+jLsaM+/powDRqxzo6hSh9CuRZGSL1Q8mvcF5WRD93a0A==", + "version": "6.13.1", + "resolved": "https://registry.npmjs.org/react-onclickoutside/-/react-onclickoutside-6.13.1.tgz", + "integrity": "sha512-LdrrxK/Yh9zbBQdFbMTXPp3dTSN9B+9YJQucdDu3JNKRrbdU+H+/TVONJoWtOwy4II8Sqf1y/DTI6w/vGPYW0w==", "requires": {} }, "react-popper": { @@ -39861,11 +39779,11 @@ } }, "react-toastify": { - "version": "9.1.3", - "resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-9.1.3.tgz", - "integrity": "sha512-fPfb8ghtn/XMxw3LkxQBk3IyagNpF/LIKjOBflbexr2AWxAH1MJgvnESwEwBn9liLFXgTKWgBSdZpw9m4OTHTg==", + "version": "10.0.5", + "resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-10.0.5.tgz", + "integrity": "sha512-mNKt2jBXJg4O7pSdbNUfDdTsK9FIdikfsIE/yUCxbAEXl4HMyJaivrVFcn3Elvt5xvCQYhUZm+hqTIu1UXM3Pw==", "requires": { - "clsx": "^1.1.1" + "clsx": "^2.1.0" } }, "react-tooltip": { @@ -40712,12 +40630,11 @@ } }, "scheduler": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.20.2.tgz", - "integrity": "sha512-2eWfGgAqqWFGqtdMmcL5zCMK1U8KlXv8SQFGglL3CEtd0aDVDWgeF/YoCmvln55m5zSk3J/20hTaSBeSObsQDQ==", + "version": "0.23.2", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", + "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", "requires": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" } }, "schema-utils": { @@ -42317,9 +42234,9 @@ } }, "undici-types": { - "version": "5.26.5", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", - "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==" + "version": "6.19.8", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==" }, "unicode-canonical-property-names-ecmascript": { "version": "2.0.0", diff --git a/package.json b/package.json index f9d29778a5..74462f25bc 100644 --- a/package.json +++ b/package.json @@ -12,12 +12,12 @@ "@dicebear/core": "^8.0.2", "@emotion/react": "^11.13.0", "@emotion/styled": "^11.13.0", - "@mui/icons-material": "^5.16.1", + "@mui/icons-material": "^5.16.7", "@mui/material": "^5.16.7", "@mui/private-theming": "^5.15.12", "@mui/system": "^5.14.12", "@mui/x-charts": "^7.8.0", - "@mui/x-data-grid": "^7.11.0", + "@mui/x-data-grid": "^7.15.0", "@mui/x-date-pickers": "^7.11.1", "@pdfme/generator": "^1.2.6", "bootstrap": "^5.3.3", @@ -35,12 +35,12 @@ "js-cookie": "^3.0.1", "markdown-toc": "^1.2.0", "prettier": "^3.3.2", - "react": "^17.0.2", + "react": "^18.3.1", "react-app-rewired": "^2.2.1", "react-beautiful-dnd": "^13.1.1", "react-bootstrap": "^2.10.4", "react-datepicker": "^7.3.0", - "react-dom": "^17.0.2", + "react-dom": "^18.3.1", "react-google-recaptcha": "^3.1.0", "react-i18next": "^12.3.1", "react-icons": "^5.2.1", @@ -49,7 +49,7 @@ "react-redux": "^7.2.5", "react-router-dom": "^6.26.0", "react-scripts": "5.0.1", - "react-toastify": "^9.0.3", + "react-toastify": "^10.0.5", "react-tooltip": "^5.27.1", "redux": "^4.1.1", "redux-thunk": "^2.3.0", @@ -96,19 +96,19 @@ }, "devDependencies": { "@babel/plugin-proposal-private-property-in-object": "^7.21.11", - "@testing-library/jest-dom": "^6.4.2", - "@testing-library/react": "^11.1.0", + "@testing-library/jest-dom": "^6.5.0", + "@testing-library/react": "^16.0.0", "@testing-library/user-event": "^12.1.10", "@types/inquirer": "^9.0.7", "@types/jest": "^26.0.24", "@types/js-cookie": "^3.0.6", - "@types/node": "^20.12.12", + "@types/node": "^22.5.2", "@types/node-fetch": "^2.6.10", - "@types/react": "^17.0.14", + "@types/react": "^18.3.3", "@types/react-beautiful-dnd": "^13.1.8", "@types/react-bootstrap": "^0.32.32", "@types/react-datepicker": "^4.1.4", - "@types/react-dom": "^17.0.9", + "@types/react-dom": "^18.3.0", "@types/react-google-recaptcha": "^2.1.5", "@types/react-router-dom": "^5.1.8", "@types/sanitize-html": "^2.13.0", diff --git a/public/locales/en/translation.json b/public/locales/en/translation.json index 3b7b875821..112eae5892 100644 --- a/public/locales/en/translation.json +++ b/public/locales/en/translation.json @@ -854,7 +854,10 @@ "yourPreviousDonations": "Your Previous Donations", "donate": "Donate", "nothingToShow": "Nothing to show here.", - "success": "Donation Successful" + "success": "Donation Successful", + "invalidAmount": "Please enter a numerical value for the donation amount.", + "donationAmountDescription": "Please enter the numerical value for the donation amount.", + "donationOutOfRange": "Donation amount must be between {{min}} and {{max}}." }, "userEvents": { "title": "Events", diff --git a/public/locales/fr/translation.json b/public/locales/fr/translation.json index ac72be3def..1498607e51 100644 --- a/public/locales/fr/translation.json +++ b/public/locales/fr/translation.json @@ -854,7 +854,10 @@ "yourPreviousDonations": "Vos dons précédents", "donate": "Faire un don", "nothingToShow": "Rien à montrer ici.", - "success": "Don réussi" + "success": "Don réussi", + "invalidAmount": "Veuillez saisir une valeur numérique pour le montant du don.", + "donationAmountDescription": "Veuillez saisir la valeur numérique du montant du don.", + "donationOutOfRange": "Le montant du don doit être compris entre {{min}} et {{max}}." }, "userEvents": { "title": "Événements", diff --git a/public/locales/hi/translation.json b/public/locales/hi/translation.json index 0f74a5978e..0de147b21a 100644 --- a/public/locales/hi/translation.json +++ b/public/locales/hi/translation.json @@ -854,7 +854,10 @@ "yourPreviousDonations": "आपका पिछला दान", "donate": "दान करें", "nothingToShow": "यहां दिखाने के लिए कुछ भी नहीं है.", - "success": "दान सफल" + "success": "दान सफल", + "invalidAmount": "कृपया दान राशि के लिए संख्यात्मक मान दर्ज करें.", + "donationAmountDescription": "कृपया दान राशि के लिए संख्यात्मक मान दर्ज करें.", + "donationOutOfRange": "दान राशि {{min}} और {{max}} के बीच होनी चाहिए." }, "userEvents": { "title": "ईवेंट", diff --git a/public/locales/sp/translation.json b/public/locales/sp/translation.json index 5d32d79368..d68701c261 100644 --- a/public/locales/sp/translation.json +++ b/public/locales/sp/translation.json @@ -1063,7 +1063,10 @@ "yourPreviousDonations": "Tus donaciones anteriores", "donate": "Donar", "nothingToShow": "Nada que mostrar aquí.", - "success": "Donación exitosa" + "success": "Donación exitosa", + "invalidAmount": "Ingrese un valor numérico para el monto de la donación.", + "donationAmountDescription": "Ingrese el valor numérico del monto de la donación.", + "donationOutOfRange": "El monto de la donación debe estar entre {{min}} y {{max}}." }, "userEvents": { "title": "Eventos", diff --git a/public/locales/zh/translation.json b/public/locales/zh/translation.json index 56d93b13da..074c03dc0e 100644 --- a/public/locales/zh/translation.json +++ b/public/locales/zh/translation.json @@ -854,7 +854,10 @@ "yourPreviousDonations": "您之前的捐款", "donate": "捐", "nothingToShow": "这里没有什么可显示的。", - "success": "捐赠成功" + "success": "捐赠成功", + "invalidAmount": "请输入捐赠金额的数值。", + "donationAmountDescription": "请输入捐款金额的数值。", + "donationOutOfRange": "捐款金额必须在 {{min}} 和 {{max}} 之间。" }, "userEvents": { "title": "活动", diff --git a/src/App.test.tsx b/src/App.test.tsx index 3146066e26..f4fba2ebf8 100644 --- a/src/App.test.tsx +++ b/src/App.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { Provider } from 'react-redux'; import { MockedProvider } from '@apollo/react-testing'; import { BrowserRouter } from 'react-router-dom'; diff --git a/src/App.tsx b/src/App.tsx index f73055f4c8..df97e649d9 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -122,12 +122,12 @@ function app(): JSX.Element { ], index: number, ) => { - const extraComponent = plugin[1]; + const ExtraComponent = plugin[1]; return ( } /> ); }, diff --git a/src/components/ActionItems/ActionItemsContainer.tsx b/src/components/ActionItems/ActionItemsContainer.tsx index 4b393514e6..fcee3b0083 100644 --- a/src/components/ActionItems/ActionItemsContainer.tsx +++ b/src/components/ActionItems/ActionItemsContainer.tsx @@ -159,7 +159,7 @@ function actionItemsContainer({ actionItemsRefetch(); hideUpdateModal(); - toast.success(t('successfulUpdation')); + toast.success(t('successfulUpdation') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -183,7 +183,7 @@ function actionItemsContainer({ actionItemsRefetch(); toggleDeleteModal(); - toast.success(t('successfulDeletion')); + toast.success(t('successfulDeletion') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); diff --git a/src/components/ActionItems/ActionItemsModalBody.tsx b/src/components/ActionItems/ActionItemsModalBody.tsx index a94901d808..860dfbf6c3 100644 --- a/src/components/ActionItems/ActionItemsModalBody.tsx +++ b/src/components/ActionItems/ActionItemsModalBody.tsx @@ -147,7 +147,7 @@ export const ActionItemsModalBody = ({ // Refetching the action items list to update the UI actionItemsRefetch(); hideCreateModal(); - toast.success(t('successfulCreation')); + toast.success(t('successfulCreation') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); diff --git a/src/components/ActionItems/ActionItemsWrapper.test.tsx b/src/components/ActionItems/ActionItemsWrapper.test.tsx index 7b27e7ccd3..ad3cdc8687 100644 --- a/src/components/ActionItems/ActionItemsWrapper.test.tsx +++ b/src/components/ActionItems/ActionItemsWrapper.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen, waitFor } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen, waitFor } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { ActionItemsWrapper } from './ActionItemsWrapper'; import { BrowserRouter } from 'react-router-dom'; diff --git a/src/components/AddOn/AddOn.test.tsx b/src/components/AddOn/AddOn.test.tsx index 8313fefcb1..68475e8196 100644 --- a/src/components/AddOn/AddOn.test.tsx +++ b/src/components/AddOn/AddOn.test.tsx @@ -15,6 +15,14 @@ describe('Testing Addon component', () => { children: 'This is a dummy text', }; + test('should render with default props', () => { + const { getByTestId } = render(); + const container = getByTestId('pluginContainer'); + expect(container).toBeInTheDocument(); + expect(container).toHaveClass('plugin-container'); + expect(container).toHaveTextContent('Default text'); + }); + test('should render props and text elements test for the page component', () => { const { getByTestId, getByText } = render( diff --git a/src/components/AddOn/AddOn.tsx b/src/components/AddOn/AddOn.tsx index 1d617535b1..850cc05ee4 100644 --- a/src/components/AddOn/AddOn.tsx +++ b/src/components/AddOn/AddOn.tsx @@ -2,9 +2,9 @@ import React from 'react'; import PropTypes from 'prop-types'; interface InterfaceAddOnProps { - extras: any; - name: string; - children: any; + extras?: any; + name?: string; + children?: React.ReactNode; } /** @@ -19,31 +19,26 @@ interface InterfaceAddOnProps { * * @returns The JSX element representing the AddOn component. */ -function addOn({ children }: InterfaceAddOnProps): JSX.Element { +function AddOn({ + children = 'Default text', + extras = {}, + name = '', +}: InterfaceAddOnProps): JSX.Element { return ( - <> -
- {children} -
- +
+ {children} +
); } -// Default props for the AddOn component -addOn.defaultProps = { - extras: {}, - name: '', - children: null, -}; - // PropTypes validation for the AddOn component -addOn.propTypes = { +AddOn.propTypes = { extras: PropTypes.shape({ components: PropTypes.shape({}), actions: PropTypes.shape({}), }), name: PropTypes.string, - children: PropTypes.any, + children: PropTypes.node, }; -export default addOn; +export default AddOn; diff --git a/src/components/AddOn/core/AddOnEntry/AddOnEntry.test.tsx b/src/components/AddOn/core/AddOnEntry/AddOnEntry.test.tsx index c09840a8f0..3d800eb59f 100644 --- a/src/components/AddOn/core/AddOnEntry/AddOnEntry.test.tsx +++ b/src/components/AddOn/core/AddOnEntry/AddOnEntry.test.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { render } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { BrowserRouter } from 'react-router-dom'; import AddOnEntry from './AddOnEntry'; import { @@ -76,6 +76,32 @@ describe('Testing AddOnEntry', () => { expect(getByTestId('AddOnEntry')).toBeInTheDocument(); }); + test('uses default values for title and description when not provided', () => { + // Render the component with only required parameters + const mockGetInstalledPlugins = jest.fn(); + render( + + + + + + + + + , + ); + + const titleElement = screen.getByText('No title provided'); // This will check for the default empty string in the title + const descriptionElement = screen.getByText('Description not available'); // This will check for the default empty string in the description + expect(titleElement).toBeInTheDocument(); // Ensure the title element with default value exists + expect(descriptionElement).toBeInTheDocument(); // Ensure the description element with default value exists + }); + it('renders correctly', () => { const props = { id: '1', @@ -110,6 +136,7 @@ describe('Testing AddOnEntry', () => { expect(getByText('Test addon description')).toBeInTheDocument(); expect(getByText('Test User')).toBeInTheDocument(); }); + it('Uninstall Button works correctly', async () => { const props = { id: '1', diff --git a/src/components/AddOn/core/AddOnEntry/AddOnEntry.tsx b/src/components/AddOn/core/AddOnEntry/AddOnEntry.tsx index cc6feafa6b..257917e2c2 100644 --- a/src/components/AddOn/core/AddOnEntry/AddOnEntry.tsx +++ b/src/components/AddOn/core/AddOnEntry/AddOnEntry.tsx @@ -1,5 +1,4 @@ import React, { useState } from 'react'; -import PropTypes from 'prop-types'; import styles from './AddOnEntry.module.css'; import { Button, Card, Spinner } from 'react-bootstrap'; import { UPDATE_INSTALL_STATUS_PLUGIN_MUTATION } from 'GraphQl/Mutations/mutations'; @@ -13,12 +12,12 @@ import { Navigate, useParams } from 'react-router-dom'; */ interface InterfaceAddOnEntryProps { id: string; - enabled: boolean; - title: string; - description: string; + enabled?: boolean; // Optional props + title?: string; // Optional props + description?: string; // Optional props createdBy: string; - component: string; - modified: any; + component?: string; // Optional props + modified?: any; // Optional props uninstalledOrgs: string[]; getInstalledPlugins: () => any; } @@ -46,11 +45,14 @@ interface InterfaceAddOnEntryProps { */ function addOnEntry({ id, - title, - description, + title = 'No title provided', // Default parameter + description = 'Description not available', // Default parameter createdBy, uninstalledOrgs, getInstalledPlugins, + // enabled = false, // Default parameter + // component = '', // Default parameter + // modified = null, // Default parameter }: InterfaceAddOnEntryProps): JSX.Element { // Translation hook with namespace 'addOnEntry' const { t } = useTranslation('translation', { keyPrefix: 'addOnEntry' }); @@ -147,21 +149,4 @@ function addOnEntry({ ); } -// Default prop values for the component -addOnEntry.defaultProps = { - enabled: false, - configurable: true, - title: '', - description: '', - isInstalled: false, -}; - -addOnEntry.propTypes = { - enabled: PropTypes.bool, - configurable: PropTypes.bool, - title: PropTypes.string, - description: PropTypes.string, - isInstalled: PropTypes.bool, -}; - export default addOnEntry; diff --git a/src/components/AddOn/core/AddOnRegister/AddOnRegister.test.tsx b/src/components/AddOn/core/AddOnRegister/AddOnRegister.test.tsx index 6df580aa76..b04b450977 100644 --- a/src/components/AddOn/core/AddOnRegister/AddOnRegister.test.tsx +++ b/src/components/AddOn/core/AddOnRegister/AddOnRegister.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen, waitFor } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen, waitFor } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { MockedProvider } from '@apollo/react-testing'; import { ADD_PLUGIN_MUTATION } from 'GraphQl/Mutations/mutations'; @@ -41,7 +41,7 @@ const mocks = [ query: ADD_PLUGIN_MUTATION, variables: { pluginName: 'Test Plugin', - pluginCreatedBy: 'Test Creator', + pluginCreatedBy: 'AdminTest Creator', pluginDesc: 'Test Description', pluginInstallStatus: false, installedOrgs: ['id'], @@ -52,7 +52,7 @@ const mocks = [ createPlugin: { _id: '1', pluginName: 'Test Plugin', - pluginCreatedBy: 'Test Creator', + pluginCreatedBy: 'AdminTest Creator', pluginDesc: 'Test Description', }, }, @@ -100,26 +100,31 @@ describe('Testing AddOnRegister', () => { - {} + , ); - - await wait(100); - - userEvent.click(screen.getByRole('button', { name: /Add New/i })); - userEvent.type(screen.getByPlaceholderText(/Ex: Donations/i), 'myplugin'); - userEvent.type( - screen.getByPlaceholderText(/This Plugin enables UI for/i), - 'test description', - ); - userEvent.type( - screen.getByPlaceholderText(/Ex: john Doe/i), - 'test creator', - ); }); + // Wait for the button to be in the document + await waitFor(() => + expect( + screen.getByRole('button', { name: /Add New/i }), + ).toBeInTheDocument(), + ); + + // Simulate user interactions + userEvent.click(screen.getByRole('button', { name: /Add New/i })); + userEvent.type(screen.getByPlaceholderText(/Ex: Donations/i), 'myplugin'); + userEvent.type( + screen.getByPlaceholderText(/This Plugin enables UI for/i), + 'test description', + ); + userEvent.type( + screen.getByPlaceholderText(/Ex: john Doe/i), + 'test creator', + ); }); test('Expect toast.success to be called on successful plugin addition', async () => { @@ -135,22 +140,22 @@ describe('Testing AddOnRegister', () => {
, ); - await waitFor(() => new Promise((resolve) => setTimeout(resolve, 0))); - - userEvent.click(screen.getByRole('button', { name: /Add New/i })); - await wait(100); - expect(screen.getByTestId('addonregisterBtn')).toBeInTheDocument(); - userEvent.type(screen.getByTestId('pluginName'), pluginData.pluginName); - userEvent.type( - screen.getByTestId('pluginCreatedBy'), - pluginData.pluginCreatedBy, - ); - userEvent.type(screen.getByTestId('pluginDesc'), pluginData.pluginDesc); - userEvent.click(screen.getByTestId('addonregisterBtn')); - - await wait(100); - expect(toast.success).toBeCalledWith('Plugin added Successfully'); }); + await waitFor(() => new Promise((resolve) => setTimeout(resolve, 0))); + + userEvent.click(screen.getByRole('button', { name: /Add New/i })); + await wait(100); + expect(screen.getByTestId('addonregisterBtn')).toBeInTheDocument(); + userEvent.type(screen.getByTestId('pluginName'), pluginData.pluginName); + userEvent.type( + screen.getByTestId('pluginCreatedBy'), + pluginData.pluginCreatedBy, + ); + userEvent.type(screen.getByTestId('pluginDesc'), pluginData.pluginDesc); + userEvent.click(screen.getByTestId('addonregisterBtn')); + + await wait(100); + expect(toast.success).toBeCalledWith('Plugin added Successfully'); }); test('Expect the window to reload after successful plugin addition', async () => { @@ -166,23 +171,24 @@ describe('Testing AddOnRegister', () => { , ); - await waitFor(() => new Promise((resolve) => setTimeout(resolve, 0))); - - userEvent.click(screen.getByRole('button', { name: /Add New/i })); - await wait(100); - expect(screen.getByTestId('addonregisterBtn')).toBeInTheDocument(); - userEvent.type(screen.getByTestId('pluginName'), pluginData.pluginName); - userEvent.type( - screen.getByTestId('pluginCreatedBy'), - pluginData.pluginCreatedBy, - ); - userEvent.type(screen.getByTestId('pluginDesc'), pluginData.pluginDesc); - userEvent.click(screen.getByTestId('addonregisterBtn')); - - await wait(3000); // Waiting for 3 seconds to reload the page as timeout is set to 2 seconds in the component - expect(mockNavigate).toHaveBeenCalledWith(0); }); + await waitFor(() => new Promise((resolve) => setTimeout(resolve, 0))); + + userEvent.click(screen.getByRole('button', { name: /Add New/i })); + await wait(100); + expect(screen.getByTestId('addonregisterBtn')).toBeInTheDocument(); + userEvent.type(screen.getByTestId('pluginName'), pluginData.pluginName); + userEvent.type( + screen.getByTestId('pluginCreatedBy'), + pluginData.pluginCreatedBy, + ); + userEvent.type(screen.getByTestId('pluginDesc'), pluginData.pluginDesc); + userEvent.click(screen.getByTestId('addonregisterBtn')); + + await wait(3000); // Waiting for 3 seconds to reload the page as timeout is set to 2 seconds in the component + expect(mockNavigate).toHaveBeenCalledWith(0); }); + test('should be redirected to /orglist if orgId is undefined', async () => { mockId = undefined; render( @@ -190,7 +196,7 @@ describe('Testing AddOnRegister', () => { - {} + diff --git a/src/components/AddOn/core/AddOnRegister/AddOnRegister.tsx b/src/components/AddOn/core/AddOnRegister/AddOnRegister.tsx index 9c572f06d7..56ed6006c2 100644 --- a/src/components/AddOn/core/AddOnRegister/AddOnRegister.tsx +++ b/src/components/AddOn/core/AddOnRegister/AddOnRegister.tsx @@ -19,6 +19,10 @@ interface InterfaceFormStateTypes { installedOrgs: [string] | []; } +interface AddOnRegisterProps { + createdBy?: string; +} + /** * A React component for registering a new add-on plugin. * @@ -31,7 +35,9 @@ interface InterfaceFormStateTypes { * * @returns A JSX element containing the button and modal for plugin registration. */ -function addOnRegister(): JSX.Element { +function addOnRegister({ + createdBy = 'Admin', +}: AddOnRegisterProps): JSX.Element { // Translation hook for the 'addOnRegister' namespace const { t } = useTranslation('translation', { keyPrefix: 'addOnRegister' }); // Translation hook for the 'common' namespace @@ -59,7 +65,7 @@ function addOnRegister(): JSX.Element { // Initial form state const [formState, setFormState] = useState({ pluginName: '', - pluginCreatedBy: '', + pluginCreatedBy: createdBy, // Using the default value here pluginDesc: '', pluginInstallStatus: false, installedOrgs: [currentUrl], @@ -82,7 +88,7 @@ function addOnRegister(): JSX.Element { if (data) { // Show a success message when the plugin is added - toast.success(tCommon('addedSuccessfully', { item: 'Plugin' })); + toast.success(tCommon('addedSuccessfully', { item: 'Plugin' }) as string); // Refresh the page after 2 seconds setTimeout(() => { navigate(0); @@ -188,11 +194,6 @@ function addOnRegister(): JSX.Element { ); } -// Default values for props if not provided -addOnRegister.defaultProps = { - createdBy: 'Admin', -}; - // Prop types validation for the component addOnRegister.propTypes = { createdBy: PropTypes.string, diff --git a/src/components/AddOn/core/AddOnStore/AddOnStore.test.tsx b/src/components/AddOn/core/AddOnStore/AddOnStore.test.tsx index 600e34dd5d..e76e2a7b73 100644 --- a/src/components/AddOn/core/AddOnStore/AddOnStore.test.tsx +++ b/src/components/AddOn/core/AddOnStore/AddOnStore.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import 'jest-location-mock'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen } from '@testing-library/react'; import { ApolloClient, ApolloProvider, diff --git a/src/components/AddOn/core/AddOnStore/AddOnStore.tsx b/src/components/AddOn/core/AddOnStore/AddOnStore.tsx index eb59b9139e..878ad64e31 100644 --- a/src/components/AddOn/core/AddOnStore/AddOnStore.tsx +++ b/src/components/AddOn/core/AddOnStore/AddOnStore.tsx @@ -314,10 +314,4 @@ function addOnStore(): JSX.Element { ); } -// Default values for props if not provided -addOnStore.defaultProps = {}; - -// Prop types validation for the component -addOnStore.propTypes = {}; - export default addOnStore; diff --git a/src/components/Advertisements/Advertisements.test.tsx b/src/components/Advertisements/Advertisements.test.tsx index b45e1bab38..c0992a1012 100644 --- a/src/components/Advertisements/Advertisements.test.tsx +++ b/src/components/Advertisements/Advertisements.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { ApolloClient, ApolloLink, @@ -7,7 +7,7 @@ import { InMemoryCache, } from '@apollo/client'; import { MockedProvider } from '@apollo/client/testing'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen } from '@testing-library/react'; import 'jest-location-mock'; import type { DocumentNode, NormalizedCacheObject } from '@apollo/client'; diff --git a/src/components/Advertisements/Advertisements.tsx b/src/components/Advertisements/Advertisements.tsx index 97d4fed674..f20c2a7d8e 100644 --- a/src/components/Advertisements/Advertisements.tsx +++ b/src/components/Advertisements/Advertisements.tsx @@ -24,7 +24,7 @@ import InfiniteScroll from 'react-infinite-scroll-component'; * */ -export default function advertisements(): JSX.Element { +export default function Advertisements(): JSX.Element { // Retrieve the organization ID from URL parameters const { orgId: currentOrgId } = useParams(); // Translation hook for internationalization @@ -65,7 +65,7 @@ export default function advertisements(): JSX.Element { }); // State to manage the list of advertisements - const [advertisements, setAdvertisements] = useState( + const [advertisements, setAdvertisements] = useState( orgAdvertisementListData?.organizations[0].advertisements?.edges.map( (edge: { node: Ad }) => edge.node, ) || [], @@ -77,10 +77,8 @@ export default function advertisements(): JSX.Element { const ads: Ad[] = orgAdvertisementListData.organizations[0].advertisements?.edges.map( (edge) => edge.node, - ); - after - ? setAdvertisements([...advertisements, ...ads]) - : setAdvertisements(ads); + ) || []; + setAdvertisements(after ? [...advertisements, ...ads] : ads); } }, [orgAdvertisementListData, after]); @@ -273,7 +271,3 @@ export default function advertisements(): JSX.Element { ); } - -advertisements.defaultProps = {}; - -advertisements.propTypes = {}; diff --git a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx index 0850d1f89f..dbd6f88cc3 100644 --- a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx +++ b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.test.tsx @@ -130,7 +130,64 @@ describe('Testing Advertisement Entry Component', () => { expect(deletionFailedText).toBeNull(); }); }); + it('should use default props when none are provided', () => { + render( + , + ): void { + throw new Error('Function not implemented.'); + }} + />, + ); + + //Check if component renders with default ''(empty string) + const elements = screen.getAllByText(''); // This will return an array of matching elements + elements.forEach((element) => expect(element).toBeInTheDocument()); + + // Check that the component renders with default `mediaUrl` (empty string) + const mediaElement = screen.getByTestId('media'); + expect(mediaElement).toHaveAttribute('src', ''); + + // Check that the component renders with default `endDate` + const defaultEndDate = new Date().toDateString(); + expect(screen.getByText(`Ends on ${defaultEndDate}`)).toBeInTheDocument(); + // Check that the component renders with default `startDate` + const defaultStartDate = new Date().toDateString(); + console.log(screen.getByText); + expect(screen.getByText(`Ends on ${defaultStartDate}`)).toBeInTheDocument(); //fix text "Ends on"? + }); + it('should correctly override default props when values are provided', () => { + const mockName = 'Test Ad'; + const mockType = 'Banner'; + const mockMediaUrl = 'https://example.com/media.png'; + const mockEndDate = new Date(2025, 11, 31); + const mockStartDate = new Date(2024, 0, 1); + const mockOrganizationId = 'org123'; + + const { getByText } = render( + , + ): void { + throw new Error('Function not implemented.'); + }} + />, + ); + + // Check that the component renders with provided values + expect(getByText(mockName)).toBeInTheDocument(); + // Add more checks based on how each prop affects rendering + }); it('should open and close the dropdown when options button is clicked', () => { const { getByTestId, queryByText, getAllByText } = render( diff --git a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx index aa8324dc19..7368ded68e 100644 --- a/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx +++ b/src/components/Advertisements/core/AdvertisementEntry/AdvertisementEntry.tsx @@ -12,12 +12,12 @@ import { toast } from 'react-toastify'; interface InterfaceAddOnEntryProps { id: string; - name: string; - mediaUrl: string; - type: string; - organizationId: string; - startDate: Date; - endDate: Date; + name?: string; + mediaUrl?: string; + type?: string; + organizationId?: string; + startDate?: Date; + endDate?: Date; setAfter: React.Dispatch>; } @@ -28,14 +28,14 @@ interface InterfaceAddOnEntryProps { * @param props - Component properties * @returns The rendered component */ -function advertisementEntry({ +function AdvertisementEntry({ id, - name, - type, - mediaUrl, - endDate, - organizationId, - startDate, + name = '', + type = '', + mediaUrl = '', + endDate = new Date(), + organizationId = '', + startDate = new Date(), setAfter, }: InterfaceAddOnEntryProps): JSX.Element { const { t } = useTranslation('translation', { keyPrefix: 'advertisement' }); @@ -75,9 +75,9 @@ function advertisementEntry({ id: id.toString(), }, }); - toast.success(t('advertisementDeleted')); + toast.success(t('advertisementDeleted') as string); setButtonLoading(false); - setAfter(null); + setAfter?.(null); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -92,6 +92,7 @@ function advertisementEntry({ const handleOptionsClick = (): void => { setDropdown(!dropdown); }; + return ( <> @@ -208,7 +209,7 @@ function advertisementEntry({ ); } -advertisementEntry.propTypes = { +AdvertisementEntry.propTypes = { name: PropTypes.string, type: PropTypes.string, organizationId: PropTypes.string, @@ -217,12 +218,4 @@ advertisementEntry.propTypes = { startDate: PropTypes.instanceOf(Date), }; -advertisementEntry.defaultProps = { - name: '', - type: '', - organizationId: '', - mediaUrl: '', - endDate: new Date(), - startDate: new Date(), -}; -export default advertisementEntry; +export default AdvertisementEntry; diff --git a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.test.tsx b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.test.tsx index c3aa536d1f..1bf16ec76b 100644 --- a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.test.tsx +++ b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { render, fireEvent, waitFor, screen } from '@testing-library/react'; import { @@ -147,12 +147,12 @@ describe('Testing Advertisement Register Component', () => { @@ -166,176 +166,242 @@ describe('Testing Advertisement Register Component', () => { }); test('create advertisement', async () => { + jest.useFakeTimers(); const setTimeoutSpy = jest.spyOn(global, 'setTimeout'); - const { getByText, queryByText, getByLabelText } = render( - - - - - - - - - , - ); - expect(getByText(translations.createAdvertisement)).toBeInTheDocument(); + await act(async () => { + render( + + + + + + + + + , + ); + }); - fireEvent.click(getByText(translations.createAdvertisement)); - expect(queryByText(translations.addNew)).toBeInTheDocument(); + expect( + screen.getByText(translations.createAdvertisement), + ).toBeInTheDocument(); - fireEvent.change(getByLabelText(translations.Rname), { - target: { value: 'Ad1' }, + await act(async () => { + fireEvent.click(screen.getByText(translations.createAdvertisement)); }); - expect(getByLabelText(translations.Rname)).toHaveValue('Ad1'); - const mediaFile = new File(['media content'], 'test.png', { - type: 'image/png', + expect(screen.queryByText(translations.addNew)).toBeInTheDocument(); + + await act(async () => { + fireEvent.change(screen.getByLabelText(translations.Rname), { + target: { value: 'Ad1' }, + }); + + const mediaFile = new File(['media content'], 'test.png', { + type: 'image/png', + }); + + fireEvent.change(screen.getByLabelText(translations.Rmedia), { + target: { + files: [mediaFile], + }, + }); }); - const mediaInput = getByLabelText(translations.Rmedia); - fireEvent.change(mediaInput, { - target: { - files: [mediaFile], - }, + await waitFor(() => { + expect(screen.getByTestId('mediaPreview')).toBeInTheDocument(); }); - const mediaPreview = await screen.findByTestId('mediaPreview'); - expect(mediaPreview).toBeInTheDocument(); + await act(async () => { + fireEvent.change(screen.getByLabelText(translations.Rtype), { + target: { value: 'BANNER' }, + }); - fireEvent.change(getByLabelText(translations.Rtype), { - target: { value: 'BANNER' }, - }); - expect(getByLabelText(translations.Rtype)).toHaveValue('BANNER'); + fireEvent.change(screen.getByLabelText(translations.RstartDate), { + target: { value: '2023-01-01' }, + }); - fireEvent.change(getByLabelText(translations.RstartDate), { - target: { value: '2023-01-01' }, + fireEvent.change(screen.getByLabelText(translations.RendDate), { + target: { value: '2023-02-01' }, + }); }); - expect(getByLabelText(translations.RstartDate)).toHaveValue('2023-01-01'); - fireEvent.change(getByLabelText(translations.RendDate), { - target: { value: '2023-02-01' }, + expect(screen.getByLabelText(translations.Rname)).toHaveValue('Ad1'); + expect(screen.getByLabelText(translations.Rtype)).toHaveValue('BANNER'); + expect(screen.getByLabelText(translations.RstartDate)).toHaveValue( + '2023-01-01', + ); + expect(screen.getByLabelText(translations.RendDate)).toHaveValue( + '2023-02-01', + ); + + await act(async () => { + fireEvent.click(screen.getByText(translations.register)); }); - expect(getByLabelText(translations.RendDate)).toHaveValue('2023-02-01'); await waitFor(() => { - fireEvent.click(getByText(translations.register)); + expect(toast.success).toBeCalledWith( + 'Advertisement created successfully.', + ); + expect(setTimeoutSpy).toHaveBeenCalled(); }); - expect(toast.success).toBeCalledWith('Advertisement created successfully.'); - expect(setTimeoutSpy).toHaveBeenCalled(); + jest.useRealTimers(); }); test('update advertisement', async () => { + jest.useFakeTimers(); const setTimeoutSpy = jest.spyOn(global, 'setTimeout'); - const { getByText, getByLabelText } = render( - - - - - - - - - , - ); - - fireEvent.click(getByText(translations.edit)); - fireEvent.change(getByLabelText(translations.Rname), { - target: { value: 'Ad1' }, + await act(async () => { + render( + + + + + + + + + , + ); }); - expect(getByLabelText(translations.Rname)).toHaveValue('Ad1'); - const mediaFile = new File(['media content'], 'test.png', { - type: 'image/png', + await waitFor(() => { + expect(screen.getByText(translations.edit)).toBeInTheDocument(); }); - const mediaInput = getByLabelText(translations.Rmedia); - fireEvent.change(mediaInput, { - target: { - files: [mediaFile], - }, + await act(async () => { + fireEvent.click(screen.getByText(translations.edit)); }); - const mediaPreview = await screen.findByTestId('mediaPreview'); - expect(mediaPreview).toBeInTheDocument(); + await act(async () => { + fireEvent.change(screen.getByLabelText(translations.Rname), { + target: { value: 'Ad1' }, + }); - fireEvent.change(getByLabelText(translations.Rtype), { - target: { value: 'BANNER' }, + const mediaFile = new File(['media content'], 'test.png', { + type: 'image/png', + }); + + fireEvent.change(screen.getByLabelText(translations.Rmedia), { + target: { + files: [mediaFile], + }, + }); }); - expect(getByLabelText(translations.Rtype)).toHaveValue('BANNER'); - fireEvent.change(getByLabelText(translations.RstartDate), { - target: { value: '2023-01-01' }, + await waitFor(() => { + expect(screen.getByTestId('mediaPreview')).toBeInTheDocument(); }); - expect(getByLabelText(translations.RstartDate)).toHaveValue('2023-01-01'); - fireEvent.change(getByLabelText(translations.RendDate), { - target: { value: '2023-02-01' }, + await act(async () => { + fireEvent.change(screen.getByLabelText(translations.Rtype), { + target: { value: 'BANNER' }, + }); + + fireEvent.change(screen.getByLabelText(translations.RstartDate), { + target: { value: '2023-01-01' }, + }); + + fireEvent.change(screen.getByLabelText(translations.RendDate), { + target: { value: '2023-02-01' }, + }); + }); + + expect(screen.getByLabelText(translations.Rname)).toHaveValue('Ad1'); + expect(screen.getByLabelText(translations.Rtype)).toHaveValue('BANNER'); + expect(screen.getByLabelText(translations.RstartDate)).toHaveValue( + '2023-01-01', + ); + expect(screen.getByLabelText(translations.RendDate)).toHaveValue( + '2023-02-01', + ); + + await act(async () => { + fireEvent.click(screen.getByText(translations.saveChanges)); }); - expect(getByLabelText(translations.RendDate)).toHaveValue('2023-02-01'); await waitFor(() => { - fireEvent.click(getByText(translations.saveChanges)); + expect(toast.success).toBeCalledWith( + 'Advertisement created successfully.', + ); + expect(setTimeoutSpy).toHaveBeenCalled(); }); - expect(toast.success).toBeCalledWith('Advertisement created successfully.'); - expect(setTimeoutSpy).toHaveBeenCalled(); + + jest.useRealTimers(); }); test('Logs error to the console and shows error toast when advertisement creation fails', async () => { + jest.useFakeTimers(); const setTimeoutSpy = jest.spyOn(global, 'setTimeout'); - const { getByText, queryByText } = render( - - - - - - - - - , - ); + const toastErrorSpy = jest.spyOn(toast, 'error'); + + await act(async () => { + render( + + + + + + + + + , + ); + }); - expect(getByText(translations.createAdvertisement)).toBeInTheDocument(); + expect( + screen.getByText(translations.createAdvertisement), + ).toBeInTheDocument(); - fireEvent.click(getByText(translations.createAdvertisement)); - expect(queryByText(translations.addNew)).toBeInTheDocument(); + await act(async () => { + fireEvent.click(screen.getByText(translations.createAdvertisement)); + }); + + expect(screen.queryByText(translations.addNew)).toBeInTheDocument(); + + await act(async () => { + fireEvent.click(screen.getByText(translations.register)); + }); await waitFor(() => { - fireEvent.click(getByText(translations.register)); + expect(toastErrorSpy).toHaveBeenCalledWith( + `An error occurred. Couldn't create advertisement`, + ); }); - expect(toast.error).toBeCalledWith( - `An error occurred. Couldn't create advertisement`, - ); + expect(setTimeoutSpy).toHaveBeenCalled(); + jest.useRealTimers(); }); test('Throws error when the end date is less than the start date', async () => { + jest.useFakeTimers(); const setTimeoutSpy = jest.spyOn(global, 'setTimeout'); const { getByText, queryByText, getByLabelText } = render( @@ -343,12 +409,12 @@ describe('Testing Advertisement Register Component', () => { @@ -403,21 +469,23 @@ describe('Testing Advertisement Register Component', () => { 'End Date should be greater than or equal to Start Date', ); expect(setTimeoutSpy).toHaveBeenCalled(); + jest.useRealTimers(); }); test('AdvertismentRegister component loads correctly in edit mode', async () => { + jest.useFakeTimers(); render( @@ -429,21 +497,23 @@ describe('Testing Advertisement Register Component', () => { await waitFor(() => { expect(screen.getByTestId('editBtn')).toBeInTheDocument(); }); + jest.useRealTimers(); }); test('Opens and closes modals on button click', async () => { + jest.useFakeTimers(); const { getByText, queryByText } = render( @@ -459,9 +529,11 @@ describe('Testing Advertisement Register Component', () => { await waitFor(() => { expect(queryByText(translations.close)).not.toBeInTheDocument(); }); + jest.useRealTimers(); }); test('Throws error when the end date is less than the start date while editing the advertisement', async () => { + jest.useFakeTimers(); const { getByText, getByLabelText, queryByText } = render( @@ -470,12 +542,12 @@ describe('Testing Advertisement Register Component', () => { { } @@ -524,21 +596,23 @@ describe('Testing Advertisement Register Component', () => { 'End Date should be greater than or equal to Start Date', ); }); + jest.useRealTimers(); }); test('Media preview renders correctly', async () => { + jest.useFakeTimers(); render( @@ -563,4 +637,5 @@ describe('Testing Advertisement Register Component', () => { fireEvent.click(closeButton); expect(mediaPreview).not.toBeInTheDocument(); }); + jest.useRealTimers(); }); diff --git a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx index cded70fa97..1d5c51ce84 100644 --- a/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx +++ b/src/components/Advertisements/core/AdvertisementRegister/AdvertisementRegister.tsx @@ -58,13 +58,13 @@ interface InterfaceFormStateTypes { * ``` */ function advertisementRegister({ - formStatus, + formStatus = 'register', idEdit, - nameEdit, - typeEdit, - advertisementMediaEdit, - endDateEdit, - startDateEdit, + nameEdit = '', + typeEdit = 'BANNER', + advertisementMediaEdit = '', + endDateEdit = new Date(), + startDateEdit = new Date(), setAfter, }: InterfaceAddOnRegisterProps): JSX.Element { const { t } = useTranslation('translation', { keyPrefix: 'advertisement' }); @@ -135,7 +135,7 @@ function advertisementRegister({ try { console.log('At handle register', formState); if (formState.endDate < formState.startDate) { - toast.error(t('endDateGreaterOrEqual')); + toast.error(t('endDateGreaterOrEqual') as string); return; } const { data } = await create({ @@ -150,7 +150,7 @@ function advertisementRegister({ }); if (data) { - toast.success(t('advertisementCreated')); + toast.success(t('advertisementCreated') as string); setFormState({ name: '', advertisementMedia: '', @@ -164,7 +164,9 @@ function advertisementRegister({ } catch (error: unknown) { if (error instanceof Error) { toast.error( - tErrors('errorOccurredCouldntCreate', { entity: 'advertisement' }), + tErrors('errorOccurredCouldntCreate', { + entity: 'advertisement', + }) as string, ); console.log('error occured', error.message); } @@ -190,7 +192,7 @@ function advertisementRegister({ updatedFields.type = formState.type; } if (formState.endDate < formState.startDate) { - toast.error(t('endDateGreaterOrEqual')); + toast.error(t('endDateGreaterOrEqual') as string); return; } const startDateFormattedString = dayjs(formState.startDate).format( @@ -231,7 +233,7 @@ function advertisementRegister({ if (data) { toast.success( - tCommon('updatedSuccessfully', { item: 'Advertisement' }), + tCommon('updatedSuccessfully', { item: 'Advertisement' }) as string, ); handleClose(); setAfter(null); @@ -430,16 +432,6 @@ function advertisementRegister({ ); } -advertisementRegister.defaultProps = { - name: '', - advertisementMedia: '', - type: 'BANNER', - startDate: new Date(), - endDate: new Date(), - organizationId: '', - formStatus: 'register', -}; - advertisementRegister.propTypes = { name: PropTypes.string, advertisementMedia: PropTypes.string, diff --git a/src/components/AgendaCategory/AgendaCategoryContainer.tsx b/src/components/AgendaCategory/AgendaCategoryContainer.tsx index fa41dd0099..5fc038aee0 100644 --- a/src/components/AgendaCategory/AgendaCategoryContainer.tsx +++ b/src/components/AgendaCategory/AgendaCategoryContainer.tsx @@ -129,7 +129,7 @@ function agendaCategoryContainer({ agendaCategoryRefetch(); hideUpdateModal(); - toast.success(t('agendaCategoryUpdated')); + toast.success(t('agendaCategoryUpdated') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(`Agenda Category Update Failed ${error.message}`); @@ -153,7 +153,7 @@ function agendaCategoryContainer({ }); agendaCategoryRefetch(); toggleDeleteModal(); - toast.success(t('agendaCategoryDeleted')); + toast.success(t('agendaCategoryDeleted') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(`Agenda Category Delete Failed, ${error.message}`); diff --git a/src/components/AgendaItems/AgendaItemsContainer.test.tsx b/src/components/AgendaItems/AgendaItemsContainer.test.tsx index cb514bbfb9..7ceb8b4d08 100644 --- a/src/components/AgendaItems/AgendaItemsContainer.test.tsx +++ b/src/components/AgendaItems/AgendaItemsContainer.test.tsx @@ -1,9 +1,8 @@ -import React from 'react'; +import React, { act } from 'react'; import { render, screen, waitFor, - act, waitForElementToBeRemoved, fireEvent, } from '@testing-library/react'; @@ -36,6 +35,15 @@ jest.mock('react-toastify', () => ({ }, })); +//temporarily fixes react-beautiful-dnd droppable method's depreciation error +//needs to be fixed in React 19 +jest.spyOn(console, 'error').mockImplementation((message) => { + if (message.includes('Support for defaultProps will be removed')) { + return; + } + console.error(message); +}); + async function wait(ms = 100): Promise { await act(async () => { return new Promise((resolve) => setTimeout(resolve, ms)); diff --git a/src/components/AgendaItems/AgendaItemsContainer.tsx b/src/components/AgendaItems/AgendaItemsContainer.tsx index 6c7f33cd5e..4e50cb5fe8 100644 --- a/src/components/AgendaItems/AgendaItemsContainer.tsx +++ b/src/components/AgendaItems/AgendaItemsContainer.tsx @@ -145,7 +145,7 @@ function AgendaItemsContainer({ }); agendaItemRefetch(); hideUpdateModal(); - toast.success(t('agendaItemUpdated')); + toast.success(t('agendaItemUpdated') as string); } catch (error) { if (error instanceof Error) { toast.error(`${error.message}`); @@ -167,7 +167,7 @@ function AgendaItemsContainer({ }); agendaItemRefetch(); toggleDeleteModal(); - toast.success(t('agendaItemDeleted')); + toast.success(t('agendaItemDeleted') as string); } catch (error) { if (error instanceof Error) { toast.error(`${error.message}`); diff --git a/src/components/ChangeLanguageDropdown/ChangeLanguageDropdown.test.tsx b/src/components/ChangeLanguageDropdown/ChangeLanguageDropdown.test.tsx index 71f01677b4..dc14f6ce17 100644 --- a/src/components/ChangeLanguageDropdown/ChangeLanguageDropdown.test.tsx +++ b/src/components/ChangeLanguageDropdown/ChangeLanguageDropdown.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render } from '@testing-library/react'; +import React, { act } from 'react'; +import { render } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; import { BrowserRouter } from 'react-router-dom'; @@ -11,8 +11,8 @@ import { MockedProvider } from '@apollo/react-testing'; import { UPDATE_USER_MUTATION } from 'GraphQl/Mutations/mutations'; import { StaticMockLink } from 'utils/StaticMockLink'; import useLocalStorage from 'utils/useLocalstorage'; -import { Provider } from 'react-redux'; -import { store } from 'state/store'; +// import { Provider } from 'react-redux'; +// import { store } from 'state/store'; const { setItem } = useLocalStorage(); async function wait(ms = 100): Promise { diff --git a/src/components/CheckIn/TableRow.tsx b/src/components/CheckIn/TableRow.tsx index c2c2def815..b7b5318f84 100644 --- a/src/components/CheckIn/TableRow.tsx +++ b/src/components/CheckIn/TableRow.tsx @@ -40,11 +40,11 @@ export const TableRow = ({ }, }) .then(() => { - toast.success(t('checkedInSuccessfully')); + toast.success(t('checkedInSuccessfully') as string); refetch(); }) .catch((err) => { - toast.error(t('errorCheckingIn')); + toast.error(t('errorCheckingIn') as string); toast.error(err.message); }); }; diff --git a/src/components/CollapsibleDropdown/CollapsibleDropdown.test.tsx b/src/components/CollapsibleDropdown/CollapsibleDropdown.test.tsx index cfb1001e6b..efee248ffb 100644 --- a/src/components/CollapsibleDropdown/CollapsibleDropdown.test.tsx +++ b/src/components/CollapsibleDropdown/CollapsibleDropdown.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { render, screen, fireEvent } from '@testing-library/react'; import { BrowserRouter } from 'react-router-dom'; @@ -72,7 +72,9 @@ describe('Testing CollapsibleDropdown component', () => { const nonActiveDropdownBtn = screen.getByText('SubCategory 2'); // Check if dropdown is rendered with correct classes - activeDropdownBtn.click(); + act(() => { + fireEvent.click(activeDropdownBtn); + }); expect(parentDropdownBtn).toBeInTheDocument(); expect(parentDropdownBtn).toHaveClass('text-white'); expect(parentDropdownBtn).toHaveClass('btn-success'); @@ -88,15 +90,21 @@ describe('Testing CollapsibleDropdown component', () => { expect(nonActiveDropdownBtn).toHaveClass('btn-light'); // Check if dropdown is collapsed after clicking on it - fireEvent.click(parentDropdownBtn); + act(() => { + fireEvent.click(parentDropdownBtn); + }); expect(props.setShowDropdown).toHaveBeenCalledWith(false); // Check if dropdown is expanded after clicking on it again - fireEvent.click(parentDropdownBtn); + act(() => { + fireEvent.click(parentDropdownBtn); + }); expect(props.setShowDropdown).toHaveBeenCalledWith(true); - // Click on non active dropdown button and check if it navigates to the correct url - nonActiveDropdownBtn.click(); + // Click on non-active dropdown button and check if it navigates to the correct URL + act(() => { + fireEvent.click(nonActiveDropdownBtn); + }); expect(window.location.pathname).toBe('/sub-category-2'); }); }); diff --git a/src/components/ContriStats/ContriStats.test.tsx b/src/components/ContriStats/ContriStats.test.tsx index 3164c880c8..8edb853684 100644 --- a/src/components/ContriStats/ContriStats.test.tsx +++ b/src/components/ContriStats/ContriStats.test.tsx @@ -14,7 +14,6 @@ const client: ApolloClient = new ApolloClient({ describe('Testing Contribution Stats', () => { const props = { - key: '123', id: '234', recentAmount: '200', highestAmount: '500', diff --git a/src/components/ContriStats/ContriStats.tsx b/src/components/ContriStats/ContriStats.tsx index 995a7ab93d..a2db307d91 100644 --- a/src/components/ContriStats/ContriStats.tsx +++ b/src/components/ContriStats/ContriStats.tsx @@ -4,7 +4,6 @@ import { useTranslation } from 'react-i18next'; import styles from './ContriStats.module.css'; interface InterfaceContriStatsProps { - key: string; id: string; recentAmount: string; highestAmount: string; @@ -18,7 +17,7 @@ interface InterfaceContriStatsProps { * * @returns JSX.Element - The rendered component displaying the contribution stats. */ -function contriStats(props: InterfaceContriStatsProps): JSX.Element { +function ContriStats(props: InterfaceContriStatsProps): JSX.Element { const { t } = useTranslation('translation', { keyPrefix: 'contriStats', }); @@ -37,4 +36,4 @@ function contriStats(props: InterfaceContriStatsProps): JSX.Element { ); } -export default contriStats; +export default ContriStats; diff --git a/src/components/DeleteOrg/DeleteOrg.test.tsx b/src/components/DeleteOrg/DeleteOrg.test.tsx index d9ac99f3ad..77ffe65c08 100644 --- a/src/components/DeleteOrg/DeleteOrg.test.tsx +++ b/src/components/DeleteOrg/DeleteOrg.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { render, screen } from '@testing-library/react'; +import { render, screen, waitFor } from '@testing-library/react'; import 'jest-location-mock'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; @@ -10,7 +10,6 @@ import { DELETE_ORGANIZATION_MUTATION, REMOVE_SAMPLE_ORGANIZATION_MUTATION, } from 'GraphQl/Mutations/mutations'; -import { act } from 'react-dom/test-utils'; import { store } from 'state/store'; import { StaticMockLink } from 'utils/StaticMockLink'; import i18nForTest from 'utils/i18nForTest'; @@ -98,6 +97,7 @@ const MOCKS_WITH_ERROR = [ error: new Error('Failed to delete sample organization'), }, ]; + const mockNavgatePush = jest.fn(); let mockURL = '123'; jest.mock('react-router-dom', () => ({ @@ -117,87 +117,119 @@ describe('Delete Organization Component', () => { test('should be able to Toggle Delete Organization Modal', async () => { mockURL = '456'; setItem('SuperAdmin', true); - render( - - - - - - - - - - , - ); - await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); - expect(screen.getByTestId(/orgDeleteModal/i)).toBeInTheDocument(); - screen.getByTestId(/closeDelOrgModalBtn/i).click(); await act(async () => { - expect(screen.queryByTestId(/orgDeleteModal/i)).not.toHaveFocus(); + render( + + + + + + + + + + , + ); + }); + act(() => { + screen.getByTestId(/openDeleteModalBtn/i).click(); + }); + expect(await screen.findByTestId(/orgDeleteModal/i)).toBeInTheDocument(); + act(() => { + screen.getByTestId(/closeDelOrgModalBtn/i).click(); + }); + await waitFor(() => { + expect(screen.queryByTestId(/orgDeleteModal/i)).not.toBeInTheDocument(); }); }); test('should be able to Toggle Delete Organization Modal When Organization is Sample Organization', async () => { mockURL = '123'; setItem('SuperAdmin', true); - render( - - - - - - - - - - , - ); + await act(async () => { + render( + + + + + + + + + + , + ); + }); await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); + act(() => { + screen.getByTestId(/openDeleteModalBtn/i).click(); + }); expect(screen.getByTestId(/orgDeleteModal/i)).toBeInTheDocument(); - screen.getByTestId(/closeDelOrgModalBtn/i).click(); - await act(async () => { - expect(screen.queryByTestId(/orgDeleteModal/i)).not.toHaveFocus(); + act(() => { + screen.getByTestId(/closeDelOrgModalBtn/i).click(); + }); + await waitFor(() => { + expect(screen.queryByTestId(/orgDeleteModal/i)).not.toBeInTheDocument(); }); }); test('Delete organization functionality should work properly', async () => { mockURL = '456'; setItem('SuperAdmin', true); - render( - - - - - - - - - , - ); - await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); - screen.getByTestId(/deleteOrganizationBtn/i).click(); + await act(async () => { + render( + + + + + + + + + , + ); + }); + screen.debug(); + act(() => { + screen.getByTestId('openDeleteModalBtn').click(); + }); + screen.debug(); + expect(await screen.findByTestId('orgDeleteModal')).toBeInTheDocument(); + const deleteButton = await screen.findByTestId('deleteOrganizationBtn'); + act(() => { + deleteButton.click(); + }); }); test('Delete organization functionality should work properly for sample org', async () => { mockURL = '123'; setItem('SuperAdmin', true); - render( - - - - - - - - - , - ); - await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); - screen.getByTestId(/deleteOrganizationBtn/i).click(); + await act(async () => { + render( + + + + + + + + + , + ); + }); + await waitFor(() => { + expect(screen.getByTestId('openDeleteModalBtn')).toBeInTheDocument(); + }); + act(() => { + screen.getByTestId('openDeleteModalBtn').click(); + }); + await waitFor(() => { + expect(screen.getByTestId('orgDeleteModal')).toBeInTheDocument(); + }); + const deleteButton = await screen.findByTestId('deleteOrganizationBtn'); + act(() => { + deleteButton.click(); + }); await wait(2000); expect(mockNavgatePush).toHaveBeenCalledWith('/orglist'); }); @@ -206,43 +238,70 @@ describe('Delete Organization Component', () => { mockURL = '123'; setItem('SuperAdmin', true); jest.spyOn(toast, 'error'); - render( - - - - - - - - - , - ); - await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); - screen.getByTestId(/deleteOrganizationBtn/i).click(); - await wait(); - expect(toast.error).toHaveBeenCalledWith( - 'Failed to delete sample organization', - ); + await act(async () => { + render( + + + + + + + + + , + ); + }); + await waitFor(() => { + expect(screen.getByTestId('openDeleteModalBtn')).toBeInTheDocument(); + }); + act(() => { + screen.getByTestId('openDeleteModalBtn').click(); + }); + await waitFor(() => { + expect(screen.getByTestId('orgDeleteModal')).toBeInTheDocument(); + }); + act(() => { + screen.getByTestId('deleteOrganizationBtn').click(); + }); + await waitFor(() => { + expect(toast.error).toHaveBeenCalledWith( + 'Failed to delete sample organization', + ); + }); }); test('Error handling for DELETE_ORGANIZATION_MUTATION mock', async () => { mockURL = '456'; setItem('SuperAdmin', true); - render( - - - - - - - - - , - ); - await wait(); - screen.getByTestId(/openDeleteModalBtn/i).click(); - screen.getByTestId(/deleteOrganizationBtn/i).click(); - await wait(); + jest.spyOn(toast, 'error'); + + await act(async () => { + render( + + + + + + + + + , + ); + }); + await waitFor(() => { + expect(screen.getByTestId('openDeleteModalBtn')).toBeInTheDocument(); + }); + act(() => { + screen.getByTestId('openDeleteModalBtn').click(); + }); + await waitFor(() => { + expect(screen.getByTestId('orgDeleteModal')).toBeInTheDocument(); + }); + act(() => { + screen.getByTestId('deleteOrganizationBtn').click(); + }); + await waitFor(() => { + expect(toast.error).toHaveBeenCalledWith('Failed to delete organization'); + }); }); }); diff --git a/src/components/DeleteOrg/DeleteOrg.tsx b/src/components/DeleteOrg/DeleteOrg.tsx index e9370cb1a6..8edf11ca22 100644 --- a/src/components/DeleteOrg/DeleteOrg.tsx +++ b/src/components/DeleteOrg/DeleteOrg.tsx @@ -69,7 +69,7 @@ function deleteOrg(): JSX.Element { // If it's a sample organization, use a specific mutation removeSampleOrganization() .then(() => { - toast.success(t('successfullyDeletedSampleOrganization')); + toast.success(t('successfullyDeletedSampleOrganization') as string); setTimeout(() => { navigate('/orglist'); }, 1000); diff --git a/src/components/EditCustomFieldDropDown/EditCustomFieldDropDown.test.tsx b/src/components/EditCustomFieldDropDown/EditCustomFieldDropDown.test.tsx index 2e3c52ad38..fc3431c7b2 100644 --- a/src/components/EditCustomFieldDropDown/EditCustomFieldDropDown.test.tsx +++ b/src/components/EditCustomFieldDropDown/EditCustomFieldDropDown.test.tsx @@ -1,6 +1,6 @@ import type { Dispatch, SetStateAction } from 'react'; -import React from 'react'; -import { act, render } from '@testing-library/react'; +import React, { act } from 'react'; +import { render } from '@testing-library/react'; import { BrowserRouter } from 'react-router-dom'; import EditOrgCustomFieldDropDown from './EditCustomFieldDropDown'; import type { InterfaceCustomFieldData } from 'components/OrgProfileFieldSettings/OrgProfileFieldSettings'; diff --git a/src/components/EventCalendar/EventHeader.test.tsx b/src/components/EventCalendar/EventHeader.test.tsx index 215182af9c..e18d066306 100644 --- a/src/components/EventCalendar/EventHeader.test.tsx +++ b/src/components/EventCalendar/EventHeader.test.tsx @@ -1,10 +1,9 @@ -import React from 'react'; +import React, { act } from 'react'; // Import act for async testing import { render, fireEvent } from '@testing-library/react'; import EventHeader from './EventHeader'; import { ViewType } from '../../screens/OrganizationEvents/OrganizationEvents'; import { I18nextProvider } from 'react-i18next'; import i18nForTest from 'utils/i18nForTest'; -import { act } from 'react-dom/test-utils'; // Import act for async testing describe('EventHeader Component', () => { const viewType = ViewType.MONTH; diff --git a/src/components/EventListCard/EventListCard.test.tsx b/src/components/EventListCard/EventListCard.test.tsx index 22ddccd8d2..b882d5887b 100644 --- a/src/components/EventListCard/EventListCard.test.tsx +++ b/src/components/EventListCard/EventListCard.test.tsx @@ -1,12 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import type { RenderResult } from '@testing-library/react'; -import { - act, - render, - screen, - fireEvent, - waitFor, -} from '@testing-library/react'; +import { render, screen, fireEvent, waitFor } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; @@ -57,6 +51,8 @@ const translations = { const renderEventListCard = ( props: InterfaceEventListCardProps, ): RenderResult => { + const { key, ...restProps } = props; // Destructure the key and separate other props + return render( @@ -66,11 +62,11 @@ const renderEventListCard = ( } + element={} /> } + element={} /> { }); test('should show an error toast when the delete event mutation fails', async () => { + // Destructure key from props[1] and pass it separately to avoid spreading it + const { key, ...otherProps } = props[1]; render( @@ -887,11 +885,11 @@ describe('Testing Event List Card', () => { } + element={} /> } + element={} /> diff --git a/src/components/EventListCard/EventListCardModals.tsx b/src/components/EventListCard/EventListCardModals.tsx index 878d8796a4..7876579a23 100644 --- a/src/components/EventListCard/EventListCardModals.tsx +++ b/src/components/EventListCard/EventListCardModals.tsx @@ -284,7 +284,7 @@ function EventListCardModals({ }); if (data) { - toast.success(t('eventUpdated')); + toast.success(t('eventUpdated') as string); setRecurringEventUpdateModalIsOpen(false); hideViewModal(); if (refetchEvents) { @@ -323,7 +323,7 @@ function EventListCardModals({ }); if (data) { - toast.success(t('eventDeleted')); + toast.success(t('eventDeleted') as string); setEventDeleteModalIsOpen(false); hideViewModal(); if (refetchEvents) { diff --git a/src/components/EventManagement/Dashboard/EventDashboard.tsx b/src/components/EventManagement/Dashboard/EventDashboard.tsx index b811bcd137..dfb3eb0245 100644 --- a/src/components/EventManagement/Dashboard/EventDashboard.tsx +++ b/src/components/EventManagement/Dashboard/EventDashboard.tsx @@ -75,6 +75,9 @@ const EventDashboard = (props: { eventId: string }): JSX.Element => { const formattedDate = `${day}${suffix} ${monthNames[monthIndex]} ${year}`; return formattedDate; } + if (!eventData || !eventData.event) { + return ; // Fallback UI while data is loading + } // Render event details return ( @@ -86,36 +89,36 @@ const EventDashboard = (props: { eventId: string }): JSX.Element => {

- {eventData.event.startTime !== null - ? `${formatTime(eventData.event.startTime)}` - : ``} + {eventData?.event?.startTime !== null + ? `${formatTime(eventData?.event?.startTime)}` + : ''} {' '} - {formatDate(eventData.event.startDate)}{' '} + {formatDate(eventData?.event?.startDate)}{' '}

{t('to')}

- {eventData.event.endTime !== null - ? `${formatTime(eventData.event.endTime)}` + {eventData?.event?.endTime !== null + ? `${formatTime(eventData?.event?.endTime)}` : ``} {' '} - {formatDate(eventData.event.endDate)}{' '} + {formatDate(eventData?.event?.endDate)}{' '}

-

{eventData.event.title}

+

{eventData?.event?.title}

- {eventData.event.description} + {eventData?.event?.description}

- Location: {eventData.event.location} + Location: {eventData?.event?.location}

Registrants:{' '} - {eventData.event.attendees.length} + {eventData?.event?.attendees?.length}


diff --git a/src/components/EventManagement/EventActionItems/EventActionItems.test.tsx b/src/components/EventManagement/EventActionItems/EventActionItems.test.tsx index 9da0b5247a..3d1e5fa7bc 100644 --- a/src/components/EventManagement/EventActionItems/EventActionItems.test.tsx +++ b/src/components/EventManagement/EventActionItems/EventActionItems.test.tsx @@ -1,9 +1,8 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; import { LocalizationProvider } from '@mui/x-date-pickers'; import { AdapterDayjs } from '@mui/x-date-pickers/AdapterDayjs'; import { - act, fireEvent, render, screen, diff --git a/src/components/EventManagement/EventActionItems/EventActionItems.tsx b/src/components/EventManagement/EventActionItems/EventActionItems.tsx index cbe878fa07..0f9d723adc 100644 --- a/src/components/EventManagement/EventActionItems/EventActionItems.tsx +++ b/src/components/EventManagement/EventActionItems/EventActionItems.tsx @@ -8,7 +8,7 @@ import { useTranslation } from 'react-i18next'; import { toast } from 'react-toastify'; import styles from './EventActionItems.module.css'; import { DataGrid } from '@mui/x-data-grid'; -import type { GridCellParams } from '@mui/x-data-grid'; +// import type { GridCellParams } from '@mui/x-data-grid'; import { Stack } from '@mui/material'; import Modal from 'react-bootstrap/Modal'; import { @@ -156,7 +156,7 @@ function eventActionItems(props: { eventId: string }): JSX.Element { setDueDate(new Date()); actionItemsRefetch(); hideCreateModal(); - toast.success(t('successfulCreation')); + toast.success(t('successfulCreation') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -188,7 +188,7 @@ function eventActionItems(props: { eventId: string }): JSX.Element { actionItemsRefetch(); hideUpdateModal(); hideActionItemStatusModal(); - toast.success(t('successfulUpdation')); + toast.success(t('successfulUpdation') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -206,7 +206,7 @@ function eventActionItems(props: { eventId: string }): JSX.Element { actionItemsRefetch(); toggleDeleteModal(); hidePreviewModal(); - toast.success(t('successfulDeletion')); + toast.success(t('successfulDeletion') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); diff --git a/src/components/EventManagement/EventAgendaItems/EventAgendaItems.test.tsx b/src/components/EventManagement/EventAgendaItems/EventAgendaItems.test.tsx index b1b3ed6094..3bce7ad11e 100644 --- a/src/components/EventManagement/EventAgendaItems/EventAgendaItems.test.tsx +++ b/src/components/EventManagement/EventAgendaItems/EventAgendaItems.test.tsx @@ -14,7 +14,7 @@ import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import i18n from 'utils/i18nForTest'; -import { toast } from 'react-toastify'; +// import { toast } from 'react-toastify'; import { LocalizationProvider } from '@mui/x-date-pickers'; import { AdapterDayjs } from '@mui/x-date-pickers/AdapterDayjs'; @@ -26,7 +26,7 @@ import EventAgendaItems from './EventAgendaItems'; import { MOCKS, MOCKS_ERROR_QUERY, - MOCKS_ERROR_MUTATION, + // MOCKS_ERROR_MUTATION, } from './EventAgendaItemsMocks'; jest.mock('react-toastify', () => ({ @@ -41,6 +41,15 @@ jest.mock('react-router-dom', () => ({ useParams: () => ({ eventId: '123' }), })); +//temporarily fixes react-beautiful-dnd droppable method's depreciation error +//needs to be fixed in React 19 +jest.spyOn(console, 'error').mockImplementation((message) => { + if (message.includes('Support for defaultProps will be removed')) { + return; + } + console.error(message); +}); + async function wait(ms = 100): Promise { await act(() => { return new Promise((resolve) => { @@ -51,7 +60,7 @@ async function wait(ms = 100): Promise { const link = new StaticMockLink(MOCKS, true); const link2 = new StaticMockLink(MOCKS_ERROR_QUERY, true); -const link3 = new StaticMockLink(MOCKS_ERROR_MUTATION, true); +// const link3 = new StaticMockLink(MOCKS_ERROR_MUTATION, true); const translations = JSON.parse( JSON.stringify(i18n.getDataByLanguage('en')?.translation.agendaItems), diff --git a/src/components/EventManagement/EventAgendaItems/EventAgendaItems.tsx b/src/components/EventManagement/EventAgendaItems/EventAgendaItems.tsx index 73a074ba9f..b49ade4626 100644 --- a/src/components/EventManagement/EventAgendaItems/EventAgendaItems.tsx +++ b/src/components/EventManagement/EventAgendaItems/EventAgendaItems.tsx @@ -126,7 +126,7 @@ function EventAgendaItems(props: { eventId: string }): JSX.Element { }); hideCreateModal(); refetchAgendaItem(); - toast.success(t('agendaItemCreated')); + toast.success(t('agendaItemCreated') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -153,20 +153,20 @@ function EventAgendaItems(props: { eventId: string }): JSX.Element { // Show error message if there is an error loading data if (agendaItemError || agendaCategoryError) { + const errorMessage = + agendaCategoryError?.message || + (agendaItemError as Error)?.message || + 'Unknown error'; + return (
Error occurred while loading{' '} - {agendaCategoryError - ? 'Agenda Categories' - : agendaItemError && 'Agenda Items'} - Data + {agendaCategoryError ? 'Agenda Categories' : 'Agenda Items'} Data
- {agendaCategoryError - ? agendaCategoryError.message - : agendaItemError && (agendaItemError as Error).message} + {errorMessage}
diff --git a/src/components/EventRegistrantsModal/EventRegistrantsModal.tsx b/src/components/EventRegistrantsModal/EventRegistrantsModal.tsx index 85fa1ca426..bd2adc8a2c 100644 --- a/src/components/EventRegistrantsModal/EventRegistrantsModal.tsx +++ b/src/components/EventRegistrantsModal/EventRegistrantsModal.tsx @@ -81,11 +81,13 @@ export const EventRegistrantsModal = (props: ModalPropType): JSX.Element => { }, }) .then(() => { - toast.success(tCommon('addedSuccessfully', { item: 'Attendee' })); + toast.success( + tCommon('addedSuccessfully', { item: 'Attendee' }) as string, + ); attendeesRefetch(); // Refresh the list of attendees }) .catch((err) => { - toast.error(t('errorAddingAttendee')); + toast.error(t('errorAddingAttendee') as string); toast.error(err.message); }); }; @@ -100,11 +102,13 @@ export const EventRegistrantsModal = (props: ModalPropType): JSX.Element => { }, }) .then(() => { - toast.success(tCommon('removedSuccessfully', { item: 'Attendee' })); + toast.success( + tCommon('removedSuccessfully', { item: 'Attendee' }) as string, + ); attendeesRefetch(); // Refresh the list of attendees }) .catch((err) => { - toast.error(t('errorRemovingAttendee')); + toast.error(t('errorRemovingAttendee') as string); toast.error(err.message); }); }; diff --git a/src/components/LeftDrawer/LeftDrawer.test.tsx b/src/components/LeftDrawer/LeftDrawer.test.tsx index 3873d66dc2..6701d2d4bb 100644 --- a/src/components/LeftDrawer/LeftDrawer.test.tsx +++ b/src/components/LeftDrawer/LeftDrawer.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { fireEvent, render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-localstorage-mock'; @@ -63,102 +63,131 @@ afterEach(() => { }); describe('Testing Left Drawer component for SUPERADMIN', () => { - test('Component should be rendered properly', () => { - setItem('UserImage', ''); - setItem('UserImage', ''); - setItem('SuperAdmin', true); - setItem('FirstName', 'John'); - setItem('LastName', 'Doe'); + test('Component should be rendered properly', async () => { setItem('UserImage', ''); setItem('SuperAdmin', true); setItem('FirstName', 'John'); setItem('LastName', 'Doe'); - render( - - - - - - - , - ); + + await act(async () => { + render( + + + + + + + , + ); + }); expect(screen.getByText('My Organizations')).toBeInTheDocument(); expect(screen.getByText('Users')).toBeInTheDocument(); expect(screen.getByText('Community Profile')).toBeInTheDocument(); - expect(screen.getByText('Community Profile')).toBeInTheDocument(); expect(screen.getByText('Talawa Admin Portal')).toBeInTheDocument(); const orgsBtn = screen.getByTestId(/orgsBtn/i); const rolesBtn = screen.getByTestId(/rolesBtn/i); const communityProfileBtn = screen.getByTestId(/communityProfileBtn/i); - orgsBtn.click(); + await act(async () => { + orgsBtn.click(); + }); + expect( orgsBtn.className.includes('text-white btn btn-success'), ).toBeTruthy(); expect(rolesBtn.className.includes('text-secondary btn')).toBeTruthy(); - expect(rolesBtn.className.includes('text-secondary btn')).toBeTruthy(); expect( communityProfileBtn.className.includes('text-secondary btn'), ).toBeTruthy(); - // Send to roles screen - userEvent.click(rolesBtn); + await act(async () => { + userEvent.click(rolesBtn); + }); + expect(global.window.location.pathname).toContain('/users'); - userEvent.click(communityProfileBtn); + + await act(async () => { + userEvent.click(communityProfileBtn); + }); }); - test('Testing Drawer when hideDrawer is null', () => { + test('Testing Drawer when hideDrawer is null', async () => { const tempProps: InterfaceLeftDrawerProps = { ...props, hideDrawer: false, }; - render( - - - - - - - , - ); + await act(async () => { + render( + + + + + + + , + ); + }); }); - test('Testing Drawer when hideDrawer is false', () => { + + test('Testing Drawer when hideDrawer is false', async () => { const tempProps: InterfaceLeftDrawerProps = { ...props, hideDrawer: false, }; + + await act(async () => { + render( + + + + + + + , + ); + }); }); - test('Testing Drawer when the screen size is less than or equal to 820px', () => { + + test('Testing Drawer when the screen size is less than or equal to 820px', async () => { const tempProps: InterfaceLeftDrawerProps = { ...props, hideDrawer: false, }; resizeWindow(800); - render( - - - - - - - , - ); + + await act(async () => { + render( + + + + + + + , + ); + }); + expect(screen.getByText('My Organizations')).toBeInTheDocument(); expect(screen.getByText('Talawa Admin Portal')).toBeInTheDocument(); const orgsBtn = screen.getByTestId(/orgsBtn/i); - orgsBtn.click(); + await act(async () => { + orgsBtn.click(); + }); + expect( orgsBtn.className.includes('text-white btn btn-success'), ).toBeTruthy(); }); +}); - describe('Testing Left Drawer component for ADMIN', () => { - test('Components should be rendered properly', () => { +describe('Testing Left Drawer component for ADMIN', () => { + test('Components should be rendered properly', async () => { + await act(async () => { render( @@ -168,24 +197,30 @@ describe('Testing Left Drawer component for SUPERADMIN', () => { , ); + }); + + expect(screen.getByText('My Organizations')).toBeInTheDocument(); + expect(screen.getByText('Talawa Admin Portal')).toBeInTheDocument(); - expect(screen.getByText('My Organizations')).toBeInTheDocument(); - expect(screen.getByText('Talawa Admin Portal')).toBeInTheDocument(); + expect(screen.getAllByText(/admin/i)).toHaveLength(1); - expect(screen.getAllByText(/admin/i)).toHaveLength(1); - expect(screen.getAllByText(/admin/i)).toHaveLength(1); + const orgsBtn = screen.getByTestId(/orgsBtn/i); - const orgsBtn = screen.getByTestId(/orgsBtn/i); + await act(async () => { orgsBtn.click(); - expect( - orgsBtn.className.includes('text-white btn btn-success'), - ).toBeTruthy(); + }); - // These screens arent meant for admins so they should not be present - expect(screen.queryByTestId(/rolesBtn/i)).toBeNull(); + expect( + orgsBtn.className.includes('text-white btn btn-success'), + ).toBeTruthy(); + // These screens aren't meant for admins, so they should not be present + expect(screen.queryByTestId(/rolesBtn/i)).toBeNull(); + + await act(async () => { userEvent.click(orgsBtn); - expect(global.window.location.pathname).toContain('/orglist'); }); + + expect(global.window.location.pathname).toContain('/orglist'); }); }); diff --git a/src/components/LeftDrawerOrg/LeftDrawerOrg.test.tsx b/src/components/LeftDrawerOrg/LeftDrawerOrg.test.tsx index 58a7168552..7dc8330c84 100644 --- a/src/components/LeftDrawerOrg/LeftDrawerOrg.test.tsx +++ b/src/components/LeftDrawerOrg/LeftDrawerOrg.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { fireEvent, render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-localstorage-mock'; @@ -12,7 +12,6 @@ import { Provider } from 'react-redux'; import { MockedProvider } from '@apollo/react-testing'; import { store } from 'state/store'; import { ORGANIZATIONS_LIST } from 'GraphQl/Queries/Queries'; -import { act } from 'react-dom/test-utils'; import { StaticMockLink } from 'utils/StaticMockLink'; import { REVOKE_REFRESH_TOKEN } from 'GraphQl/Mutations/mutations'; import useLocalStorage from 'utils/useLocalstorage'; diff --git a/src/components/LoginPortalToggle/LoginPortalToggle.test.tsx b/src/components/LoginPortalToggle/LoginPortalToggle.test.tsx index ac971e52ea..327e6cccea 100644 --- a/src/components/LoginPortalToggle/LoginPortalToggle.test.tsx +++ b/src/components/LoginPortalToggle/LoginPortalToggle.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render } from '@testing-library/react'; +import React, { act } from 'react'; +import { render } from '@testing-library/react'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/MemberRequestCard/MemberRequestCard.test.tsx b/src/components/MemberRequestCard/MemberRequestCard.test.tsx index 121213e4e7..a38a046ea2 100644 --- a/src/components/MemberRequestCard/MemberRequestCard.test.tsx +++ b/src/components/MemberRequestCard/MemberRequestCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; @@ -16,7 +16,7 @@ const MOCKS = [ { request: { query: ACCEPT_ORGANIZATION_REQUEST_MUTATION, - variable: { id: '123' }, + variables: { id: '123' }, }, result: { data: { @@ -31,7 +31,7 @@ const MOCKS = [ { request: { query: REJECT_ORGANIZATION_REQUEST_MUTATION, - variable: { id: '234' }, + variables: { userid: '234' }, }, result: { data: { @@ -44,7 +44,9 @@ const MOCKS = [ }, }, ]; + const link = new StaticMockLink(MOCKS, true); + async function wait(ms = 100): Promise { await act(() => { return new Promise((resolve) => { @@ -55,7 +57,6 @@ async function wait(ms = 100): Promise { describe('Testing Member Request Card', () => { const props = { - key: '123', id: '1', memberName: 'John Doe', memberLocation: 'India', @@ -89,14 +90,13 @@ describe('Testing Member Request Card', () => { expect(screen.getByText(props.email)).toBeInTheDocument(); }); - it('Should render text elements when props value is not passed', async () => { + it('should render text elements when props value is not passed', async () => { global.confirm = (): boolean => false; render( { window.location.reload(); @@ -136,5 +135,4 @@ function memberRequestCard( ); } -export {}; -export default memberRequestCard; +export default MemberRequestCard; diff --git a/src/components/OrgActionItemCategories/OrgActionItemCategories.tsx b/src/components/OrgActionItemCategories/OrgActionItemCategories.tsx index 72b1fdd23c..ea86fcf965 100644 --- a/src/components/OrgActionItemCategories/OrgActionItemCategories.tsx +++ b/src/components/OrgActionItemCategories/OrgActionItemCategories.tsx @@ -83,7 +83,7 @@ const OrgActionItemCategories = (): JSX.Element => { setModalIsOpen(false); - toast.success(t('successfulCreation')); + toast.success(t('successfulCreation') as string); } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); @@ -96,7 +96,7 @@ const OrgActionItemCategories = (): JSX.Element => { const handleEdit = async (e: ChangeEvent): Promise => { e.preventDefault(); if (name === currName) { - toast.error(t('sameNameConflict')); // Show error if the name is the same + toast.error(t('sameNameConflict') as string); // Show error if the name is the same } else { try { await updateActionItemCategory({ @@ -111,7 +111,7 @@ const OrgActionItemCategories = (): JSX.Element => { refetch(); // Refetch the list of categories setModalIsOpen(false); // Close the modal - toast.success(t('successfulUpdation')); // Show success toast + toast.success(t('successfulUpdation') as string); // Show success toast } catch (error: unknown) { if (error instanceof Error) { toast.error(error.message); // Show error toast @@ -137,7 +137,9 @@ const OrgActionItemCategories = (): JSX.Element => { refetch(); // Refetch the list of categories toast.success( - disabledStatus ? t('categoryEnabled') : t('categoryDisabled'), + disabledStatus + ? (t('categoryEnabled') as string) + : (t('categoryDisabled') as string), ); // Show success toast } catch (error: unknown) { if (error instanceof Error) { diff --git a/src/components/OrgAdminListCard/OrgAdminListCard.test.tsx b/src/components/OrgAdminListCard/OrgAdminListCard.test.tsx index 282dedb330..7baea946d2 100644 --- a/src/components/OrgAdminListCard/OrgAdminListCard.test.tsx +++ b/src/components/OrgAdminListCard/OrgAdminListCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen, waitFor } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen, waitFor } from '@testing-library/react'; import type { RenderResult } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; diff --git a/src/components/OrgAdminListCard/OrgAdminListCard.tsx b/src/components/OrgAdminListCard/OrgAdminListCard.tsx index 418592819f..f8fc454823 100644 --- a/src/components/OrgAdminListCard/OrgAdminListCard.tsx +++ b/src/components/OrgAdminListCard/OrgAdminListCard.tsx @@ -44,7 +44,7 @@ function orgAdminListCard(props: InterfaceOrgPeopleListCardProps): JSX.Element { }, }); if (data) { - toast.success(t('adminRemoved')); + toast.success(t('adminRemoved') as string); setTimeout(() => { window.location.reload(); }, 2000); diff --git a/src/components/OrgListCard/OrgListCard.test.tsx b/src/components/OrgListCard/OrgListCard.test.tsx index 25d7f01ed1..4072265ea4 100644 --- a/src/components/OrgListCard/OrgListCard.test.tsx +++ b/src/components/OrgListCard/OrgListCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import 'jest-location-mock'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/OrgPeopleListCard/OrgPeopleListCard.test.tsx b/src/components/OrgPeopleListCard/OrgPeopleListCard.test.tsx index 57a6e0265f..7cee31107f 100644 --- a/src/components/OrgPeopleListCard/OrgPeopleListCard.test.tsx +++ b/src/components/OrgPeopleListCard/OrgPeopleListCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/OrgPeopleListCard/OrgPeopleListCard.tsx b/src/components/OrgPeopleListCard/OrgPeopleListCard.tsx index f22abc2c01..d24773c02a 100644 --- a/src/components/OrgPeopleListCard/OrgPeopleListCard.tsx +++ b/src/components/OrgPeopleListCard/OrgPeopleListCard.tsx @@ -56,7 +56,7 @@ function orgPeopleListCard( // If the mutation is successful, show a success message and reload the page /* istanbul ignore next */ if (data) { - toast.success(t('memberRemoved')); + toast.success(t('memberRemoved') as string); setTimeout(() => { window.location.reload(); }, 2000); diff --git a/src/components/OrgPostCard/OrgPostCard.test.tsx b/src/components/OrgPostCard/OrgPostCard.test.tsx index 863ed2822b..2c1f0fd824 100644 --- a/src/components/OrgPostCard/OrgPostCard.test.tsx +++ b/src/components/OrgPostCard/OrgPostCard.test.tsx @@ -112,7 +112,6 @@ describe('Testing Organization Post Card', () => { }); const props = { - key: '123', id: '12', postTitle: 'Event Info', postInfo: 'Time change', @@ -139,7 +138,7 @@ describe('Testing Organization Post Card', () => { const { getByAltText, getByTestId } = render( - + , ); @@ -152,7 +151,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -169,7 +168,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -188,7 +187,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -200,7 +199,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -269,7 +268,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -289,7 +288,6 @@ describe('Testing Organization Post Card', () => { }); test('Testing pin post functionality fail case', async () => { const props2 = { - key: '123', id: '', postTitle: 'Event Info', postInfo: 'Time change', @@ -301,7 +299,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -317,7 +315,7 @@ describe('Testing Organization Post Card', () => { - + , @@ -340,7 +338,6 @@ describe('Testing Organization Post Card', () => { }); test('Testing post delete functionality fail case', async () => { const props2 = { - key: '123', id: '', postTitle: 'Event Info', postInfo: 'Time change', @@ -353,7 +350,7 @@ describe('Testing Organization Post Card', () => { - + , @@ -371,7 +368,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -385,7 +382,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -398,7 +395,6 @@ describe('Testing Organization Post Card', () => { }); test('renders without "Read more" button when postInfo length is less than or equal to 43', () => { const props2 = { - key: '123', id: '12', postTitle: 'Event Info', postInfo: 'Lorem ipsum dolor sit amet', @@ -410,7 +406,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -419,7 +415,7 @@ describe('Testing Organization Post Card', () => { const link2 = new StaticMockLink(MOCKS, true); render( - + , ); userEvent.click(screen.getByAltText('image')); @@ -446,7 +442,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -480,7 +476,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -519,7 +515,7 @@ describe('Testing Organization Post Card', () => { render( - + , ); @@ -539,7 +535,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -558,7 +554,7 @@ describe('Testing Organization Post Card', () => { const { getByTestId } = render( - + , ); @@ -576,7 +572,6 @@ describe('Testing Organization Post Card', () => { }); test('for rendering when no image and no video is available', async () => { const props2 = { - key: '123', id: '', postTitle: 'Event Info', postInfo: 'Time change', @@ -589,7 +584,7 @@ describe('Testing Organization Post Card', () => { const { getByAltText } = render( - + , ); diff --git a/src/components/OrgPostCard/OrgPostCard.tsx b/src/components/OrgPostCard/OrgPostCard.tsx index be71622c7c..b7cc419d12 100644 --- a/src/components/OrgPostCard/OrgPostCard.tsx +++ b/src/components/OrgPostCard/OrgPostCard.tsx @@ -16,7 +16,7 @@ import { errorHandler } from 'utils/errorHandler'; import type { InterfacePostForm } from 'utils/interfaces'; import styles from './OrgPostCard.module.css'; interface InterfaceOrgPostCardProps { - key: string; + postID: string; id: string; postTitle: string; postInfo: string; @@ -25,9 +25,13 @@ interface InterfaceOrgPostCardProps { postVideo: string | null; pinned: boolean; } -export default function orgPostCard( +export default function OrgPostCard( props: InterfaceOrgPostCardProps, ): JSX.Element { + const { + postID, // Destructure the key prop from props + // ...rest // Spread the rest of the props + } = props; const [postformState, setPostFormState] = useState({ posttitle: '', postinfo: '', @@ -68,12 +72,13 @@ export default function orgPostCard( } }; const toggleShowEditModal = (): void => { + const { postTitle, postInfo, postPhoto, postVideo, pinned } = props; setPostFormState({ - posttitle: props.postTitle, - postinfo: props.postInfo, - postphoto: props.postPhoto, - postvideo: props.postVideo, - pinned: props.pinned, + posttitle: postTitle, + postinfo: postInfo, + postphoto: postPhoto, + postvideo: postVideo, + pinned: pinned, }); setPostPhotoUpdated(false); setPostVideoUpdated(false); @@ -128,13 +133,14 @@ export default function orgPostCard( } } function handleEditModal(): void { + const { postPhoto, postVideo } = props; setModalVisible(false); setMenuVisible(false); setShowEditModal(true); setPostFormState({ ...postformState, - postphoto: props.postPhoto, - postvideo: props.postVideo, + postphoto: postPhoto, + postvideo: postVideo, }); } function handleDeleteModal(): void { @@ -165,7 +171,7 @@ export default function orgPostCard( }, }); if (data) { - toast.success(t('postDeleted')); + toast.success(t('postDeleted') as string); toggleShowDeleteModal(); setTimeout(() => { window.location.reload(); @@ -203,7 +209,7 @@ export default function orgPostCard( }, }); if (data) { - toast.success(t('postUpdated')); + toast.success(t('postUpdated') as string); setTimeout(() => { window.location.reload(); }, 2000); @@ -216,7 +222,11 @@ export default function orgPostCard( }; return ( <> -
+
{ ...customFieldData, }, }); - toast.success(t('fieldSuccessMessage')); + toast.success(t('fieldSuccessMessage') as string); setCustomFieldData({ type: '', name: '' }); refetch(); } catch (error) { @@ -89,7 +89,7 @@ const OrgProfileFieldSettings = (): JSX.Element => { }, }); - toast.success(t('fieldRemovalSuccess')); + toast.success(t('fieldRemovalSuccess') as string); refetch(); } catch (error) { toast.error((error as Error).message); diff --git a/src/components/OrgUpdate/OrgUpdate.test.tsx b/src/components/OrgUpdate/OrgUpdate.test.tsx index 9c56a61d78..6304bb3ec9 100644 --- a/src/components/OrgUpdate/OrgUpdate.test.tsx +++ b/src/components/OrgUpdate/OrgUpdate.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; import { StaticMockLink } from 'utils/StaticMockLink'; diff --git a/src/components/OrgUpdate/OrgUpdate.tsx b/src/components/OrgUpdate/OrgUpdate.tsx index f33c758acd..6e0be28a56 100644 --- a/src/components/OrgUpdate/OrgUpdate.tsx +++ b/src/components/OrgUpdate/OrgUpdate.tsx @@ -144,7 +144,7 @@ function orgUpdate(props: InterfaceOrgUpdateProps): JSX.Element { // istanbul ignore next if (data) { refetch({ id: orgId }); - toast.success(t('successfulUpdated')); + toast.success(t('successfulUpdated') as string); } } catch (error: unknown) { errorHandler(t, error); diff --git a/src/components/OrganizationCard/OrganizationCard.test.tsx b/src/components/OrganizationCard/OrganizationCard.test.tsx index e0095d20b2..e4abf3a1fd 100644 --- a/src/components/OrganizationCard/OrganizationCard.test.tsx +++ b/src/components/OrganizationCard/OrganizationCard.test.tsx @@ -6,7 +6,6 @@ describe('Testing the Organization Card', () => { test('should render props and text elements test for the page component', () => { const props = { id: '123', - key: '456', image: 'https://via.placeholder.com/80', firstName: 'John', lastName: 'Doe', @@ -24,7 +23,6 @@ describe('Testing the Organization Card', () => { test('Should render text elements when props value is not passed', () => { const props = { id: '123', - key: '456', image: '', firstName: 'John', lastName: 'Doe', diff --git a/src/components/OrganizationCard/OrganizationCard.tsx b/src/components/OrganizationCard/OrganizationCard.tsx index 05f8712ec5..ae513eff5d 100644 --- a/src/components/OrganizationCard/OrganizationCard.tsx +++ b/src/components/OrganizationCard/OrganizationCard.tsx @@ -12,47 +12,45 @@ interface InterfaceOrganizationCardProps { /** * Component to display an organization's card with its image and owner details. * - * @param image - URL of the organization's image. - * @param id - Unique identifier for the organization. - * @param name - Name of the organization. - * @param lastName - Last name of the owner's name. - * @param firstName - First name of the owner's name. + * @param props - Properties for the organization card. * @returns JSX element representing the organization card. */ -function organizationCard(props: InterfaceOrganizationCardProps): JSX.Element { +function OrganizationCard(props: InterfaceOrganizationCardProps): JSX.Element { const uri = '/superorghome/i=' + props.id; return ( - <> - -
-
- {props.image ? ( - - ) : ( - - )} -
-

{props.name}

-
- Owner: - {props.firstName} - -   - {props.lastName} - -
-
+
+ + ); } -export {}; -export default organizationCard; +export default OrganizationCard; diff --git a/src/components/OrganizationCardStart/OrganizationCardStart.test.tsx b/src/components/OrganizationCardStart/OrganizationCardStart.test.tsx index 2a9f03a9be..dd65c8649e 100644 --- a/src/components/OrganizationCardStart/OrganizationCardStart.test.tsx +++ b/src/components/OrganizationCardStart/OrganizationCardStart.test.tsx @@ -6,12 +6,11 @@ describe('Testing the Organization Cards', () => { test('should render props and text elements test for the page component', () => { const props = { id: '123', - key: '456', image: 'https://via.placeholder.com/80', name: 'Sample', }; - render(); + render(); expect(screen.getByText(props.name)).toBeInTheDocument(); }); @@ -19,12 +18,11 @@ describe('Testing the Organization Cards', () => { test('Should render text elements when props value is not passed', () => { const props = { id: '123', - key: '456', image: '', name: 'Sample', }; - render(); + render(); expect(screen.getByText(props.name)).toBeInTheDocument(); }); diff --git a/src/components/ProfileDropdown/ProfileDropdown.test.tsx b/src/components/ProfileDropdown/ProfileDropdown.test.tsx index 0efef50591..82ce420aea 100644 --- a/src/components/ProfileDropdown/ProfileDropdown.test.tsx +++ b/src/components/ProfileDropdown/ProfileDropdown.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { BrowserRouter } from 'react-router-dom'; import ProfileDropdown from './ProfileDropdown'; diff --git a/src/components/RequestsTableItem/RequestsTableItem.test.tsx b/src/components/RequestsTableItem/RequestsTableItem.test.tsx index 866abc7f68..bbd895300b 100644 --- a/src/components/RequestsTableItem/RequestsTableItem.test.tsx +++ b/src/components/RequestsTableItem/RequestsTableItem.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { StaticMockLink } from 'utils/StaticMockLink'; import i18nForTest from 'utils/i18nForTest'; diff --git a/src/components/RequestsTableItem/RequestsTableItem.tsx b/src/components/RequestsTableItem/RequestsTableItem.tsx index 2b647a9ce2..07feb5d289 100644 --- a/src/components/RequestsTableItem/RequestsTableItem.tsx +++ b/src/components/RequestsTableItem/RequestsTableItem.tsx @@ -67,7 +67,7 @@ const RequestsTableItem = (props: Props): JSX.Element => { }); /* istanbul ignore next */ if (data) { - toast.success(t('acceptedSuccessfully')); + toast.success(t('acceptedSuccessfully') as string); resetAndRefetch(); } } catch (error: unknown) { @@ -95,7 +95,7 @@ const RequestsTableItem = (props: Props): JSX.Element => { }); /* istanbul ignore next */ if (data) { - toast.success(t('rejectedSuccessfully')); + toast.success(t('rejectedSuccessfully') as string); resetAndRefetch(); } } catch (error: unknown) { diff --git a/src/components/UserListCard/UserListCard.test.tsx b/src/components/UserListCard/UserListCard.test.tsx index 9f59bb92ce..e2ad552507 100644 --- a/src/components/UserListCard/UserListCard.test.tsx +++ b/src/components/UserListCard/UserListCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; @@ -14,7 +14,7 @@ const MOCKS = [ { request: { query: ADD_ADMIN_MUTATION, - variable: { userid: '784', orgid: '554' }, + variables: { userid: '784', orgid: '554' }, }, result: { data: { @@ -28,6 +28,7 @@ const MOCKS = [ }, ]; const link = new StaticMockLink(MOCKS, true); + async function wait(ms = 100): Promise { await act(() => { return new Promise((resolve) => { @@ -41,7 +42,6 @@ describe('Testing User List Card', () => { test('Should render props and text elements test for the page component', async () => { const props = { - key: 123, id: '456', }; @@ -49,7 +49,7 @@ describe('Testing User List Card', () => { - + , @@ -62,7 +62,6 @@ describe('Testing User List Card', () => { test('Should render text elements when props value is not passed', async () => { const props = { - key: 123, id: '456', }; @@ -70,7 +69,7 @@ describe('Testing User List Card', () => { - + , diff --git a/src/components/UserListCard/UserListCard.tsx b/src/components/UserListCard/UserListCard.tsx index 193569424f..76ad110ea1 100644 --- a/src/components/UserListCard/UserListCard.tsx +++ b/src/components/UserListCard/UserListCard.tsx @@ -47,7 +47,7 @@ function userListCard(props: InterfaceUserListCardProps): JSX.Element { /* istanbul ignore next */ if (data) { - toast.success(t('addedAsAdmin')); + toast.success(t('addedAsAdmin') as string); setTimeout(() => { window.location.reload(); }, 2000); diff --git a/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx b/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx index 4fcbb18070..65f5e40f76 100644 --- a/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx +++ b/src/components/UserPasswordUpdate/UserPasswordUpdate.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; @@ -20,7 +20,7 @@ const MOCKS = [ { request: { query: UPDATE_USER_PASSWORD_MUTATION, - variable: { + variables: { previousPassword: 'anshgoyal', newPassword: 'anshgoyalansh', confirmNewPassword: 'anshgoyalansh', @@ -49,15 +49,10 @@ async function wait(ms = 5): Promise { } describe('Testing User Password Update', () => { - const props = { - key: '123', - id: '1', - }; - const formData = { previousPassword: 'Palisadoes', newPassword: 'ThePalisadoesFoundation', - wrongPassword: 'This is wrong passoword', + wrongPassword: 'This is wrong password', confirmNewPassword: 'ThePalisadoesFoundation', }; @@ -67,7 +62,7 @@ describe('Testing User Password Update', () => { render( - + , ); @@ -102,7 +97,7 @@ describe('Testing User Password Update', () => { render( - + , ); @@ -117,7 +112,7 @@ describe('Testing User Password Update', () => { render( - + , ); diff --git a/src/components/UserPasswordUpdate/UserPasswordUpdate.tsx b/src/components/UserPasswordUpdate/UserPasswordUpdate.tsx index e1f8c64f0b..1ea75811c1 100644 --- a/src/components/UserPasswordUpdate/UserPasswordUpdate.tsx +++ b/src/components/UserPasswordUpdate/UserPasswordUpdate.tsx @@ -45,12 +45,12 @@ const UserUpdate: React.FC< !formState.newPassword || !formState.confirmNewPassword ) { - toast.error(t('passCantBeEmpty')); + toast.error(t('passCantBeEmpty') as string); return; } if (formState.newPassword !== formState.confirmNewPassword) { - toast.error(t('passNoMatch')); + toast.error(t('passNoMatch') as string); return; } @@ -64,7 +64,9 @@ const UserUpdate: React.FC< }); /* istanbul ignore next */ if (data) { - toast.success(tCommon('updatedSuccessfully', { item: 'Password' })); + toast.success( + tCommon('updatedSuccessfully', { item: 'Password' }) as string, + ); setTimeout(() => { window.location.reload(); }, 2000); diff --git a/src/components/UserPortal/CommentCard/CommentCard.test.tsx b/src/components/UserPortal/CommentCard/CommentCard.test.tsx index 35b81167fc..f02e5a606a 100644 --- a/src/components/UserPortal/CommentCard/CommentCard.test.tsx +++ b/src/components/UserPortal/CommentCard/CommentCard.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; diff --git a/src/components/UserPortal/CommentCard/CommentCard.tsx b/src/components/UserPortal/CommentCard/CommentCard.tsx index 026c2f1af3..9e8c46d241 100644 --- a/src/components/UserPortal/CommentCard/CommentCard.tsx +++ b/src/components/UserPortal/CommentCard/CommentCard.tsx @@ -92,7 +92,7 @@ function commentCard(props: InterfaceCommentCardProps): JSX.Element { } } catch (error: unknown) { /* istanbul ignore next */ - toast.error(error); + toast.error(error as string); } } else { try { @@ -109,7 +109,7 @@ function commentCard(props: InterfaceCommentCardProps): JSX.Element { } } catch (error: unknown) { /* istanbul ignore next */ - toast.error(error); + toast.error(error as string); } } }; diff --git a/src/components/UserPortal/ContactCard/ContactCard.test.tsx b/src/components/UserPortal/ContactCard/ContactCard.test.tsx index 7f55196bb0..03136387a7 100644 --- a/src/components/UserPortal/ContactCard/ContactCard.test.tsx +++ b/src/components/UserPortal/ContactCard/ContactCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/DonationCard/DonationCard.test.tsx b/src/components/UserPortal/DonationCard/DonationCard.test.tsx index 4e49ed8b26..cddf62dd6c 100644 --- a/src/components/UserPortal/DonationCard/DonationCard.test.tsx +++ b/src/components/UserPortal/DonationCard/DonationCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render } from '@testing-library/react'; +import React, { act } from 'react'; +import { render } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/EventCard/EventCard.tsx b/src/components/UserPortal/EventCard/EventCard.tsx index 3b9d328154..e93138ce42 100644 --- a/src/components/UserPortal/EventCard/EventCard.tsx +++ b/src/components/UserPortal/EventCard/EventCard.tsx @@ -103,7 +103,7 @@ function eventCard(props: InterfaceEventCardProps): JSX.Element { } } catch (error: unknown) { /* istanbul ignore next */ - toast.error(error); + toast.error(error as string); } } }; diff --git a/src/components/UserPortal/OrganizationCard/OrganizationCard.test.tsx b/src/components/UserPortal/OrganizationCard/OrganizationCard.test.tsx index dba4286290..5546166b9a 100644 --- a/src/components/UserPortal/OrganizationCard/OrganizationCard.test.tsx +++ b/src/components/UserPortal/OrganizationCard/OrganizationCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { fireEvent, render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/OrganizationCard/OrganizationCard.tsx b/src/components/UserPortal/OrganizationCard/OrganizationCard.tsx index b88e17ae95..e1c2c23beb 100644 --- a/src/components/UserPortal/OrganizationCard/OrganizationCard.tsx +++ b/src/components/UserPortal/OrganizationCard/OrganizationCard.tsx @@ -110,23 +110,23 @@ function organizationCard(props: InterfaceOrganizationCardProps): JSX.Element { organizationId: props.id, }, }); - toast.success(t('MembershipRequestSent')); + toast.success(t('MembershipRequestSent') as string); } else { await joinPublicOrganization({ variables: { organizationId: props.id, }, }); - toast.success(t('orgJoined')); + toast.success(t('orgJoined') as string); } refetch(); } catch (error: unknown) { /* istanbul ignore next */ if (error instanceof Error) { if (error.message === 'User is already a member') { - toast.error(t('AlreadyJoined')); + toast.error(t('AlreadyJoined') as string); } else { - toast.error(t('errorOccured')); + toast.error(t('errorOccured') as string); } } } diff --git a/src/components/UserPortal/OrganizationNavbar/OrganizationNavbar.test.tsx b/src/components/UserPortal/OrganizationNavbar/OrganizationNavbar.test.tsx index 027234b641..038ff626df 100644 --- a/src/components/UserPortal/OrganizationNavbar/OrganizationNavbar.test.tsx +++ b/src/components/UserPortal/OrganizationNavbar/OrganizationNavbar.test.tsx @@ -1,7 +1,7 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; import 'jest-localstorage-mock'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter, Router } from 'react-router-dom'; diff --git a/src/components/UserPortal/OrganizationSidebar/OrganizationSidebar.test.tsx b/src/components/UserPortal/OrganizationSidebar/OrganizationSidebar.test.tsx index c5fbcf335e..c08240462a 100644 --- a/src/components/UserPortal/OrganizationSidebar/OrganizationSidebar.test.tsx +++ b/src/components/UserPortal/OrganizationSidebar/OrganizationSidebar.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/PeopleCard/PeopleCard.test.tsx b/src/components/UserPortal/PeopleCard/PeopleCard.test.tsx index 5cadfa923b..fd5d6c7f93 100644 --- a/src/components/UserPortal/PeopleCard/PeopleCard.test.tsx +++ b/src/components/UserPortal/PeopleCard/PeopleCard.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render } from '@testing-library/react'; +import React, { act } from 'react'; +import { render } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/PostCard/PostCard.test.tsx b/src/components/UserPortal/PostCard/PostCard.test.tsx index 64f6670223..f7d9217308 100644 --- a/src/components/UserPortal/PostCard/PostCard.test.tsx +++ b/src/components/UserPortal/PostCard/PostCard.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; diff --git a/src/components/UserPortal/PostCard/PostCard.tsx b/src/components/UserPortal/PostCard/PostCard.tsx index 3eed167f65..f8fcdaebca 100644 --- a/src/components/UserPortal/PostCard/PostCard.tsx +++ b/src/components/UserPortal/PostCard/PostCard.tsx @@ -128,7 +128,7 @@ export default function postCard(props: InterfacePostCard): JSX.Element { } } catch (error: unknown) { /* istanbul ignore next */ - toast.error(error); + toast.error(error as string); } } else { try { @@ -144,7 +144,7 @@ export default function postCard(props: InterfacePostCard): JSX.Element { } } catch (error: unknown) { /* istanbul ignore next */ - toast.error(error); + toast.error(error as string); } } }; @@ -245,7 +245,7 @@ export default function postCard(props: InterfacePostCard): JSX.Element { props.fetchPosts(); // Refresh the posts toggleEditPost(); - toast.success(tCommon('updatedSuccessfully', { item: 'Post' })); + toast.success(tCommon('updatedSuccessfully', { item: 'Post' }) as string); } catch (error: unknown) { /* istanbul ignore next */ errorHandler(t, error); diff --git a/src/components/UserPortal/PromotedPost/PromotedPost.test.tsx b/src/components/UserPortal/PromotedPost/PromotedPost.test.tsx index b5d753551c..6ec8ec5de7 100644 --- a/src/components/UserPortal/PromotedPost/PromotedPost.test.tsx +++ b/src/components/UserPortal/PromotedPost/PromotedPost.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, waitFor } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, waitFor } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/Register/Register.test.tsx b/src/components/UserPortal/Register/Register.test.tsx index cdcfb9ebc0..f9929a0588 100644 --- a/src/components/UserPortal/Register/Register.test.tsx +++ b/src/components/UserPortal/Register/Register.test.tsx @@ -1,6 +1,6 @@ import type { SetStateAction } from 'react'; -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import userEvent from '@testing-library/user-event'; import { I18nextProvider } from 'react-i18next'; diff --git a/src/components/UserPortal/Register/Register.tsx b/src/components/UserPortal/Register/Register.tsx index 952acf7987..11a810c955 100644 --- a/src/components/UserPortal/Register/Register.tsx +++ b/src/components/UserPortal/Register/Register.tsx @@ -57,11 +57,11 @@ export default function register(props: InterfaceRegisterProps): JSX.Element { registerVariables.lastName ) ) { - toast.error(t('invalidDetailsMessage')); // Error if fields are missing + toast.error(t('invalidDetailsMessage') as string); // Error if fields are missing } else if ( registerVariables.password !== registerVariables.confirmPassword ) { - toast.error(t('passwordNotMatch')); // Error if passwords do not match + toast.error(t('passwordNotMatch') as string); // Error if passwords do not match } else { try { await registerMutation({ @@ -73,7 +73,7 @@ export default function register(props: InterfaceRegisterProps): JSX.Element { }, }); - toast.success(t('afterRegister')); // Success message + toast.success(t('afterRegister') as string); // Success message // Reset form fields /* istanbul ignore next */ diff --git a/src/components/UserPortal/StartPostModal/StartPostModal.test.tsx b/src/components/UserPortal/StartPostModal/StartPostModal.test.tsx index 5a2e815af4..5c436705cd 100644 --- a/src/components/UserPortal/StartPostModal/StartPostModal.test.tsx +++ b/src/components/UserPortal/StartPostModal/StartPostModal.test.tsx @@ -1,7 +1,7 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; import type { RenderResult } from '@testing-library/react'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import userEvent from '@testing-library/user-event'; import { CREATE_POST_MUTATION } from 'GraphQl/Mutations/mutations'; diff --git a/src/components/UserPortal/StartPostModal/StartPostModal.tsx b/src/components/UserPortal/StartPostModal/StartPostModal.tsx index 060963865a..d48b8425fa 100644 --- a/src/components/UserPortal/StartPostModal/StartPostModal.tsx +++ b/src/components/UserPortal/StartPostModal/StartPostModal.tsx @@ -93,7 +93,7 @@ const startPostModal = ({ /* istanbul ignore next */ if (data) { toast.dismiss(); - toast.success(t('postNowVisibleInFeed')); + toast.success(t('postNowVisibleInFeed') as string); fetchPosts(); handleHide(); } diff --git a/src/components/UserPortal/UserNavbar/UserNavbar.test.tsx b/src/components/UserPortal/UserNavbar/UserNavbar.test.tsx index 59e3585aa0..8c3447f25a 100644 --- a/src/components/UserPortal/UserNavbar/UserNavbar.test.tsx +++ b/src/components/UserPortal/UserNavbar/UserNavbar.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; diff --git a/src/components/UserPortal/UserSidebar/UserSidebar.test.tsx b/src/components/UserPortal/UserSidebar/UserSidebar.test.tsx index 2984604951..7218bc745c 100644 --- a/src/components/UserPortal/UserSidebar/UserSidebar.test.tsx +++ b/src/components/UserPortal/UserSidebar/UserSidebar.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import type { RenderResult } from '@testing-library/react'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen, waitFor } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; @@ -19,14 +19,17 @@ import useLocalStorage from 'utils/useLocalstorage'; const { setItem } = useLocalStorage(); const resizeWindow = (width: number): void => { - window.innerWidth = width; - fireEvent(window, new Event('resize')); + act(() => { + window.innerWidth = width; + fireEvent(window, new Event('resize')); + }); }; const props = { hideDrawer: true, setHideDrawer: jest.fn(), }; + const MOCKS = [ { request: { @@ -366,38 +369,48 @@ describe('Testing UserSidebar Component [User Portal]', () => { }); test('Component should be rendered properly', async () => { - renderUserSidebar('properId', link); - await wait(); + await act(async () => { + renderUserSidebar('properId', link); + await wait(); + }); }); test('Component should be rendered properly when userImage is present', async () => { - renderUserSidebar('imagePresent', link); - await wait(); + await act(async () => { + renderUserSidebar('imagePresent', link); + await wait(); + }); }); test('Component should be rendered properly when organizationImage is present', async () => { - renderUserSidebar('imagePresent', link); - await wait(); + await act(async () => { + renderUserSidebar('imagePresent', link); + await wait(); + }); }); test('Component should be rendered properly when joinedOrganizations list is empty', async () => { - renderUserSidebar('orgEmpty', link); - await wait(); + await act(async () => { + renderUserSidebar('orgEmpty', link); + await wait(); + }); }); - test('Testing Drawer when the screen size is less than or equal to 820px', () => { - resizeWindow(800); - render( - - - - - - - - - , - ); + test('Testing Drawer when the screen size is less than or equal to 820px', async () => { + await act(async () => { + resizeWindow(800); + render( + + + + + + + + + , + ); + }); expect(screen.getByText('My Organizations')).toBeInTheDocument(); expect(screen.getByText('Settings')).toBeInTheDocument(); expect(screen.getByText('Talawa User Portal')).toBeInTheDocument(); @@ -405,13 +418,21 @@ describe('Testing UserSidebar Component [User Portal]', () => { const orgsBtn = screen.getAllByTestId(/orgsBtn/i); - orgsBtn[0].click(); - expect( - orgsBtn[0].className.includes('text-white btn btn-success'), - ).toBeTruthy(); - settingsBtn.click(); - expect( - settingsBtn.className.includes('text-white btn btn-success'), - ).toBeTruthy(); + act(() => { + orgsBtn[0].click(); + }); + await waitFor(() => + expect( + orgsBtn[0].className.includes('text-white btn btn-success'), + ).toBeTruthy(), + ); + act(() => { + settingsBtn.click(); + }); + await waitFor(() => + expect( + settingsBtn.className.includes('text-white btn btn-success'), + ).toBeTruthy(), + ); }); }); diff --git a/src/components/UserPortal/UserSidebarOrg/UserSidebarOrg.test.tsx b/src/components/UserPortal/UserSidebarOrg/UserSidebarOrg.test.tsx index d9d70478b3..c187dabb55 100644 --- a/src/components/UserPortal/UserSidebarOrg/UserSidebarOrg.test.tsx +++ b/src/components/UserPortal/UserSidebarOrg/UserSidebarOrg.test.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +import React, { act } from 'react'; import { fireEvent, render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-localstorage-mock'; @@ -12,7 +12,6 @@ import { Provider } from 'react-redux'; import { MockedProvider } from '@apollo/react-testing'; import { store } from 'state/store'; import { ORGANIZATIONS_LIST } from 'GraphQl/Queries/Queries'; -import { act } from 'react-dom/test-utils'; import { StaticMockLink } from 'utils/StaticMockLink'; import { REVOKE_REFRESH_TOKEN } from 'GraphQl/Mutations/mutations'; import useLocalStorage from 'utils/useLocalstorage'; diff --git a/src/components/UsersTableItem/UserTableItem.test.tsx b/src/components/UsersTableItem/UserTableItem.test.tsx index bfa0a31c86..e87b41f7f2 100644 --- a/src/components/UsersTableItem/UserTableItem.test.tsx +++ b/src/components/UsersTableItem/UserTableItem.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen } from '@testing-library/react'; import { I18nextProvider } from 'react-i18next'; import { toast } from 'react-toastify'; import { StaticMockLink } from 'utils/StaticMockLink'; diff --git a/src/components/UsersTableItem/UsersTableItem.tsx b/src/components/UsersTableItem/UsersTableItem.tsx index 9d6ec65187..9e94b8a9f5 100644 --- a/src/components/UsersTableItem/UsersTableItem.tsx +++ b/src/components/UsersTableItem/UsersTableItem.tsx @@ -56,7 +56,9 @@ const UsersTableItem = (props: Props): JSX.Element => { }, }); if (data) { - toast.success(tCommon('removedSuccessfully', { item: 'User' })); + toast.success( + tCommon('removedSuccessfully', { item: 'User' }) as string, + ); resetAndRefetch(); } } catch (error: unknown) { @@ -77,7 +79,7 @@ const UsersTableItem = (props: Props): JSX.Element => { }, }); if (data) { - toast.success(t('roleUpdated')); + toast.success(t('roleUpdated') as string); resetAndRefetch(); } } catch (error: unknown) { diff --git a/src/components/Venues/VenueModal.test.tsx b/src/components/Venues/VenueModal.test.tsx index 652816adaf..b299c8ff20 100644 --- a/src/components/Venues/VenueModal.test.tsx +++ b/src/components/Venues/VenueModal.test.tsx @@ -1,7 +1,7 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; import type { RenderResult } from '@testing-library/react'; -import { act, render, screen, fireEvent } from '@testing-library/react'; +import { render, screen, fireEvent } from '@testing-library/react'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import 'jest-location-mock'; diff --git a/src/components/Venues/VenueModal.tsx b/src/components/Venues/VenueModal.tsx index fd672d4ea2..476964b910 100644 --- a/src/components/Venues/VenueModal.tsx +++ b/src/components/Venues/VenueModal.tsx @@ -77,13 +77,13 @@ const VenueModal = ({ */ const handleSubmit = useCallback(async () => { if (formState.name.trim().length === 0) { - toast.error(t('venueTitleError')); + toast.error(t('venueTitleError') as string); return; } const capacityNum = parseInt(formState.capacity); if (isNaN(capacityNum) || capacityNum <= 0) { - toast.error(t('venueCapacityError')); + toast.error(t('venueCapacityError') as string); return; } @@ -100,7 +100,9 @@ const VenueModal = ({ }); /* istanbul ignore next */ if (data) { - toast.success(edit ? t('venueUpdated') : t('venueAdded')); + toast.success( + edit ? (t('venueUpdated') as string) : (t('venueAdded') as string), + ); refetchVenues(); onHide(); setFormState({ diff --git a/src/components/plugins/DummyPlugin/DummyPlugin.test.jsx b/src/components/plugins/DummyPlugin/DummyPlugin.test.jsx index ae792966fe..e1abb52a1e 100644 --- a/src/components/plugins/DummyPlugin/DummyPlugin.test.jsx +++ b/src/components/plugins/DummyPlugin/DummyPlugin.test.jsx @@ -1,15 +1,16 @@ +import React from 'react'; import { MockedProvider } from '@apollo/react-testing'; import { render } from '@testing-library/react'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import { I18nextProvider } from 'react-i18next'; -import React from 'react'; - import { store } from 'state/store'; import DummyPlugin from './DummyPlugin'; import i18nForTest from 'utils/i18nForTest'; import { StaticMockLink } from 'utils/StaticMockLink'; + const link = new StaticMockLink([], true); + describe('Testing dummy plugin', () => { test('should render props and text elements test for the page component', () => { const { getByText } = render( @@ -21,7 +22,7 @@ describe('Testing dummy plugin', () => { - + , ); expect(getByText(/Welcome to the Dummy Plugin!/i)).toBeInTheDocument(); diff --git a/src/components/plugins/DummyPlugin/DummyPlugin.tsx b/src/components/plugins/DummyPlugin/DummyPlugin.tsx index d3c8ed8d41..5b837ea076 100644 --- a/src/components/plugins/DummyPlugin/DummyPlugin.tsx +++ b/src/components/plugins/DummyPlugin/DummyPlugin.tsx @@ -9,7 +9,7 @@ import AddOn from 'components/AddOn/AddOn'; * * @returns JSX.Element - Renders the `AddOn` component containing a welcome message. */ -function dummyPlugin(): JSX.Element { +function DummyPlugin(): JSX.Element { return (
Welcome to the Dummy Plugin!
@@ -17,8 +17,4 @@ function dummyPlugin(): JSX.Element { ); } -dummyPlugin.defaultProps = {}; - -dummyPlugin.propTypes = {}; - -export default dummyPlugin; +export default DummyPlugin; diff --git a/src/components/plugins/DummyPlugin2/DummyPlugin2.test.jsx b/src/components/plugins/DummyPlugin2/DummyPlugin2.test.jsx index bcec97367c..c2cfe03a1e 100644 --- a/src/components/plugins/DummyPlugin2/DummyPlugin2.test.jsx +++ b/src/components/plugins/DummyPlugin2/DummyPlugin2.test.jsx @@ -5,14 +5,14 @@ import { BrowserRouter } from 'react-router-dom'; import { store } from 'state/store'; import DummyPlugin2 from './DummyPlugin2'; -describe('Testing dummy plugin 2', () => { - test('should render props and text elements test for the page component', () => { +describe('Testing DummyPlugin2', () => { + test('should render DummyPlugin2 component', () => { render( - + , ); }); }); diff --git a/src/components/plugins/DummyPlugin2/DummyPlugin2.tsx b/src/components/plugins/DummyPlugin2/DummyPlugin2.tsx index 9d7e5e4498..dca6d63ee3 100644 --- a/src/components/plugins/DummyPlugin2/DummyPlugin2.tsx +++ b/src/components/plugins/DummyPlugin2/DummyPlugin2.tsx @@ -7,12 +7,8 @@ import React from 'react'; * This component currently does not have any additional functionality * or properties. */ -function dummyPlugin2(): JSX.Element { +function DummyPlugin2(): JSX.Element { return
; } -dummyPlugin2.defaultProps = {}; - -dummyPlugin2.propTypes = {}; - -export default dummyPlugin2; +export default DummyPlugin2; diff --git a/src/index.tsx b/src/index.tsx index 44cb0578b6..7a1bbeba59 100644 --- a/src/index.tsx +++ b/src/index.tsx @@ -1,5 +1,5 @@ import React, { Suspense } from 'react'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import { BrowserRouter } from 'react-router-dom'; import type { NormalizedCacheObject } from '@apollo/client'; import { @@ -131,7 +131,10 @@ const client: ApolloClient = new ApolloClient({ }); const fallbackLoader =
; -ReactDOM.render( +const container = document.getElementById('root'); +const root = createRoot(container!); // Note the use of '!' is to assert the container is not null + +root.render( @@ -146,5 +149,4 @@ ReactDOM.render( , - document.getElementById('root'), ); diff --git a/src/screens/BlockUser/BlockUser.test.tsx b/src/screens/BlockUser/BlockUser.test.tsx index bb490fbe3f..c851470d9b 100644 --- a/src/screens/BlockUser/BlockUser.test.tsx +++ b/src/screens/BlockUser/BlockUser.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { BLOCK_USER_MUTATION, diff --git a/src/screens/BlockUser/BlockUser.tsx b/src/screens/BlockUser/BlockUser.tsx index f32ffe6091..1f36257bb1 100644 --- a/src/screens/BlockUser/BlockUser.tsx +++ b/src/screens/BlockUser/BlockUser.tsx @@ -1,5 +1,5 @@ import { useMutation, useQuery } from '@apollo/client'; -import React, { useEffect, useState } from 'react'; +import React, { useEffect, useState, useCallback } from 'react'; import { Dropdown, Form, Table } from 'react-bootstrap'; import Button from 'react-bootstrap/Button'; import { toast } from 'react-toastify'; @@ -83,90 +83,99 @@ const Requests = (): JSX.Element => { return; } - if (showBlockedMembers == false) { - setMembersData(memberData?.organizationsMemberConnection.edges); + if (!showBlockedMembers) { + setMembersData(memberData?.organizationsMemberConnection.edges || []); } else { const blockUsers = memberData?.organizationsMemberConnection.edges.filter( (user: InterfaceMember) => user.organizationsBlockedBy.some((org) => org._id === currentUrl), ); - setMembersData(blockUsers); + setMembersData(blockUsers || []); } - }, [memberData, showBlockedMembers]); + }, [memberData, showBlockedMembers, currentUrl]); // Handler for blocking a user - const handleBlockUser = async (userId: string): Promise => { - try { - const { data } = await blockUser({ - variables: { - userId, - orgId: currentUrl, - }, - }); - /* istanbul ignore next */ - if (data) { - toast.success(t('blockedSuccessfully')); - memberRefetch(); + const handleBlockUser = useCallback( + async (userId: string): Promise => { + try { + const { data } = await blockUser({ + variables: { + userId, + orgId: currentUrl, + }, + }); + /* istanbul ignore next */ + if (data) { + toast.success(t('blockedSuccessfully') as string); + memberRefetch(); + } + } catch (error: unknown) { + /* istanbul ignore next */ + errorHandler(t, error); } - } catch (error: unknown) { - /* istanbul ignore next */ - errorHandler(t, error); - } - }; + }, + [blockUser, currentUrl, memberRefetch, t], + ); // Handler for unblocking a user - const handleUnBlockUser = async (userId: string): Promise => { - try { - const { data } = await unBlockUser({ - variables: { - userId, - orgId: currentUrl, - }, - }); - /* istanbul ignore next */ - if (data) { - toast.success(t('Un-BlockedSuccessfully')); - memberRefetch(); + const handleUnBlockUser = useCallback( + async (userId: string): Promise => { + try { + const { data } = await unBlockUser({ + variables: { + userId, + orgId: currentUrl, + }, + }); + /* istanbul ignore next */ + if (data) { + toast.success(t('Un-BlockedSuccessfully') as string); + memberRefetch(); + } + } catch (error: unknown) { + /* istanbul ignore next */ + errorHandler(t, error); } - } catch (error: unknown) { - /* istanbul ignore next */ - errorHandler(t, error); - } - }; + }, + [unBlockUser, currentUrl, memberRefetch, t], + ); // Display error if member query fails - /* istanbul ignore next */ - if (memberError) { - toast.error(memberError.message); - } + useEffect(() => { + if (memberError) { + toast.error(memberError.message); + } + }, [memberError]); // Search handler - const handleSearch = (value: string): void => { - setSearchByName(value); - memberRefetch({ - orgId: currentUrl, - firstName_contains: searchByFirstName ? value : '', - lastName_contains: searchByFirstName ? '' : value, - }); - }; + const handleSearch = useCallback( + (value: string): void => { + setSearchByName(value); + memberRefetch({ + orgId: currentUrl, + firstName_contains: searchByFirstName ? value : '', + lastName_contains: searchByFirstName ? '' : value, + }); + }, + [searchByFirstName, memberRefetch, currentUrl], + ); // Search by Enter key - const handleSearchByEnter = ( - e: React.KeyboardEvent, - ): void => { - if (e.key === 'Enter') { - const { value } = e.currentTarget; - handleSearch(value); - } - }; + const handleSearchByEnter = useCallback( + (e: React.KeyboardEvent): void => { + if (e.key === 'Enter') { + const { value } = e.currentTarget; + handleSearch(value); + } + }, + [handleSearch], + ); // Search button click handler - const handleSearchByBtnClick = (): void => { - const inputValue = - (document.getElementById('searchBlockedUsers') as HTMLInputElement) - ?.value || ''; + const handleSearchByBtnClick = useCallback((): void => { + const inputValue = searchByName; handleSearch(inputValue); - }; + }, [handleSearch, searchByName]); // Header titles for the table const headerTitles: string[] = [ @@ -195,6 +204,8 @@ const Requests = (): JSX.Element => { data-testid="searchByName" autoComplete="off" required + value={searchByName} + onChange={(e) => setSearchByName(e.target.value)} onKeyUp={handleSearchByEnter} />
{/* Table */} - {loadingMembers == false && + {loadingMembers === false && membersData.length === 0 && searchByName.length > 0 ? (
@@ -269,7 +280,7 @@ const Requests = (): JSX.Element => { {tCommon('noResultsFoundFor')} "{searchByName}"
- ) : loadingMembers == false && membersData.length === 0 ? ( + ) : loadingMembers === false && membersData.length === 0 ? (

{t('noSpammerFound')}

diff --git a/src/screens/CommunityProfile/CommunityProfile.test.tsx b/src/screens/CommunityProfile/CommunityProfile.test.tsx index 033c48464f..d7e056caa4 100644 --- a/src/screens/CommunityProfile/CommunityProfile.test.tsx +++ b/src/screens/CommunityProfile/CommunityProfile.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-localstorage-mock'; import 'jest-location-mock'; @@ -213,53 +213,53 @@ describe('Testing Community Profile Screen', () => { , ); - await wait(); + }); + await wait(); - const communityName = screen.getByPlaceholderText(/Community Name/i); - const websiteLink = screen.getByPlaceholderText(/Website Link/i); - const logo = screen.getByTestId(/fileInput/i); - const facebook = screen.getByTestId(/facebook/i); - const instagram = screen.getByTestId(/instagram/i); - const X = screen.getByTestId(/X/i); - const linkedIn = screen.getByTestId(/linkedIn/i); - const github = screen.getByTestId(/github/i); - const youtube = screen.getByTestId(/youtube/i); - const reddit = screen.getByTestId(/reddit/i); - const slack = screen.getByTestId(/slack/i); - const saveChangesBtn = screen.getByTestId(/saveChangesBtn/i); - const resetChangeBtn = screen.getByTestId(/resetChangesBtn/i); + const communityName = screen.getByPlaceholderText(/Community Name/i); + const websiteLink = screen.getByPlaceholderText(/Website Link/i); + const logo = screen.getByTestId(/fileInput/i); + const facebook = screen.getByTestId(/facebook/i); + const instagram = screen.getByTestId(/instagram/i); + const X = screen.getByTestId(/X/i); + const linkedIn = screen.getByTestId(/linkedIn/i); + const github = screen.getByTestId(/github/i); + const youtube = screen.getByTestId(/youtube/i); + const reddit = screen.getByTestId(/reddit/i); + const slack = screen.getByTestId(/slack/i); + const saveChangesBtn = screen.getByTestId(/saveChangesBtn/i); + const resetChangeBtn = screen.getByTestId(/resetChangesBtn/i); - userEvent.type(communityName, profileVariables.name); - userEvent.type(websiteLink, profileVariables.websiteLink); - userEvent.type(facebook, profileVariables.socialUrl); - userEvent.type(instagram, profileVariables.socialUrl); - userEvent.type(X, profileVariables.socialUrl); - userEvent.type(linkedIn, profileVariables.socialUrl); - userEvent.type(github, profileVariables.socialUrl); - userEvent.type(youtube, profileVariables.socialUrl); - userEvent.type(reddit, profileVariables.socialUrl); - userEvent.type(slack, profileVariables.socialUrl); - userEvent.upload(logo, profileVariables.logo); - await wait(); + userEvent.type(communityName, profileVariables.name); + userEvent.type(websiteLink, profileVariables.websiteLink); + userEvent.type(facebook, profileVariables.socialUrl); + userEvent.type(instagram, profileVariables.socialUrl); + userEvent.type(X, profileVariables.socialUrl); + userEvent.type(linkedIn, profileVariables.socialUrl); + userEvent.type(github, profileVariables.socialUrl); + userEvent.type(youtube, profileVariables.socialUrl); + userEvent.type(reddit, profileVariables.socialUrl); + userEvent.type(slack, profileVariables.socialUrl); + userEvent.upload(logo, profileVariables.logo); + await wait(); - expect(communityName).toHaveValue(profileVariables.name); - expect(websiteLink).toHaveValue(profileVariables.websiteLink); - // expect(logo).toBeTruthy(); - expect(facebook).toHaveValue(profileVariables.socialUrl); - expect(instagram).toHaveValue(profileVariables.socialUrl); - expect(X).toHaveValue(profileVariables.socialUrl); - expect(linkedIn).toHaveValue(profileVariables.socialUrl); - expect(github).toHaveValue(profileVariables.socialUrl); - expect(youtube).toHaveValue(profileVariables.socialUrl); - expect(reddit).toHaveValue(profileVariables.socialUrl); - expect(slack).toHaveValue(profileVariables.socialUrl); - expect(saveChangesBtn).not.toBeDisabled(); - expect(resetChangeBtn).not.toBeDisabled(); - await wait(); + expect(communityName).toHaveValue(profileVariables.name); + expect(websiteLink).toHaveValue(profileVariables.websiteLink); + // expect(logo).toBeTruthy(); + expect(facebook).toHaveValue(profileVariables.socialUrl); + expect(instagram).toHaveValue(profileVariables.socialUrl); + expect(X).toHaveValue(profileVariables.socialUrl); + expect(linkedIn).toHaveValue(profileVariables.socialUrl); + expect(github).toHaveValue(profileVariables.socialUrl); + expect(youtube).toHaveValue(profileVariables.socialUrl); + expect(reddit).toHaveValue(profileVariables.socialUrl); + expect(slack).toHaveValue(profileVariables.socialUrl); + expect(saveChangesBtn).not.toBeDisabled(); + expect(resetChangeBtn).not.toBeDisabled(); + await wait(); - userEvent.click(saveChangesBtn); - await wait(); - }); + userEvent.click(saveChangesBtn); + await wait(); }); test('If the queried data has some fields null then the input field should be empty', async () => { diff --git a/src/screens/CommunityProfile/CommunityProfile.tsx b/src/screens/CommunityProfile/CommunityProfile.tsx index 2f2d956b55..02f8b8aca4 100644 --- a/src/screens/CommunityProfile/CommunityProfile.tsx +++ b/src/screens/CommunityProfile/CommunityProfile.tsx @@ -146,7 +146,7 @@ const CommunityProfile = (): JSX.Element => { }, }, }); - toast.success(t('profileChangedMsg')); + toast.success(t('profileChangedMsg') as string); } catch (error: unknown) { /* istanbul ignore next */ errorHandler(t, error as Error); @@ -179,7 +179,7 @@ const CommunityProfile = (): JSX.Element => { resetPreLoginImageryId: preLoginData?._id, }, }); - toast.success(t(`resetData`)); + toast.success(t(`resetData`) as string); } catch (error: unknown) { /* istanbul ignore next */ errorHandler(t, error as Error); diff --git a/src/screens/EventManagement/EventManagement.tsx b/src/screens/EventManagement/EventManagement.tsx index adcc536a08..74dfc806ce 100644 --- a/src/screens/EventManagement/EventManagement.tsx +++ b/src/screens/EventManagement/EventManagement.tsx @@ -139,14 +139,13 @@ const EventManagement = (): JSX.Element => { const props = { variant, className, - key: value, size: 'sm' as 'sm' | 'lg', onClick: () => handleClick(value), 'data-testid': `${value}Btn`, }; return ( - diff --git a/src/screens/ForgotPassword/ForgotPassword.test.tsx b/src/screens/ForgotPassword/ForgotPassword.test.tsx index eb62cb0178..b20dfbf767 100644 --- a/src/screens/ForgotPassword/ForgotPassword.test.tsx +++ b/src/screens/ForgotPassword/ForgotPassword.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen } from '@testing-library/react'; +import { render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-localstorage-mock'; import 'jest-location-mock'; diff --git a/src/screens/ForgotPassword/ForgotPassword.tsx b/src/screens/ForgotPassword/ForgotPassword.tsx index 06761879a5..a2e638fd59 100644 --- a/src/screens/ForgotPassword/ForgotPassword.tsx +++ b/src/screens/ForgotPassword/ForgotPassword.tsx @@ -92,17 +92,17 @@ const ForgotPassword = (): JSX.Element => { if (data) { setItem('otpToken', data.otp.otpToken); - toast.success(t('OTPsent')); + toast.success(t('OTPsent') as string); setShowEnterEmail(false); } } catch (error: unknown) { if (error instanceof Error) { if (error.message === 'User not found') { - toast.warn(tErrors('emailNotRegistered')); + toast.warn(tErrors('emailNotRegistered') as string); } else if (error.message === 'Failed to fetch') { - toast.error(tErrors('talawaApiUnavailable')); + toast.error(tErrors('talawaApiUnavailable') as string); } else { - toast.error(tErrors('errorSendingMail')); + toast.error(tErrors('errorSendingMail') as string); } } } @@ -120,7 +120,7 @@ const ForgotPassword = (): JSX.Element => { const { userOtp, newPassword, confirmNewPassword } = forgotPassFormData; if (newPassword !== confirmNewPassword) { - toast.error(t('passwordMismatches')); + toast.error(t('passwordMismatches') as string); return; } @@ -141,7 +141,7 @@ const ForgotPassword = (): JSX.Element => { /* istanbul ignore next */ if (data) { - toast.success(t('passwordChanges')); + toast.success(t('passwordChanges') as string); setShowEnterEmail(true); setForgotPassFormData({ userOtp: '', diff --git a/src/screens/FundCampaignPledge/PledgeDeleteModal.tsx b/src/screens/FundCampaignPledge/PledgeDeleteModal.tsx index 2ecf6c7c23..288baa16ef 100644 --- a/src/screens/FundCampaignPledge/PledgeDeleteModal.tsx +++ b/src/screens/FundCampaignPledge/PledgeDeleteModal.tsx @@ -61,7 +61,7 @@ const PledgeDeleteModal: React.FC = ({ }); refetchPledge(); hide(); - toast.success(t('pledgeDeleted')); + toast.success(t('pledgeDeleted') as string); } catch (error: unknown) { toast.error((error as Error).message); } diff --git a/src/screens/FundCampaignPledge/PledgeModal.tsx b/src/screens/FundCampaignPledge/PledgeModal.tsx index ab126101de..782a051277 100644 --- a/src/screens/FundCampaignPledge/PledgeModal.tsx +++ b/src/screens/FundCampaignPledge/PledgeModal.tsx @@ -157,7 +157,7 @@ const PledgeModal: React.FC = ({ ...updatedFields, }, }); - toast.success(t('pledgeUpdated')); + toast.success(t('pledgeUpdated') as string); refetchPledge(); hide(); } catch (error: unknown) { @@ -183,7 +183,7 @@ const PledgeModal: React.FC = ({ }, }); - toast.success(t('pledgeCreated')); + toast.success(t('pledgeCreated') as string); refetchPledge(); setFormState({ pledgeUsers: [], diff --git a/src/screens/LoginPage/LoginPage.test.tsx b/src/screens/LoginPage/LoginPage.test.tsx index 1eeb683654..3733a454dd 100644 --- a/src/screens/LoginPage/LoginPage.test.tsx +++ b/src/screens/LoginPage/LoginPage.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render, screen, fireEvent, within } from '@testing-library/react'; +import { render, screen, fireEvent, within } from '@testing-library/react'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import userEvent from '@testing-library/user-event'; diff --git a/src/screens/LoginPage/LoginPage.tsx b/src/screens/LoginPage/LoginPage.tsx index d69f4f88d5..56b9012ee8 100644 --- a/src/screens/LoginPage/LoginPage.tsx +++ b/src/screens/LoginPage/LoginPage.tsx @@ -175,7 +175,7 @@ const loginPage = (): JSX.Element => { return data.recaptcha; } catch (error) { /* istanbul ignore next */ - toast.error(t('captchaError')); + toast.error(t('captchaError') as string); } }; @@ -198,7 +198,7 @@ const loginPage = (): JSX.Element => { const isVerified = await verifyRecaptcha(recaptchaToken); /* istanbul ignore next */ if (!isVerified) { - toast.error(t('Please_check_the_captcha')); + toast.error(t('Please_check_the_captcha') as string); return; } const isValidatedString = (value: string): boolean => @@ -239,7 +239,9 @@ const loginPage = (): JSX.Element => { /* istanbul ignore next */ if (signUpData) { toast.success( - t(role === 'admin' ? 'successfullyRegistered' : 'afterRegister'), + t( + role === 'admin' ? 'successfullyRegistered' : 'afterRegister', + ) as string, ); setShowTab('LOGIN'); setSignFormState({ @@ -256,20 +258,20 @@ const loginPage = (): JSX.Element => { errorHandler(t, error); } } else { - toast.warn(t('passwordMismatches')); + toast.warn(t('passwordMismatches') as string); } } else { if (!isValidatedString(signfirstName)) { - toast.warn(t('firstName_invalid')); + toast.warn(t('firstName_invalid') as string); } if (!isValidatedString(signlastName)) { - toast.warn(t('lastName_invalid')); + toast.warn(t('lastName_invalid') as string); } if (!validatePassword(signPassword)) { - toast.warn(t('password_invalid')); + toast.warn(t('password_invalid') as string); } if (signEmail.length < 8) { - toast.warn(t('email_invalid')); + toast.warn(t('email_invalid') as string); } } }; @@ -279,7 +281,7 @@ const loginPage = (): JSX.Element => { const isVerified = await verifyRecaptcha(recaptchaToken); /* istanbul ignore next */ if (!isVerified) { - toast.error(t('Please_check_the_captcha')); + toast.error(t('Please_check_the_captcha') as string); return; } @@ -300,7 +302,7 @@ const loginPage = (): JSX.Element => { appUserProfile.isSuperAdmin || appUserProfile.adminFor.length !== 0; if (role === 'admin' && !isAdmin) { - toast.warn(tErrors('notAuthorised')); + toast.warn(tErrors('notAuthorised') as string); return; } const loggedInUserId = user._id; @@ -324,7 +326,7 @@ const loginPage = (): JSX.Element => { navigate(role === 'admin' ? '/orglist' : '/user/organizations'); } else { - toast.warn(tErrors('notFound')); + toast.warn(tErrors('notFound') as string); } } catch (error) { /* istanbul ignore next */ diff --git a/src/screens/ManageTag/ManageTag.tsx b/src/screens/ManageTag/ManageTag.tsx index 5262724892..86a44fb169 100644 --- a/src/screens/ManageTag/ManageTag.tsx +++ b/src/screens/ManageTag/ManageTag.tsx @@ -113,7 +113,7 @@ function ManageTag(): JSX.Element { userTagAssignedMembersRefetch(); toggleUnassignTagModal(); - toast.success(t('successfullyUnassigned')); + toast.success(t('successfullyUnassigned') as string); } catch (error: unknown) { /* istanbul ignore next */ if (error instanceof Error) { diff --git a/src/screens/MemberDetail/MemberDetail.test.tsx b/src/screens/MemberDetail/MemberDetail.test.tsx index e094354552..7b3707754c 100644 --- a/src/screens/MemberDetail/MemberDetail.test.tsx +++ b/src/screens/MemberDetail/MemberDetail.test.tsx @@ -442,7 +442,6 @@ describe('MemberDetail', () => { test('should display warnings for blank form submission', async () => { jest.spyOn(toast, 'warning'); const props = { - key: '123', id: '1', toggleStateValue: jest.fn(), }; @@ -452,7 +451,7 @@ describe('MemberDetail', () => { - + @@ -467,6 +466,7 @@ describe('MemberDetail', () => { expect(toast.warning).toHaveBeenCalledWith('Last Name cannot be blank!'); expect(toast.warning).toHaveBeenCalledWith('Email cannot be blank!'); }); + test('display admin', async () => { const props = { id: 'rishav-jha-mech', diff --git a/src/screens/MemberDetail/MemberDetail.tsx b/src/screens/MemberDetail/MemberDetail.tsx index ef623146d4..6d92ccbb4a 100644 --- a/src/screens/MemberDetail/MemberDetail.tsx +++ b/src/screens/MemberDetail/MemberDetail.tsx @@ -194,7 +194,7 @@ const MemberDetail: React.FC = ({ id }): JSX.Element => { setItem('Email', email); setItem('UserImage', image); } - toast.success(tCommon('successfullyUpdated')); + toast.success(tCommon('successfullyUpdated') as string); } } catch (error: unknown) { if (error instanceof Error) { diff --git a/src/screens/OrgContribution/OrgContribution.test.tsx b/src/screens/OrgContribution/OrgContribution.test.tsx index d2abc134df..f9f63c6807 100644 --- a/src/screens/OrgContribution/OrgContribution.test.tsx +++ b/src/screens/OrgContribution/OrgContribution.test.tsx @@ -1,6 +1,6 @@ -import React from 'react'; +import React, { act } from 'react'; import { MockedProvider } from '@apollo/react-testing'; -import { act, render } from '@testing-library/react'; +import { render } from '@testing-library/react'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; import 'jest-location-mock'; diff --git a/src/screens/OrgList/OrgList.test.tsx b/src/screens/OrgList/OrgList.test.tsx index e7e0f7d4be..5b889ff07d 100644 --- a/src/screens/OrgList/OrgList.test.tsx +++ b/src/screens/OrgList/OrgList.test.tsx @@ -515,22 +515,21 @@ describe('Organisations Page testing as Admin', () => { ); await wait(); + }); + const sortDropdown = await waitFor(() => screen.getByTestId('sort')); + expect(sortDropdown).toBeInTheDocument(); - const searchInput = screen.getByTestId('sort'); - expect(searchInput).toBeInTheDocument(); - - const inputText = screen.getByTestId('sortOrgs'); + const sortToggle = screen.getByTestId('sortOrgs'); - fireEvent.click(inputText); - const toggleText = screen.getByTestId('latest'); + fireEvent.click(sortToggle); + const latestOption = await waitFor(() => screen.getByTestId('latest')); - fireEvent.click(toggleText); + fireEvent.click(latestOption); - expect(searchInput).toBeInTheDocument(); - fireEvent.click(inputText); - const toggleTite = screen.getByTestId('oldest'); - fireEvent.click(toggleTite); - expect(searchInput).toBeInTheDocument(); - }); + expect(sortDropdown).toBeInTheDocument(); + fireEvent.click(sortToggle); + const oldestOption = await waitFor(() => screen.getByTestId('oldest')); + fireEvent.click(oldestOption); + expect(sortDropdown).toBeInTheDocument(); }); }); diff --git a/src/screens/OrgList/OrgList.tsx b/src/screens/OrgList/OrgList.tsx index 5c91678217..7023bfa9b1 100644 --- a/src/screens/OrgList/OrgList.tsx +++ b/src/screens/OrgList/OrgList.tsx @@ -169,11 +169,11 @@ function orgList(): JSX.Element { const triggerCreateSampleOrg = (): void => { createSampleOrganization() .then(() => { - toast.success(t('sampleOrgSuccess')); + toast.success(t('sampleOrgSuccess') as string); window.location.reload(); }) .catch(() => { - toast.error(t('sampleOrgDuplicate')); + toast.error(t('sampleOrgDuplicate') as string); }); }; diff --git a/src/screens/OrgPost/OrgPost.test.tsx b/src/screens/OrgPost/OrgPost.test.tsx index 465da98642..9829589350 100644 --- a/src/screens/OrgPost/OrgPost.test.tsx +++ b/src/screens/OrgPost/OrgPost.test.tsx @@ -1,11 +1,11 @@ import { MockedProvider } from '@apollo/react-testing'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import { fireEvent, render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import 'jest-location-mock'; import { I18nextProvider } from 'react-i18next'; import { Provider } from 'react-redux'; import { BrowserRouter } from 'react-router-dom'; -import React from 'react'; +import React, { act } from 'react'; import { CREATE_POST_MUTATION } from 'GraphQl/Mutations/mutations'; import { ORGANIZATION_POST_LIST } from 'GraphQl/Queries/Queries'; import { ToastContainer } from 'react-toastify'; @@ -286,25 +286,34 @@ describe('Organisation Post Page', () => { , ); + }); + await wait(); - await wait(); - - const searchInput = screen.getByTestId('searchByName'); - expect(searchInput).toHaveAttribute('placeholder', 'Search By Title'); + const searchInput = screen.getByTestId('searchByName'); + expect(searchInput).toHaveAttribute('placeholder', 'Search By Title'); - const inputText = screen.getByTestId('searchBy'); + const inputText = screen.getByTestId('searchBy'); + await act(async () => { fireEvent.click(inputText); - const toggleText = screen.getByTestId('Text'); + }); + const toggleText = screen.getByTestId('Text'); + + await act(async () => { fireEvent.click(toggleText); + }); - expect(searchInput).toHaveAttribute('placeholder', 'Search By Text'); + expect(searchInput).toHaveAttribute('placeholder', 'Search By Text'); + await act(async () => { fireEvent.click(inputText); - const toggleTite = screen.getByTestId('searchTitle'); + }); + const toggleTite = screen.getByTestId('searchTitle'); + await act(async () => { fireEvent.click(toggleTite); - expect(searchInput).toHaveAttribute('placeholder', 'Search By Title'); }); + + expect(searchInput).toHaveAttribute('placeholder', 'Search By Title'); }); test('Testing search latest and oldest toggle', async () => { await act(async () => { @@ -320,25 +329,34 @@ describe('Organisation Post Page', () => { , ); + }); + await wait(); - await wait(); - - const searchInput = screen.getByTestId('sort'); - expect(searchInput).toBeInTheDocument(); + const searchInput = screen.getByTestId('sort'); + expect(searchInput).toBeInTheDocument(); - const inputText = screen.getByTestId('sortpost'); + const inputText = screen.getByTestId('sortpost'); + await act(async () => { fireEvent.click(inputText); - const toggleText = screen.getByTestId('latest'); + }); + + const toggleText = screen.getByTestId('latest'); + await act(async () => { fireEvent.click(toggleText); + }); - expect(searchInput).toBeInTheDocument(); + expect(searchInput).toBeInTheDocument(); + await act(async () => { fireEvent.click(inputText); - const toggleTite = screen.getByTestId('oldest'); + }); + + const toggleTite = screen.getByTestId('oldest'); + await act(async () => { fireEvent.click(toggleTite); - expect(searchInput).toBeInTheDocument(); }); + expect(searchInput).toBeInTheDocument(); }); test('After creating a post, the data should be refetched', async () => { const refetchMock = jest.fn(); @@ -575,34 +593,43 @@ describe('Organisation Post Page', () => { , ); + }); + await wait(); - await wait(); - + await act(async () => { userEvent.click(screen.getByTestId('createPostModalBtn')); + }); - const postTitleInput = screen.getByTestId('modalTitle'); + const postTitleInput = screen.getByTestId('modalTitle'); + await act(async () => { fireEvent.change(postTitleInput, { target: { value: 'Test Post' } }); + }); - const postInfoTextarea = screen.getByTestId('modalinfo'); + const postInfoTextarea = screen.getByTestId('modalinfo'); + await act(async () => { fireEvent.change(postInfoTextarea, { target: { value: 'Test post information' }, }); + }); - const videoFile = new File(['video content'], 'video.mp4', { - type: 'video/mp4', - }); + const videoFile = new File(['video content'], 'video.mp4', { + type: 'video/mp4', + }); + await act(async () => { userEvent.upload(screen.getByTestId('addMediaField'), videoFile); + }); - // Check if the video is displayed - const videoPreview = await screen.findByTestId('videoPreview'); - expect(videoPreview).toBeInTheDocument(); + // Check if the video is displayed + const videoPreview = await screen.findByTestId('videoPreview'); + expect(videoPreview).toBeInTheDocument(); - // Check if the close button for the video works - const closeVideoPreviewButton = screen.getByTestId('mediaCloseButton'); + // Check if the close button for the video works + const closeVideoPreviewButton = screen.getByTestId('mediaCloseButton'); + await act(async () => { fireEvent.click(closeVideoPreviewButton); - expect(videoPreview).not.toBeInTheDocument(); }); + expect(videoPreview).not.toBeInTheDocument(); }); test('Sorting posts by pinned status', async () => { // Mocked data representing posts with different pinned statuses diff --git a/src/screens/OrgPost/OrgPost.tsx b/src/screens/OrgPost/OrgPost.tsx index e1a0559f11..1931f6c76a 100644 --- a/src/screens/OrgPost/OrgPost.tsx +++ b/src/screens/OrgPost/OrgPost.tsx @@ -170,7 +170,7 @@ function orgPost(): JSX.Element { /* istanbul ignore next */ if (data) { - toast.success(t('postCreatedSuccess')); + toast.success(t('postCreatedSuccess') as string); refetch(); setPostFormState({ posttitle: '', @@ -309,7 +309,7 @@ function orgPost(): JSX.Element {
diff --git a/src/screens/UserPortal/Campaigns/PledgeModal.tsx b/src/screens/UserPortal/Campaigns/PledgeModal.tsx index 6ab2832c2f..d9076f52c8 100644 --- a/src/screens/UserPortal/Campaigns/PledgeModal.tsx +++ b/src/screens/UserPortal/Campaigns/PledgeModal.tsx @@ -166,7 +166,7 @@ const PledgeModal: React.FC = ({ ...updatedFields, }, }); - toast.success(t('pledgeUpdated')); + toast.success(t('pledgeUpdated') as string); refetchPledge(); hide(); } catch (error: unknown) { @@ -198,7 +198,7 @@ const PledgeModal: React.FC = ({ }, }); - toast.success(t('pledgeCreated')); + toast.success(t('pledgeCreated') as string); refetchPledge(); setFormState({ pledgeUsers: [], diff --git a/src/screens/UserPortal/Chat/Chat.test.tsx b/src/screens/UserPortal/Chat/Chat.test.tsx index a0716d3151..6ab225c6bb 100644 --- a/src/screens/UserPortal/Chat/Chat.test.tsx +++ b/src/screens/UserPortal/Chat/Chat.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, fireEvent, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { fireEvent, render, screen, waitFor } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider, useTranslation } from 'react-i18next'; @@ -2294,15 +2294,26 @@ describe('Testing Chat Screen [User Portal]', () => { , ); + screen.debug(); + await waitFor(() => { + const closeMenuBtn = screen.queryByTestId('closeMenu'); + expect(closeMenuBtn).toBeInTheDocument(); + if (closeMenuBtn) { + closeMenuBtn.click(); + } else { + throw new Error('Close menu button not found'); + } + }); - await wait(); - - const closeMenubtn = screen.getByTestId('closeMenu'); - expect(closeMenubtn).toBeInTheDocument(); - closeMenubtn.click(); - const openMenuBtn = screen.getByTestId('openMenu'); - expect(openMenuBtn).toBeInTheDocument(); - openMenuBtn.click(); + await waitFor(() => { + const openMenuBtn = screen.queryByTestId('openMenu'); + expect(openMenuBtn).toBeInTheDocument(); + if (openMenuBtn) { + openMenuBtn.click(); + } else { + throw new Error('Open menu button not found'); + } + }); }); test('Testing sidebar when the screen size is less than or equal to 820px', async () => { @@ -2328,12 +2339,16 @@ describe('Testing Chat Screen [User Portal]', () => { , ); - await wait(); - expect(screen.getByText('My Organizations')).toBeInTheDocument(); - expect(screen.getByText('Talawa User Portal')).toBeInTheDocument(); - - const chatBtn = screen.getByTestId('chatBtn'); - - chatBtn.click(); + screen.debug(); + await waitFor(() => { + expect(screen.getByText('My Organizations')).toBeInTheDocument(); + expect(screen.getByText('Talawa User Portal')).toBeInTheDocument(); + }); + await waitFor(() => { + const chatBtn = screen.getByTestId('chatBtn'); + act(() => { + chatBtn.click(); + }); + }); }); }); diff --git a/src/screens/UserPortal/Donate/Donate.test.tsx b/src/screens/UserPortal/Donate/Donate.test.tsx index fc30ba7503..c4d435415e 100644 --- a/src/screens/UserPortal/Donate/Donate.test.tsx +++ b/src/screens/UserPortal/Donate/Donate.test.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { act, render, screen } from '@testing-library/react'; +import React, { act } from 'react'; +import { render, screen } from '@testing-library/react'; import { MockedProvider } from '@apollo/react-testing'; import { I18nextProvider } from 'react-i18next'; @@ -16,6 +16,7 @@ import Donate from './Donate'; import userEvent from '@testing-library/user-event'; import useLocalStorage from 'utils/useLocalstorage'; import { DONATE_TO_ORGANIZATION } from 'GraphQl/Mutations/mutations'; +import { toast } from 'react-toastify'; const MOCKS = [ { @@ -136,6 +137,13 @@ jest.mock('react-router-dom', () => ({ useParams: () => ({ orgId: '' }), })); +jest.mock('react-toastify', () => ({ + toast: { + error: jest.fn(), + success: jest.fn(), + }, +})); + describe('Testing Donate Screen [User Portal]', () => { Object.defineProperty(window, 'matchMedia', { writable: true, @@ -280,4 +288,103 @@ describe('Testing Donate Screen [User Portal]', () => { userEvent.click(screen.getByTestId('donateBtn')); await wait(); }); + + test('displays error toast for donation amount below minimum', async () => { + render( + + + + + + + + + , + ); + + await wait(); + + userEvent.type(screen.getByTestId('donationAmount'), '0.5'); + userEvent.click(screen.getByTestId('donateBtn')); + + await wait(); + + expect(toast.error).toHaveBeenCalledWith( + 'Donation amount must be between 1 and 10000000.', + ); + }); + + test('displays error toast for donation amount above maximum', async () => { + render( + + + + + + + + + , + ); + + await wait(); + + userEvent.type(screen.getByTestId('donationAmount'), '10000001'); + userEvent.click(screen.getByTestId('donateBtn')); + + await wait(); + + expect(toast.error).toHaveBeenCalledWith( + 'Donation amount must be between 1 and 10000000.', + ); + }); + + test('displays error toast for empty donation amount', async () => { + render( + + + + + + + + + , + ); + + await wait(); + + userEvent.click(screen.getByTestId('donateBtn')); + + await wait(); + + expect(toast.error).toHaveBeenCalledWith( + 'Please enter a numerical value for the donation amount.', + ); + }); + + test('displays error toast for invalid (non-numeric) donation amount', async () => { + render( + + + + + + + + + , + ); + + await wait(); + + userEvent.type(screen.getByTestId('donationAmount'), 'abc'); + userEvent.click(screen.getByTestId('donateBtn')); + + await wait(); + + expect(toast.error).toHaveBeenCalledWith( + 'Please enter a numerical value for the donation amount.', + ); + }); }); diff --git a/src/screens/UserPortal/Donate/Donate.tsx b/src/screens/UserPortal/Donate/Donate.tsx index 4a87bd2490..c6e6d918d5 100644 --- a/src/screens/UserPortal/Donate/Donate.tsx +++ b/src/screens/UserPortal/Donate/Donate.tsx @@ -128,6 +128,28 @@ export default function donate(): JSX.Element { }, [donationData]); const donateToOrg = (): void => { + // check if the amount is non empty and is a number + if (amount === '' || Number.isNaN(Number(amount))) { + toast.error(t(`invalidAmount`)); + return; + } + + // check if the amount is non negative and within the range + const minDonation = 1; + const maxDonation = 10000000; + if ( + Number(amount) <= 0 || + Number(amount) < minDonation || + Number(amount) > maxDonation + ) { + toast.error( + t(`donationOutOfRange`, { min: minDonation, max: maxDonation }), + ); + return; + } + + const formattedAmount = Number(amount.trim()); + try { donate({ variables: { @@ -135,12 +157,12 @@ export default function donate(): JSX.Element { createDonationOrgId2: organizationId, payPalId: 'paypalId', nameOfUser: userName, - amount: Number(amount), + amount: formattedAmount, nameOfOrg: organizationDetails.name, }, }); refetch(); - toast.success(t(`success`)); + toast.success(t(`success`) as string); } catch (error: unknown) { /* istanbul ignore next */ errorHandler(t, error); @@ -216,6 +238,9 @@ export default function donate(): JSX.Element { />
+ + {t('donationAmountDescription')} +