From 8f3b6c202c34ce2e6073bdbc513817251f02e39c Mon Sep 17 00:00:00 2001 From: Thomas T Date: Wed, 1 Nov 2023 12:05:17 +0100 Subject: [PATCH] first commit --- .env | 2 + .eslintrc.cjs | 8 + .github/workflows/test.yml | 33 + .gitignore | 132 +++ CODE_OF_CONDUCT.md | 134 +++ Dockerfile | 11 + LICENSE | 11 + README.md | 47 + bun.lockb | Bin 0 -> 53277 bytes docker-compose-with-redis-cache.yml | 10 + docker-compose.yml | 23 + docs/CONTRIBUTING.md | 37 + docs/calculation.md | 92 ++ docs/constants.md | 7 + docs/index.md | 12 + docs/space_constants.md | 17 + docs/variables.md | 5 + package.json | 30 + postman_collection.json | 64 ++ src/calculations/calculator.ts | 510 +++++++++ .../interfaces/calculationresult.ts | 26 + src/calculations/interfaces/config.ts | 7 + src/calculations/interfaces/constant.ts | 5 + src/calculations/interfaces/space.ts | 11 + .../interfaces/space_calculation.ts | 13 + src/calculations/interfaces/space_constant.ts | 17 + src/calculations/interfaces/space_result.ts | 14 + src/calculations/interfaces/variable.ts | 32 + .../spaces/common area/auditorium.ts | 14 + .../spaces/common area/canteen.ts | 14 + .../spaces/common area/cleaning.ts | 11 + src/calculations/spaces/common area/course.ts | 14 + .../spaces/common area/exerciseroom.ts | 21 + src/calculations/spaces/common area/hwc.ts | 18 + .../spaces/common area/kitchen.ts | 14 + src/calculations/spaces/common area/lobby.ts | 26 + .../spaces/common area/reception.ts | 11 + .../spaces/common area/special.ts | 11 + .../spaces/common area/stockarchive.ts | 11 + src/calculations/spaces/common area/toilet.ts | 18 + .../spaces/common area/waitingzone.ts | 11 + .../spaces/common area/wardrobe.ts | 11 + src/calculations/spaces/dummy.ts | 32 + src/calculations/spaces/index.ts | 115 +++ src/calculations/spaces/main_space_class.ts | 279 +++++ src/calculations/spaces/shared area/dockin.ts | 25 + src/calculations/spaces/shared area/hwc.ts | 21 + src/calculations/spaces/shared area/lounge.ts | 15 + .../spaces/shared area/meetingroom_large.ts | 35 + .../spaces/shared area/meetingroom_medium.ts | 35 + .../spaces/shared area/meetingroom_small.ts | 35 + .../spaces/shared area/multiroom.ts | 28 + .../spaces/shared area/personalstorage.ts | 14 + .../spaces/shared area/projectroom.ts | 14 + .../spaces/shared area/reception.ts | 14 + .../spaces/shared area/special.ts | 11 + src/calculations/spaces/shared area/toilet.ts | 21 + .../spaces/shared area/touchdown.ts | 25 + .../spaces/shared area/waitingzone.ts | 14 + .../spaces/shared area/wardrobe.ts | 14 + .../spaces/work related area/celloffice.ts | 22 + .../spaces/work related area/cleaning.ts | 11 + .../spaces/work related area/coffeestation.ts | 18 + .../spaces/work related area/copyarchive.ts | 11 + .../spaces/work related area/dockin.ts | 22 + .../spaces/work related area/focusroom.ts | 19 + .../spaces/work related area/hwc.ts | 18 + .../spaces/work related area/landscape.ts | 24 + .../work related area/meetingroom_large.ts | 37 + .../work related area/meetingroom_medium.ts | 37 + .../work related area/meetingroom_mini.ts | 34 + .../work related area/meetingroom_small.ts | 37 + .../work related area/minikitchenlounge.ts | 18 + .../spaces/work related area/multiroom.ts | 26 + .../work related area/personalstorage.ts | 22 + .../spaces/work related area/projectroom.ts | 19 + .../spaces/work related area/quietzone.ts | 19 + .../spaces/work related area/reception.ts | 16 + .../spaces/work related area/special.ts | 11 + .../spaces/work related area/toilet.ts | 18 + .../spaces/work related area/touchdown.ts | 22 + .../spaces/work related area/waitingzone.ts | 16 + .../spaces/work related area/wardrobe.ts | 13 + src/config/default.json | 965 ++++++++++++++++++ src/index.ts | 59 ++ test/main.test.ts | 10 + test/scenarios/1.json | 48 + tsconfig.json | 35 + 88 files changed, 3869 insertions(+) create mode 100644 .env create mode 100644 .eslintrc.cjs create mode 100644 .github/workflows/test.yml create mode 100644 .gitignore create mode 100644 CODE_OF_CONDUCT.md create mode 100644 Dockerfile create mode 100644 LICENSE create mode 100644 README.md create mode 100755 bun.lockb create mode 100644 docker-compose-with-redis-cache.yml create mode 100644 docker-compose.yml create mode 100644 docs/CONTRIBUTING.md create mode 100644 docs/calculation.md create mode 100644 docs/constants.md create mode 100644 docs/index.md create mode 100644 docs/space_constants.md create mode 100644 docs/variables.md create mode 100644 package.json create mode 100644 postman_collection.json create mode 100644 src/calculations/calculator.ts create mode 100644 src/calculations/interfaces/calculationresult.ts create mode 100644 src/calculations/interfaces/config.ts create mode 100644 src/calculations/interfaces/constant.ts create mode 100644 src/calculations/interfaces/space.ts create mode 100644 src/calculations/interfaces/space_calculation.ts create mode 100644 src/calculations/interfaces/space_constant.ts create mode 100644 src/calculations/interfaces/space_result.ts create mode 100644 src/calculations/interfaces/variable.ts create mode 100644 src/calculations/spaces/common area/auditorium.ts create mode 100644 src/calculations/spaces/common area/canteen.ts create mode 100644 src/calculations/spaces/common area/cleaning.ts create mode 100644 src/calculations/spaces/common area/course.ts create mode 100644 src/calculations/spaces/common area/exerciseroom.ts create mode 100644 src/calculations/spaces/common area/hwc.ts create mode 100644 src/calculations/spaces/common area/kitchen.ts create mode 100644 src/calculations/spaces/common area/lobby.ts create mode 100644 src/calculations/spaces/common area/reception.ts create mode 100644 src/calculations/spaces/common area/special.ts create mode 100644 src/calculations/spaces/common area/stockarchive.ts create mode 100644 src/calculations/spaces/common area/toilet.ts create mode 100644 src/calculations/spaces/common area/waitingzone.ts create mode 100644 src/calculations/spaces/common area/wardrobe.ts create mode 100644 src/calculations/spaces/dummy.ts create mode 100644 src/calculations/spaces/index.ts create mode 100644 src/calculations/spaces/main_space_class.ts create mode 100644 src/calculations/spaces/shared area/dockin.ts create mode 100644 src/calculations/spaces/shared area/hwc.ts create mode 100644 src/calculations/spaces/shared area/lounge.ts create mode 100644 src/calculations/spaces/shared area/meetingroom_large.ts create mode 100644 src/calculations/spaces/shared area/meetingroom_medium.ts create mode 100644 src/calculations/spaces/shared area/meetingroom_small.ts create mode 100644 src/calculations/spaces/shared area/multiroom.ts create mode 100644 src/calculations/spaces/shared area/personalstorage.ts create mode 100644 src/calculations/spaces/shared area/projectroom.ts create mode 100644 src/calculations/spaces/shared area/reception.ts create mode 100644 src/calculations/spaces/shared area/special.ts create mode 100644 src/calculations/spaces/shared area/toilet.ts create mode 100644 src/calculations/spaces/shared area/touchdown.ts create mode 100644 src/calculations/spaces/shared area/waitingzone.ts create mode 100644 src/calculations/spaces/shared area/wardrobe.ts create mode 100644 src/calculations/spaces/work related area/celloffice.ts create mode 100644 src/calculations/spaces/work related area/cleaning.ts create mode 100644 src/calculations/spaces/work related area/coffeestation.ts create mode 100644 src/calculations/spaces/work related area/copyarchive.ts create mode 100644 src/calculations/spaces/work related area/dockin.ts create mode 100644 src/calculations/spaces/work related area/focusroom.ts create mode 100644 src/calculations/spaces/work related area/hwc.ts create mode 100644 src/calculations/spaces/work related area/landscape.ts create mode 100644 src/calculations/spaces/work related area/meetingroom_large.ts create mode 100644 src/calculations/spaces/work related area/meetingroom_medium.ts create mode 100644 src/calculations/spaces/work related area/meetingroom_mini.ts create mode 100644 src/calculations/spaces/work related area/meetingroom_small.ts create mode 100644 src/calculations/spaces/work related area/minikitchenlounge.ts create mode 100644 src/calculations/spaces/work related area/multiroom.ts create mode 100644 src/calculations/spaces/work related area/personalstorage.ts create mode 100644 src/calculations/spaces/work related area/projectroom.ts create mode 100644 src/calculations/spaces/work related area/quietzone.ts create mode 100644 src/calculations/spaces/work related area/reception.ts create mode 100644 src/calculations/spaces/work related area/special.ts create mode 100644 src/calculations/spaces/work related area/toilet.ts create mode 100644 src/calculations/spaces/work related area/touchdown.ts create mode 100644 src/calculations/spaces/work related area/waitingzone.ts create mode 100644 src/calculations/spaces/work related area/wardrobe.ts create mode 100644 src/config/default.json create mode 100644 src/index.ts create mode 100644 test/main.test.ts create mode 100644 test/scenarios/1.json create mode 100644 tsconfig.json diff --git a/.env b/.env new file mode 100644 index 0000000..df6d0f8 --- /dev/null +++ b/.env @@ -0,0 +1,2 @@ +PORT=1337 +USE_CACHE_REDIS=0 \ No newline at end of file diff --git a/.eslintrc.cjs b/.eslintrc.cjs new file mode 100644 index 0000000..52f4238 --- /dev/null +++ b/.eslintrc.cjs @@ -0,0 +1,8 @@ +/* eslint-env bun */ +module.exports = { + extends: ['eslint:recommended', 'plugin:@typescript-eslint/recommended'], + parser: '@typescript-eslint/parser', + plugins: ['@typescript-eslint'], + root: true, + ignorePatterns: ['**/*.js'], +} diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 0000000..8db1894 --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,33 @@ +# Workflow for tests + +name: Test + +on: + push: + branches: ["main"] + pull_request: + branches: ["main"] + # Allows you to run this workflow manually from the Actions tab + workflow_dispatch: + +permissions: + contents: read + +jobs: + test: + runs-on: ubuntu-latest + steps: + - name: Setup Bun + uses: antongolub/action-setup-bun@v1 + + - name: Setup repo + uses: actions/checkout@v3 + + - name: Install + run: bun install + + - name: Run linter + run: bun lint + + - name: Run tests + run: bun test diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..b8c49d4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,132 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Diagnostic reports (https://nodejs.org/api/report.html) +report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage +*.lcov + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) +web_modules/ + +# TypeScript cache +*.tsbuildinfo + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Optional stylelint cache +.stylelintcache + +# Microbundle cache +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variable files +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) +.cache +.parcel-cache + +# Next.js build output +.next +out + +# Nuxt.js build / generate output +.nuxt +dist + +# Gatsby files +.cache/ +# Comment in the public line in if your project uses Gatsby and not Next.js +# https://nextjs.org/blog/next-9-1#public-directory-support +# public + +# vuepress build output +.vuepress/dist + +# vuepress v2.x temp and cache directory +.temp +.cache + +# Docusaurus cache and generated files +.docusaurus + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +# DynamoDB Local files +.dynamodb/ + +# TernJS port file +.tern-port + +# Stores VSCode versions used for testing VSCode extensions +.vscode-test + +# yarn v2 +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# Build +build/ diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 0000000..94dc90a --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,134 @@ + +# Contributor Covenant Code of Conduct + +## Our Pledge + +We as members, contributors, and leaders pledge to make participation in our +community a harassment-free experience for everyone, regardless of age, body +size, visible or invisible disability, ethnicity, sex characteristics, gender +identity and expression, level of experience, education, socio-economic status, +nationality, personal appearance, race, caste, color, religion, or sexual +identity and orientation. + +We pledge to act and interact in ways that contribute to an open, welcoming, +diverse, inclusive, and healthy community. + +## Our Standards + +Examples of behavior that contributes to a positive environment for our +community include: + +* Demonstrating empathy and kindness toward other people +* Being respectful of differing opinions, viewpoints, and experiences +* Giving and gracefully accepting constructive feedback +* Accepting responsibility and apologizing to those affected by our mistakes, + and learning from the experience +* Focusing on what is best not just for us as individuals, but for the overall + community + +Examples of unacceptable behavior include: + +* The use of sexualized language or imagery, and sexual attention or advances of + any kind +* Trolling, insulting or derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or email address, + without their explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Enforcement Responsibilities + +Community leaders are responsible for clarifying and enforcing our standards of +acceptable behavior and will take appropriate and fair corrective action in +response to any behavior that they deem inappropriate, threatening, offensive, +or harmful. + +Community leaders have the right and responsibility to remove, edit, or reject +comments, commits, code, wiki edits, issues, and other contributions that are +not aligned to this Code of Conduct, and will communicate reasons for moderation +decisions when appropriate. + +## Scope + +This Code of Conduct applies within all community spaces, and also applies when +an individual is officially representing the community in public spaces. +Examples of representing our community include using an official e-mail address, +posting via an official social media account, or acting as an appointed +representative at an online or offline event. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported to the community leaders responsible for enforcement at +huulbaek@gmail.com. +All complaints will be reviewed and investigated promptly and fairly. + +All community leaders are obligated to respect the privacy and security of the +reporter of any incident. + +## Enforcement Guidelines + +Community leaders will follow these Community Impact Guidelines in determining +the consequences for any action they deem in violation of this Code of Conduct: + +### 1. Correction + +**Community Impact**: Use of inappropriate language or other behavior deemed +unprofessional or unwelcome in the community. + +**Consequence**: A private, written warning from community leaders, providing +clarity around the nature of the violation and an explanation of why the +behavior was inappropriate. A public apology may be requested. + +### 2. Warning + +**Community Impact**: A violation through a single incident or series of +actions. + +**Consequence**: A warning with consequences for continued behavior. No +interaction with the people involved, including unsolicited interaction with +those enforcing the Code of Conduct, for a specified period of time. This +includes avoiding interactions in community spaces as well as external channels +like social media. Violating these terms may lead to a temporary or permanent +ban. + +### 3. Temporary Ban + +**Community Impact**: A serious violation of community standards, including +sustained inappropriate behavior. + +**Consequence**: A temporary ban from any sort of interaction or public +communication with the community for a specified period of time. No public or +private interaction with the people involved, including unsolicited interaction +with those enforcing the Code of Conduct, is allowed during this period. +Violating these terms may lead to a permanent ban. + +### 4. Permanent Ban + +**Community Impact**: Demonstrating a pattern of violation of community +standards, including sustained inappropriate behavior, harassment of an +individual, or aggression toward or disparagement of classes of individuals. + +**Consequence**: A permanent ban from any sort of public interaction within the +community. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], +version 2.1, available at +[https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1]. + +Community Impact Guidelines were inspired by +[Mozilla's code of conduct enforcement ladder][Mozilla CoC]. + +For answers to common questions about this code of conduct, see the FAQ at +[https://www.contributor-covenant.org/faq][FAQ]. Translations are available at +[https://www.contributor-covenant.org/translations][translations]. + +[homepage]: https://www.contributor-covenant.org +[v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html +[Mozilla CoC]: https://github.com/mozilla/diversity +[FAQ]: https://www.contributor-covenant.org/faq +[translations]: https://www.contributor-covenant.org/translations + diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..712c661 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,11 @@ +FROM oven/bun:latest + +WORKDIR /app +COPY package.json ./ +COPY bun.lockb ./ + +RUN bun install --no-save + +COPY . . + +CMD ["bun", "run", "src/index.ts"] diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..63ae4ab --- /dev/null +++ b/LICENSE @@ -0,0 +1,11 @@ +Permission to use, copy, modify, and/or +distribute this software for any purpose with or without fee is hereby +granted. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL +WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY +SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER +RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, +NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE +OR PERFORMANCE OF THIS SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..f65e2c7 --- /dev/null +++ b/README.md @@ -0,0 +1,47 @@ +# Agiliate Engine +The Agiliate Engine is a sophisticated computing engine designed to calculate the required office space for a company. It takes into account various parameters such as the number of employees, workspace types, and government regulations to provide an accurate estimation of the space needed. + +Read this quick start file or the [documentation](./docs/index.md) to learn more about the Agiliate Engine. + +## Configuration +The application's configuration parameters are stored in the src/config/default.json file. These parameters can be modified to suit the specific needs of your company. The configuration file to use is defined in the constructor of the `Calculator` class in `src/calculator.js`. You can also change the variables and constants in the request, as shown in the example Postman collection, `postman_collection.json`. + +## Installation and requirements +The application requires either [Bun](https://bun.sh/) or [Docker](https://www.docker.com/) to run. If using Bun run + +```bash +bun install +``` + +to install the dependencies. If using Docker Compose, run + +```bash +docker-compose up +``` + +to start the application. Or use + +```bash +docker build -t agiliate-engine . && docker run -p 1337:1337 agiliate-engine +``` + +## Development server +Start a development hot reload server with + +```bash +bun run --watch src/index.ts +``` + +## Running +Run the application with + +```bash +bun run src/index.ts +``` + +## Tests +Run the tests with + +```bash +bun test +``` diff --git a/bun.lockb b/bun.lockb new file mode 100755 index 0000000000000000000000000000000000000000..147b2acaef570056850f412c71a0b4f17687c613 GIT binary patch literal 53277 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 OvujgVuMqx!e?I|km(7R( literal 0 HcmV?d00001 diff --git a/docker-compose-with-redis-cache.yml b/docker-compose-with-redis-cache.yml new file mode 100644 index 0000000..68b5cda --- /dev/null +++ b/docker-compose-with-redis-cache.yml @@ -0,0 +1,10 @@ +version: '3.8' + +services: + agiliate-engine: + command: bun run start-with-redis + redis: + container_name: redis + image: redis:latest + ports: + - 6379:6379 diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 0000000..01d572b --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,23 @@ +version: '3.8' + +services: + agiliate-engine: + build: + context: . + container_name: agiliate-engine + restart: unless-stopped + working_dir: /app + ports: + - 1337:1337 + volumes: + - .:/app + - node_modules:/app/node_modules + command: bun run start + redis: + container_name: redis + image: redis:latest + ports: + - 6379:6379 + +volumes: + node_modules: diff --git a/docs/CONTRIBUTING.md b/docs/CONTRIBUTING.md new file mode 100644 index 0000000..4b2effe --- /dev/null +++ b/docs/CONTRIBUTING.md @@ -0,0 +1,37 @@ +# Contributing to Agiliate Engine +First and foremost, please do! We welcome all contributions, big or small, no matter your level of experience – be that: + +- Reporting a bug +- Discussing the current state of the code +- Submitting a fix +- Proposing new features + +Please also read our [code of conduct](../CODE_OF_CONDUCT.md) before you start. + +We use GitHub to host code, to track issues and feature requests, as well as accept pull requests – pull requests are the best way to propose changes to the codebase: + +1. Fork the repo and create your branch from `main`. +2. If you've added code that should be tested, add or update tests in the `test` folder. +3. If needed, update the documentation. +4. Ensure the test suite passes. +5. Make sure your code lints. +6. Let's get your changes merged! + +**Any contributions you make will be under the 0BSD license** +In short, when you submit code changes, your submissions are understood to be under the same [0BSD license](https://opensource.org/license/0bsd/) that covers the project. Feel free to contact the maintainers if that's a concern. + +Please report [bugs and issues on GitHub](https://github.com/nodalit/agiliate-engine). + +**Great Bug Reports** could contain: + +- A quick summary and/or background +- Steps to reproduce + - Be specific + - Give sample code if you can. +- What you expected would happen +- What actually happens + +We *love* good bug reports. + +*Be constistent with style* +Lint that mother – run `bun lint`. diff --git a/docs/calculation.md b/docs/calculation.md new file mode 100644 index 0000000..7d16a6f --- /dev/null +++ b/docs/calculation.md @@ -0,0 +1,92 @@ +# Calculation +The [Calculator](../src/calculations/calculator.ts) class is the main class responsible for performing calculations. It is initialized with variables, custom space constants, custom constants, and an optional configuration file. + +```typescript +new Calculator(variables, customSpaceConstants, customConstants, config) +``` + +where the `variables`, `customSpaceConstants`, `customConstants` and `config` are in the JSON body of a POST request: + +```json +{ + "variables": { + "accessToCoworking": false, + "accessToCanteen": true, + "accessToCourseSpace": true, + "accessToAuditorium": true, + "accessToCellOffice": true, + "accessToReception": false, + "accessToExercise": true, + "specialAreaOffice": 80, + "specialAreaShared": 0, + "specialAreaCommon": 100, + "seatsInAuditorium": 50, + "numberOfEmployees": 330, + "concurrencyAttendanceShare": 1.0, + "peakConcurrencyAttendanceShare": 1.0, + "overCapacityShare": 0.0, + "homeOfficeAverageShare": 0.0, + "shareOfEmployeesInAuditorium": 0.30, + "touchdownShare": 0.0555, + "dockinShare": 0.2197, + "coworkingShare": 0.00, + "cellOfficeShare": 0.00, + "landscapeShare": 0.3630, + "projectroomShare": 0.1891, + "focusroomShare": 0.1054, + "quietzoneShare": 0.0672, + "miniMeetingroomShare": 0.2105, + "smallMeetingroomShare": 0.3017, + "mediumMeetingroomShare": 0.3600, + "largeMeetingroomShare": 0.1277 + }, + "customSpaceConstants": { + "auditorium": { + "minimumSquareMeters": 120 + } + }, + "customConstants": { + "governmentMinimumSquaremetersPerWorkSpace": 6 + } +} +``` + +There is also an example of a complete request in the [Postman collection](../postman_collection.json) + +The config property is populated by reading the provided configuration file. The constants property is a merge of the default constants from the configuration file and the custom constants. + +## Calculation Process + +The main calculation process is performed by the `result` method. This method performs several steps to calculate the results: + +1. *First Run*: The `#processSpacesFirst` method is called to calculate the initial results for all spaces. These results are needed in subsequent calculations. + +2. *Aggregate Subspace Results*: The `#sumResults` method is called to aggregate the results of the subspaces. + +3. *Calculate Total Workplace Area*: The `#calculateTotalWorkplaceArea` method is called to calculate the total workplace area. + +4. *Calculate Total Compensation Area*: The `#calculateTotalCompensationArea` method is called to calculate the total area that needs compensation calculations. + +5. *Calculate Total Employees Per Workspace Type Unadjusted*: The `#calculateTotalEmployeesPerWorkspaceTypeUnadjusted` method is called to calculate the total employees per workspace type (unadjusted). + +6. *Second Run*: The `#processSpacesSecond` method is called to calculate the second run results for all spaces. + +7. *Calculate Total Unadjusted Area*: The `#calculateTotalUnadjustedArea` method is called to calculate the total unadjusted area. + +8. *Third Run*: The `#processSpacesThird` method is called to calculate the third run results for all spaces. + +9. *Fourth Run*: The `#processSpacesFourth` method is called to calculate the fourth run results for all spaces. + +10. *Calculate Total Adjusted Area Including Compensation and Adjustment*: The `#calculateTotalAdjustedAreaInclCompensationAndAdjustment` method is called to calculate the total area including compensation and adjustment. + +11. *Compute Utility and Inner Walls*: The `#computeUtilityAndInnerWalls` method is called to calculate the utility floor space and inner walls of the top level spaces. + +12. *Compute Technical and Communication Area*: The `#computeTechnicalAndCommunicationArea` method is called to calculate the technical and communication areas. + +13. *Add Technical and Communication Area*: The `#addTechnicalAndCommunicationArea` method is called to add the technical and communication areas to the result. + +14. *Calculate Final Results*: The final results are calculated and returned. + +## Helper methods + +The Calculator class also contains several private helper methods that are used to perform specific calculations. These include methods to calculate the corridor area, inner wall area, and to add custom spaces to the result. diff --git a/docs/constants.md b/docs/constants.md new file mode 100644 index 0000000..075241a --- /dev/null +++ b/docs/constants.md @@ -0,0 +1,7 @@ +# Constants + +Constants are predefined values used in the Agiliate Engine. They are documented in the [constant.ts](../src/calculations/interfaces/constant.ts) file + +The `constants` property holds the default constants for the calculations. The `customConstants` property can be used to override the default constants. + +An example of a constant is `governmentMinimumSquaremetersPerWorkSpace`. It is used to define the minimum square meters per workspace required by the government. The default value can be overridden in the `customConstants` object. diff --git a/docs/index.md b/docs/index.md new file mode 100644 index 0000000..4e93203 --- /dev/null +++ b/docs/index.md @@ -0,0 +1,12 @@ +# Agiliate Engine documentation + +The Agiliate Engine is a sophisticated computing engine designed to calculate the required office space for a company. It takes into account various parameters such as the number of employees, workspace types, and government regulations to provide an accurate estimation of the space needed. + +You are most welcome to contribute to this project. Please read the [contribution guidelines](./contribution.md) before you start. + +You should read about the main calculation process in the [calculation.md](./calculation.md) file. + +## Input +- [Variables](./variables.md) +- [Constants](./constants.md) +- [Space Constants](./space_constants.md) diff --git a/docs/space_constants.md b/docs/space_constants.md new file mode 100644 index 0000000..69f47c6 --- /dev/null +++ b/docs/space_constants.md @@ -0,0 +1,17 @@ +# Space Constants + +Space constants are predefined space values used in the Agiliate Engine. They are documented in the [space_constant.ts](../src/calculations/interfaces/space_constant.ts) file + +The `customSpaceConstants` object can be used to override the default constants for specific spaces. + +For example, if you wish to overwrite the `minimumSquareMeters` for the auditorium space, you can do so by setting the `auditorium` property in the `customSpaceConstants` object. + +```json +{ + "customSpaceConstants": { + "auditorium": { + "minimumSquareMeters": 100 + } + } +} +``` diff --git a/docs/variables.md b/docs/variables.md new file mode 100644 index 0000000..0fb0a6c --- /dev/null +++ b/docs/variables.md @@ -0,0 +1,5 @@ +# Variables + +The IVariable interface in the codebase represents a set of variables used in the calculations. They are sent to the application as a *POST request* at the `calculate` enpoint. See the example [Postman collection](../postman_collection.json) + +The variables are documented in the [variables.ts](../src/calculations/interfaces/variable.ts) file. diff --git a/package.json b/package.json new file mode 100644 index 0000000..97069f8 --- /dev/null +++ b/package.json @@ -0,0 +1,30 @@ +{ + "name": "agiliate-engine", + "version": "1.0.0", + "description": "This is the Open Source version of the Agiliate Engine.", + "main": "src/index.ts", + "scripts": { + "dev": "bun run --watch src/index.ts", + "start": "bun install --no-save && bun run src/index.ts", + "start-with-redis": "bun install --no-save && USE_CACHE_REDIS=1 bun run src/index.ts", + "test": "bun test", + "lint": "bun eslint ." + }, + "keywords": [], + "author": "", + "license": "ISC", + "devDependencies": { + "bun-types": "latest", + "@typescript-eslint/eslint-plugin": "^6.8.0", + "@typescript-eslint/parser": "^6.8.0", + "eslint": "^8.52.0", + "typescript": "^5.1.3" + }, + "peerDependencies": { + "typescript": "^5.1.3", + "redis": "^4.6.10" + }, + "dependencies": { + "redis": "^4.6.10" + } +} diff --git a/postman_collection.json b/postman_collection.json new file mode 100644 index 0000000..567f615 --- /dev/null +++ b/postman_collection.json @@ -0,0 +1,64 @@ +{ + "info": { + "_postman_id": "ea84e19f-893b-4463-9400-6a99fd19272b", + "name": "Agiliate OSS", + "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json", + "_exporter_id": "1849425", + "_collection_link": "https://meetrteam.postman.co/workspace/Team-Workspace~2b7cece5-f18f-4293-8e9d-f8467b285c02/collection/1849425-ea84e19f-893b-4463-9400-6a99fd19272b?action=share&source=collection_link&creator=1849425" + }, + "item": [ + { + "name": "Calculate", + "request": { + "method": "POST", + "header": [], + "body": { + "mode": "raw", + "raw": "{\n \"variables\": {\n \"accessToCoworking\": false,\n \"accessToCanteen\": true,\n \"accessToCourseSpace\": true,\n \"accessToAuditorium\": true,\n \"accessToCellOffice\": true,\n \"accessToReception\": false,\n \"accessToExercise\": true,\n \"specialAreaOffice\": 80,\n \"specialAreaShared\": 0,\n \"specialAreaCommon\": 100,\n \"seatsInAuditorium\": 50,\n \"numberOfEmployees\": 330,\n \"concurrencyAttendanceShare\": 1.0,\n \"peakConcurrencyAttendanceShare\": 1.0,\n \"overCapacityShare\": 0.0,\n \"homeOfficeAverageShare\": 0.0,\n \"shareOfEmployeesInAuditorium\": 0.30,\n \"touchdownShare\": 0.0555,\n \"dockinShare\": 0.2197,\n \"coworkingShare\": 0.00,\n \"cellOfficeShare\": 0.00,\n \"landscapeShare\": 0.3630,\n \"projectroomShare\": 0.1891,\n \"focusroomShare\": 0.1054,\n \"quietzoneShare\": 0.0672,\n \"miniMeetingroomShare\": 0.2105,\n \"smallMeetingroomShare\": 0.3017,\n \"mediumMeetingroomShare\": 0.3600,\n \"largeMeetingroomShare\": 0.1277\n },\n \"customSpaceConstants\": {\n \"auditorium\": {\n \"minimumSquareMeters\": 120\n }\n },\n \"customConstants\": {\n \"governmentMinimumSquaremetersPerWorkSpace\": 6\n }\n}", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{base_url}}/calculate", + "host": [ + "{{base_url}}" + ], + "path": [ + "calculate" + ] + } + }, + "response": [] + } + ], + "event": [ + { + "listen": "prerequest", + "script": { + "type": "text/javascript", + "exec": [ + "" + ] + } + }, + { + "listen": "test", + "script": { + "type": "text/javascript", + "exec": [ + "" + ] + } + } + ], + "variable": [ + { + "key": "base_url", + "value": "http://localhost:1337", + "type": "string" + } + ] +} \ No newline at end of file diff --git a/src/calculations/calculator.ts b/src/calculations/calculator.ts new file mode 100644 index 0000000..06c4bcd --- /dev/null +++ b/src/calculations/calculator.ts @@ -0,0 +1,510 @@ +import { readFileSync } from 'fs' +import { join } from 'path' +import { ISpace } from './interfaces/space' +import { IConfig } from './interfaces/config' +import { ISpaceCalculation } from './interfaces/space_calculation' +import { ISpaceResult } from './interfaces/space_result' +import { ISpaceConstant } from './interfaces/space_constant' +import { IVariable } from './interfaces/variable' +import { IConstant } from './interfaces/constant' +import { ICalculationResult } from './interfaces/calculationresult' +import getSpace from './spaces' +import Dummy from './spaces/dummy' + +/** + * This is the main class for the calculator. + */ +export default class Calculator { + variables: IVariable + customSpaceConstants: ISpaceConstant + customConstants: IConstant + config: IConfig + constants: IConstant + totalWorkplaceArea: number = 0 + totalCompensationArea: number = 0 + totalUnadjustedArea: number = 0 + totalEmployeesPerWorkplaceTypeUnadjusted: number = 0 + totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation: number = 0 + allEmployeesPerWorkplaceTypeUnadjusted: number[] = [] + + /** + * Initialize the calculator with the context from Koa and the optional config file to use + * @param {IVariable} variables – The variables to use + * @param {ISpaceConstant} customSpaceConstants – Custom space constants + * @param {IConstant} customConstants – Custom constants + * @param {string} [configFile] – The config file to use + */ + constructor(variables: IVariable, customSpaceConstants: ISpaceConstant, customConstants: IConstant , configFile: string = 'default.json') { + this.variables = variables + this.customSpaceConstants = customSpaceConstants + this.customConstants = customConstants + this.config = JSON.parse(readFileSync(join(import.meta.dir, '..', 'config', configFile), 'utf-8')) + // Merge custom constants with the default constants + this.constants = { + ...this.config.constants, + ...customConstants + } + } + + /** + * This is the main method responsible for calculating the results + * @returns {ICalculationResult} The result of the calculation + */ + result (): ICalculationResult { + // Get the results for all spaces, first run + let spaceResults = this.#processSpacesFirst(this.config.spaces) + // Get aggregate subspace results, first run + for (const space of spaceResults) { + space.result = this.#sumResults(space) + } + // Get some aggregate summed results + this.#calculateTotalWorkplaceArea(spaceResults) + this.#calculateTotalCompensationArea(spaceResults) + this.#calculateTotalEmployeesPerWorkspaceTypeUnadjusted(spaceResults) + + // Get aggregate subspace results, second run + spaceResults = this.#processSpacesSecond(spaceResults) + + // Get some more sums + this.#calculateTotalUnadjustedArea(spaceResults) + + // Get aggregate subspace results, third run + spaceResults = this.#processSpacesThird(spaceResults) + + // Get aggregate subspace results, fourth run + spaceResults = this.#processSpacesFourth(spaceResults) + + this.#calculateTotalAdjustedAreaInclCompensationAndAdjustment(spaceResults) + + // Get aggregate subspace results, second run (not checked yet! maybe another sumresults function only calculating the needed) + for (const space of spaceResults) { + space.result = this.#sumResults(space) + } + + // Calculate the utility floor space and inner walls of the top level spaces + const { utilityFloorSpace, innerwallsAreaSum, updatedSpaceResults } = this.#computeUtilityAndInnerWalls(spaceResults) + spaceResults = updatedSpaceResults + + const netArea = utilityFloorSpace - innerwallsAreaSum + const netAreaPerEmployee = netArea / this.variables.numberOfEmployees + + // Calculate the technical and communication areas + const { technicalArea, communicationArea } = this.#computeTechnicalAndCommunicationArea(utilityFloorSpace) + spaceResults = this.#addTechnicalAndCommunicationArea(spaceResults, technicalArea) + + const grossArea = utilityFloorSpace + technicalArea + communicationArea + + const grossAreaPerEmployee = grossArea / this.variables.numberOfEmployees + const utilityFloorSpacePerEmployee = utilityFloorSpace / this.variables.numberOfEmployees + const grossAreaPerDimensionedAttendance = grossArea / this.#dimensionedAttendance() + const grossNetFactor = grossArea / netArea + + return { + totals: { + unadjustedArea: Math.ceil(this.totalUnadjustedArea), + workplaceArea: Math.ceil(this.totalWorkplaceArea), + compensationArea: Math.ceil(this.totalCompensationArea), + employeesPerWorkplaceTypeUnadjusted: this.totalEmployeesPerWorkplaceTypeUnadjusted, + adjustedAreaInclCompensationWithAdjustmentAndCompensation: Math.ceil(this.totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation), + utilityFloorSpace: Math.ceil(utilityFloorSpace), + netArea: Math.ceil(netArea), + grossArea: Math.ceil(grossArea), + netAreaPerEmployee: Math.round(netAreaPerEmployee * 100) / 100, + grossAreaPerEmployee: Math.round(grossAreaPerEmployee * 100) / 100, + utilityFloorSpacePerEmployee: Math.round(utilityFloorSpacePerEmployee * 100) / 100, + grossAreaPerDimensionedAttendance: Math.round(grossAreaPerDimensionedAttendance * 100) / 100, + grossNetFactor: Math.round(grossNetFactor * 100) / 100, + }, + spaces: spaceResults + } + } + + /** + * This method is used to add a custom space to the result, for example corridors and inner walls, which are not really spaces + * @param {string} name - The name of the space + * @param {ISpaceResult} result - The result of the space + * @returns {ISpace} + */ + #addCustomSpace (name: string, result: ISpaceResult): ISpace { + return { + name, + result, + constants: { + areaPerRole: 0, + personsPerType: 0, + shouldCalculateCompensation: false, + adhereToGovernmentMinimum: false, + } + } + } + + /** + * This method add the technical and communication areas to the result + * @param {number} area - The area + * @param {ISpace[]} spaceResults - The space results to add to + * @returns {ISpace[]} The updated space results + * @todo This function will be deprecated once we have a better way of handling technical and communication areas + */ + #addTechnicalAndCommunicationArea (spaceResults: ISpace[], area: number): ISpace[] { + const spaceResult: ISpaceResult = { + areaExclCompensation: area, + adjustedAreaInclCompensation: area, + adjustedAreaInclCompensationWithAdjustmentAndCompensation: area, + } + spaceResults.push(this.#addCustomSpace('communication area', spaceResult)) + spaceResults.push(this.#addCustomSpace('technical area', spaceResult)) + return spaceResults + } + + /** + * This method computes the technical and communication areas + * @param {number} utilityFloorSpace - The utility floor space + * @returns {object} + */ + #computeTechnicalAndCommunicationArea (utilityFloorSpace: number): {technicalArea: number, communicationArea: number} { + const pct = ((utilityFloorSpace / 0.74) - utilityFloorSpace) / (utilityFloorSpace * 2) + const technicalArea = utilityFloorSpace * pct + const communicationArea = utilityFloorSpace * pct + return { + technicalArea, + communicationArea + } + } + + /** + * Calculate the corridors and inner walls of the top level spaces + * @param {ISpace[]} spaceResults - The space results to calculate + * @returns {object} + */ + #computeUtilityAndInnerWalls (spaceResults: ISpace[]): {utilityFloorSpace: number, innerwallsAreaSum: number, updatedSpaceResults: ISpace[]} { + let utilityFloorSpace: number = this.totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation // utilityFloorSpace is the total adjusted area incl. compensation and adjustment + corridors and inner walls + let innerwallsAreaSum: number = 0 + for (const space of spaceResults) { + if (space.shouldCalculateCorridor) { + const corridorArea = this.#calculateCorridorArea(space.result.adjustedAreaInclCompensation!) + const corridorResult: ISpaceResult = { + areaExclCompensation: corridorArea, + adjustedAreaInclCompensation: corridorArea, + adjustedAreaInclCompensationWithAdjustmentAndCompensation: corridorArea, + } + space.spaces?.push(this.#addCustomSpace(`corridor – ${space.name}`, corridorResult)) + utilityFloorSpace += corridorArea + // Update top level space results + space.result.netArea = space.result.adjustedAreaInclCompensationWithAdjustmentAndCompensation! + corridorArea + } + if (space.shouldCalculateInnerwalls) { + const innerwallsArea = this.#calculateInnerwallArea(space.result.areaExclCompensation!) + const innerwallsResult: ISpaceResult = { + areaExclCompensation: innerwallsArea, + adjustedAreaInclCompensation: innerwallsArea, + adjustedAreaInclCompensationWithAdjustmentAndCompensation: innerwallsArea, + } + space.spaces?.push(this.#addCustomSpace(`inner wall – ${space.name}`, innerwallsResult)) + utilityFloorSpace += innerwallsArea + innerwallsAreaSum += innerwallsArea + // Update top level space results + space.result.utilityFloorSpace = space.result.netArea! + innerwallsArea + } + } + return { + utilityFloorSpace, + innerwallsAreaSum, + updatedSpaceResults: spaceResults + } + } + + /** + * This method calculates the sum results for spaces + * @param {ISpace} space - The space to calculate + * @returns {ISpaceResult} + */ + #sumResults(space: ISpace): ISpaceResult { + // Sum up the results of the children, if they exist + let childSum_areaExclCompensation = 0 + let childSum_adjustedAreaInclCompensation = 0 + let childSum_notAdjustedAddonArea = 0 + let childSum_adjustedAreaInclCompensationWithAdjustmentAndCompensation = 0 + + if (space.spaces) { + for (const child of space.spaces) { + const r = this.#sumResults(child) + childSum_areaExclCompensation += r.areaExclCompensation || 0 + childSum_adjustedAreaInclCompensation += r.adjustedAreaInclCompensation || 0 + childSum_notAdjustedAddonArea += r.notAdjustedAddonArea || 0 + childSum_adjustedAreaInclCompensationWithAdjustmentAndCompensation += r.adjustedAreaInclCompensationWithAdjustmentAndCompensation || 0 + } + } + + // Use the sum if we don't already have a result + space.result.areaExclCompensation = space.result.areaExclCompensation || childSum_areaExclCompensation + space.result.adjustedAreaInclCompensation = space.result.adjustedAreaInclCompensation || childSum_adjustedAreaInclCompensation + space.result.notAdjustedAddonArea = space.result.notAdjustedAddonArea || childSum_notAdjustedAddonArea + space.result.adjustedAreaInclCompensationWithAdjustmentAndCompensation = space.result.adjustedAreaInclCompensationWithAdjustmentAndCompensation || childSum_adjustedAreaInclCompensationWithAdjustmentAndCompensation + + // Return the total sum + return { + areaExclCompensation: space.result.areaExclCompensation, + adjustedAreaInclCompensation: space.result.adjustedAreaInclCompensation, + notAdjustedAddonArea: space.result.notAdjustedAddonArea, + adjustedAreaInclCompensationWithAdjustmentAndCompensation: space.result.adjustedAreaInclCompensationWithAdjustmentAndCompensation + } + } + + /** + * Gets dimensioned attendance. We use a dummy because it's not space dependent. + * @returns {number} + */ + #dimensionedAttendance (): number { + return new Dummy(this.variables, this.config).dimensionedAttendance() + } + + /** + * Loop through all spaces and calculate the first run + * @param {ISpace[]} spaces + * @returns {ISpace[]} + */ + #processSpacesFirst(spaces: ISpace[]): ISpace[] { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.#processSpacesFirst(space.spaces) + } else { + space.result = this.#calculateSpaceFirstRun(space) + } + } + return spaces + } + + /** + * Loop through all spaces and calculate the second run + * @param {ISpace[]} spaces + * @returns {ISpace[]} + */ + #processSpacesSecond(spaces: ISpace[]): ISpace[] { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.#processSpacesSecond(space.spaces) + } else { + space.result = { + ... space.result, + ... this.#calculateSpaceSecondRun(space) + } + this.allEmployeesPerWorkplaceTypeUnadjusted.push(space.result.employeesPerWorkplaceTypeUnadjusted || 0) + } + } + return spaces + } + + /** + * Loop through all spaces and calculate the third run + * @param {ISpace[]} spaces + * @returns {ISpace[]} + */ + #processSpacesThird(spaces: ISpace[]): ISpace[] { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.#processSpacesThird(space.spaces) + } else { + space.result = { + ... space.result, + ... this.#calculateSpaceThirdRun(space) + } + } + } + return spaces + } + + /** + * Loop through all spaces and calculate the fourth run + * @param {ISpace[]} spaces + * @returns {ISpace[]} + */ + #processSpacesFourth(spaces: ISpace[]): ISpace[] { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.#processSpacesFourth(space.spaces) + } else { + // Calculate the space + space.result = { + ... space.result, + ... this.#calculateSpaceFourthRun(space) + } + } + } + return spaces + } + + /** + * Calculates a space based on the first run + * @param {ISpace} space - The space to calculate + * @returns {ISpaceResult} The calculated result object + */ + #calculateSpaceFirstRun (space: ISpace): ISpaceResult { + const Space = getSpace(space.name) + const spaceCalculation: ISpaceCalculation = new Space(space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return { + areaExclCompensation: spaceCalculation.calculateAreaExclCompensation(), + employeesPerWorkplaceTypeUnadjusted: spaceCalculation.calculateEmployeesPerWorkplaceTypeUnadjusted(), + } + } + + /** + * Calculates a space the second time + * @param {ISpace} space - The space to calculate + * @returns {ISpaceResult} The calculated space + */ + #calculateSpaceSecondRun (space: ISpace): ISpaceResult { + const Space = getSpace(space.name) + const spaceCalculation: ISpaceCalculation = new Space(space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return { + notAdjustedAddonArea: spaceCalculation.calculateNotAdjustedAddonArea(this.totalWorkplaceArea, this.totalCompensationArea), + } + } + + /** + * Calculates a space the third time + * @param {ISpace} space - The space to calculate + * @returns {ISpaceResult} The calculated space + */ + #calculateSpaceThirdRun (space: ISpace): ISpaceResult { + const Space = getSpace(space.name) + const spaceCalculation: ISpaceCalculation = new Space(space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return { + notAdjustedAddonPart: spaceCalculation.calculateNotAdjustedAddonPart(this.totalWorkplaceArea, this.totalCompensationArea), + adjustedAddonArea: spaceCalculation.calculateAdjustedAddonArea(this.totalWorkplaceArea, this.totalCompensationArea, this.totalUnadjustedArea), + adjustedAreaInclCompensation: spaceCalculation.calculateAdjustedAreaInclCompensation(this.totalWorkplaceArea, this.totalCompensationArea, this.totalUnadjustedArea), + employeesPerWorkplaceTypeAdjusted: spaceCalculation.calculateEmployeesPerWorkplaceTypeAdjusted(this.totalEmployeesPerWorkplaceTypeUnadjusted, this.allEmployeesPerWorkplaceTypeUnadjusted), + } + } + + /** + * Calculates a space the fourth time + * @param {ISpace} space - The space to calculate + * @returns {ISpaceResult} The calculated space + */ + #calculateSpaceFourthRun (space: ISpace): ISpaceResult { + const Space = getSpace(space.name) + const spaceCalculation: ISpaceCalculation = new Space(space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return { + adjustedAreaInclCompensationWithAdjustmentAndCompensation: spaceCalculation.calculateAdjustedAreaInclCompensationWithAdjustmentAndCompensation(), + numberOfRooms: spaceCalculation.calculateNumberOfRooms(), + numberOfSeats: spaceCalculation.calculateNumberOfSeats(), + } + } + + /** + * Calculates the total work place area for all spaces + * @param {ISpace[]} spaces - The spaces to calculate + * @returns {number} The total work place area + */ + #calculateTotalWorkplaceArea (spaces: ISpace[]): number { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.totalWorkplaceArea = this.#calculateTotalWorkplaceArea(space.spaces) + } else { + // Calculate the total area + if (space.constants.countsTowardsWorkspaceArea) { + this.totalWorkplaceArea += space.result.areaExclCompensation ?? 0 + } + } + } + return this.totalWorkplaceArea + } + + /** + * Calculates the total area that needs compensation + * @param {ISpace[]} spaces - The spaces to calculate + * @returns {number} The total work place area + */ + #calculateTotalCompensationArea (spaces: ISpace[]): number { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.totalCompensationArea = this.#calculateTotalCompensationArea(space.spaces) + } else { + // Calculate the total area + if (space.constants.shouldCalculateCompensation) { + this.totalCompensationArea += space.result.areaExclCompensation ?? 0 + } + } + } + return this.totalCompensationArea + } + + /** + * Calculates the total area including compensation and adjustment + * @param {ISpace[]} spaces - The spaces to calculate + * @returns {number} The total work place area + */ + #calculateTotalAdjustedAreaInclCompensationAndAdjustment (spaces: ISpace[]): number { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation = this.#calculateTotalAdjustedAreaInclCompensationAndAdjustment(space.spaces) + } else { + // Calculate the total area + this.totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation += space.result.adjustedAreaInclCompensationWithAdjustmentAndCompensation ?? 0 + } + } + return this.totalAdjustedAreaInclCompensationWithAdjustmentAndCompensation + } + + /** + * This method gets the un-adjusted area sum + * @param {ISpace[]} spaces - The spaces to calculate + * @returns {number} The total un-adjusted area + */ + #calculateTotalUnadjustedArea (spaces: ISpace[]): number { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.totalUnadjustedArea = this.#calculateTotalUnadjustedArea(space.spaces) + } else { + // Calculate the total area + if (space.constants.shouldCalculateCompensation) { + this.totalUnadjustedArea += space.result.notAdjustedAddonArea ?? 0 + } + } + } + return this.totalUnadjustedArea + } + + /** + * This method gets the employees per workplacetype (unadjusted) sum + * @param {ISpace[]} spaces - The spaces to calculate + * @returns {number} The total employees per workplace type (unadjusted) + */ + #calculateTotalEmployeesPerWorkspaceTypeUnadjusted (spaces: ISpace[]): number { + for (const space of spaces) { + // If this space has nested spaces, process those + if (space.spaces) { + this.totalEmployeesPerWorkplaceTypeUnadjusted = this.#calculateTotalEmployeesPerWorkspaceTypeUnadjusted(space.spaces) + } else { + this.totalEmployeesPerWorkplaceTypeUnadjusted += space.result.employeesPerWorkplaceTypeUnadjusted ?? 0 + } + } + return this.totalEmployeesPerWorkplaceTypeUnadjusted + } + + /** + * Calculates the area of corridors. + * @param {number} areaInclCompensation - The area including compensation + * @returns {number} + */ + #calculateCorridorArea (areaInclCompensation: number): number { + return areaInclCompensation * this.constants.corridorAddonShare + } + + /** + * Calculates the area of corridors. + * @param {number} areaExclCompensation - The area excluding compensation + * @returns {number} + */ + #calculateInnerwallArea (areaExclCompensation: number): number { + return areaExclCompensation * this.constants.innerwallsAddonShare + } +} diff --git a/src/calculations/interfaces/calculationresult.ts b/src/calculations/interfaces/calculationresult.ts new file mode 100644 index 0000000..812fde3 --- /dev/null +++ b/src/calculations/interfaces/calculationresult.ts @@ -0,0 +1,26 @@ +import { ISpace } from './space' + +export interface ICalculationResult { + /** + * Object containing various total values. + */ + totals: { + unadjustedArea: number // Total amount of unadjusted area + workplaceArea: number // Total area dedicated to workplaces + compensationArea: number // Total area used for compensation calculation + employeesPerWorkplaceTypeUnadjusted: number // Total number of employees per unadjusted workplace type + adjustedAreaInclCompensationWithAdjustmentAndCompensation: number // Total adjusted area including compensation with adjustment and compensation + netArea: number // Total net area + utilityFloorSpace: number // Total utility floor space + grossArea: number // Total gross area + utilityFloorSpacePerEmployee: number // Utility floor space per employee + netAreaPerEmployee: number // Net area per employee + grossAreaPerEmployee: number // Gross area per employee + grossAreaPerDimensionedAttendance: number // Gross area per dimensioned attendance + grossNetFactor: number // Gross net factor + } + /** + * Object containing all spaces and detailed calculation results + */ + spaces: ISpace[] +} diff --git a/src/calculations/interfaces/config.ts b/src/calculations/interfaces/config.ts new file mode 100644 index 0000000..9c3ca85 --- /dev/null +++ b/src/calculations/interfaces/config.ts @@ -0,0 +1,7 @@ +import { IConstant } from './constant' +import { ISpace } from './space' + +export interface IConfig { + constants: IConstant // The constants used in the calculations + spaces: ISpace[] // The array of spaces to be calculated +} diff --git a/src/calculations/interfaces/constant.ts b/src/calculations/interfaces/constant.ts new file mode 100644 index 0000000..a6185ce --- /dev/null +++ b/src/calculations/interfaces/constant.ts @@ -0,0 +1,5 @@ +export interface IConstant { + governmentMinimumSquaremetersPerWorkSpace: number // Minimum required square meters per workspace by regulation + corridorAddonShare: number // Share of corridor area + innerwallsAddonShare: number // Share of inner walls area +} diff --git a/src/calculations/interfaces/space.ts b/src/calculations/interfaces/space.ts new file mode 100644 index 0000000..ed567ff --- /dev/null +++ b/src/calculations/interfaces/space.ts @@ -0,0 +1,11 @@ +import { ISpaceConstant } from './space_constant' +import { ISpaceResult } from './space_result' + +export interface ISpace { + name: string, // The name of the space + constants: ISpaceConstant // The constants specific to this space + spaces?: ISpace[] // Optional nested spaces within this space + result: ISpaceResult // The result of the calculations for this space + shouldCalculateCorridor?: boolean // Whether to calculate the corridor area needed for this space + shouldCalculateInnerwalls?: boolean // Whether to calculate the inner walls area needed for this space +} diff --git a/src/calculations/interfaces/space_calculation.ts b/src/calculations/interfaces/space_calculation.ts new file mode 100644 index 0000000..6ab15c8 --- /dev/null +++ b/src/calculations/interfaces/space_calculation.ts @@ -0,0 +1,13 @@ +export interface ISpaceCalculation { + calculateEmployeesPerWorkplaceTypeUnadjusted: () => number // Function to calculate the unadjusted number of employees per workplace type + calculateAreaExclCompensation: () => number // Function to calculate the area excluding compensation + calculateAdjustedAreaInclCompensation: (totalWorkplceArea: number, totalCompensationArea: number, totalUnadjustedArea: number) => number // Function to calculate the adjusted area including compensation + calculateNotAdjustedAddonArea: (totalWorkplaceArea: number, totalCompensationArea: number) => number // Function to calculate the not adjusted add-on area + calculateAdjustedAddonArea: (totalWorkplaceArea: number, totalCompensationArea: number, totalUnadjustedArea: number) => number // Function to calculate the adjusted add-on area + calculateNotAdjustedAddonPart: (totalWorkplaceArea: number, totalCompensationArea: number) => number // Function to calculate the not adjusted add-on part + calculateEmployeesPerWorkplaceTypeAdjusted: (totalEmployeesPerWorkplaceTypeUnadjusted: number, allEmployeesPerWorkplaceTypeUnadjusted: number[]) => number // Function to calculate the adjusted number of employees per workplace type + calculateAdjustedAreaInclCompensationWithAdjustmentAndCompensation: () => number // Function to calculate the adjusted area including compensation with adjustment and compensation + dimensionedAttendance: () => number // Function to calculate the dimensioned attendance + calculateNumberOfRooms: () => number // Function to calculate the number of rooms, if applicable + calculateNumberOfSeats: () => number // Function to calculate the number of seats, if applicable +} diff --git a/src/calculations/interfaces/space_constant.ts b/src/calculations/interfaces/space_constant.ts new file mode 100644 index 0000000..5d6fbf9 --- /dev/null +++ b/src/calculations/interfaces/space_constant.ts @@ -0,0 +1,17 @@ +export interface ISpaceConstant { + seatPersonRoom?: string // Optional, defines the type of space: seat, person or room + shouldCalculateNumberOfRooms?: boolean // Whether to calculate the number of rooms + shouldCalculateNumberOfSeats?: boolean // Whether to calculate the number of seats + areaPerRole: number // The area required per role + personsPerType: number // The number of persons per type of space + minimumPersons?: number // The minimum number of persons for this space + minimumSquareMeters?: number // The minimum square meters required for this space + shouldCalculateCompensation: boolean // Whether to calculate compensation for this space + countsTowardsWorkspaceArea?: boolean // Whether this space counts towards the total workspace area + seatAreaPerSpot?: number // The seat area per spot + requiresAdditionalStorage?: boolean // Whether this space requires additional storage + adhereToGovernmentMinimum: boolean // Whether to adhere to the government minimum for this space + surchargeInternalCorridorShare?: number // The surcharge for the internal corridor share + areaPerPersonExcludingCorridor?: number // The area per person excluding the corridor + unitsPerPerson?: number // The units per person for this space +} diff --git a/src/calculations/interfaces/space_result.ts b/src/calculations/interfaces/space_result.ts new file mode 100644 index 0000000..c7bf556 --- /dev/null +++ b/src/calculations/interfaces/space_result.ts @@ -0,0 +1,14 @@ +export interface ISpaceResult { + areaExclCompensation?: number // The area excluding compensation + notAdjustedAddonArea?: number // The not adjusted add-on area + notAdjustedAddonPart?: number // The not adjusted add-on part + adjustedAddonArea?: number // The adjusted add-on area + adjustedAreaInclCompensation?: number // The adjusted area including compensation + adjustedAreaInclCompensationWithAdjustmentAndCompensation?: number // The adjusted area including compensation with adjustment and compensation + employeesPerWorkplaceTypeUnadjusted?: number // The unadjusted number of employees per workplace type + employeesPerWorkplaceTypeAdjusted?: number // The adjusted number of employees per workplace type + netArea?: number // The net area of the space + utilityFloorSpace?: number // The utility floor space of the space + numberOfRooms?: number // The number of rooms, if applicable + numberOfSeats?: number // The number of seats, if applicable +} diff --git a/src/calculations/interfaces/variable.ts b/src/calculations/interfaces/variable.ts new file mode 100644 index 0000000..4b1adec --- /dev/null +++ b/src/calculations/interfaces/variable.ts @@ -0,0 +1,32 @@ +export interface IVariable { + accessToCanteen: boolean // Whether there is a need for access to a canteen + accessToAuditorium: boolean // Whether there is a need for access to an auditorium + seatsInAuditorium: number // The number of seats in the auditorium + shareOfEmployeesInAuditorium: number // The percentage of employees in the auditorium + accessToCourseSpace: boolean // Whether there is a need for access to a course space + accessToGym: boolean // Whether there is a need for access to a gym + accessToCellOffice: boolean // Whether there is a need for access to a cell office + accessToCoworking: boolean // Whether there is a need for access to a coworking space + accessToReception: boolean // Whether there is a need for access to a reception + specialAreaOffice: number // The area of the special office + specialAreaShared: number // The area of the shared space + specialAreaCommon: number // The area of the common space + accessToExercise: boolean // Whether there is a need for access to an exercise area + coworkingShare: number // The percentage of coworking space + touchdownShare: number // The percentage of touchdown space + dockinShare: number // The percentage of dock-in space + cellOfficeShare: number // The percentage of cell office space + landscapeShare: number // The percentage of landscape space + numberOfEmployees: number // The total number of employees + concurrencyAttendanceShare: number // The percentage of concurrent attendance + peakConcurrencyAttendanceShare: number // The peak percentage of concurrent attendance + overCapacityShare: number // The percentage of overcapacity + homeOfficeAverageShare: number // The average percentage of home office + projectroomShare: number // The percentage of project room space + focusroomShare: number // The percentage of focus room space + quietzoneShare: number // The percentage of quiet zone space + miniMeetingroomShare: number // The percentage of mini meeting room space + smallMeetingroomShare: number // The percentage of small meeting room space + mediumMeetingroomShare: number // The percentage of medium meeting room space + largeMeetingroomShare: number // The percentage of large meeting room space +} diff --git a/src/calculations/spaces/common area/auditorium.ts b/src/calculations/spaces/common area/auditorium.ts new file mode 100644 index 0000000..8c4df0a --- /dev/null +++ b/src/calculations/spaces/common area/auditorium.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class Auditorium extends MainSpace { + /** + * Calculates the area of the auditorum. 0 if there is no canteen. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToAuditorium) { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.seatsInAuditorium + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/canteen.ts b/src/calculations/spaces/common area/canteen.ts new file mode 100644 index 0000000..0180ebd --- /dev/null +++ b/src/calculations/spaces/common area/canteen.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class Canteen extends MainSpace { + /** + * Calculates the area of the canteen. 0 if there is no canteen. + * @returns + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCanteen) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/cleaning.ts b/src/calculations/spaces/common area/cleaning.ts new file mode 100644 index 0000000..c0eb733 --- /dev/null +++ b/src/calculations/spaces/common area/cleaning.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonCleaning extends MainSpace { + /** + * Calculates the area of common cleaning. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/course.ts b/src/calculations/spaces/common area/course.ts new file mode 100644 index 0000000..2afe965 --- /dev/null +++ b/src/calculations/spaces/common area/course.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class Course extends MainSpace { + /** + * Calculates the area of course room. 0 if there is no need. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCourseSpace) { + return this.dimensionedAttendance() * this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/exerciseroom.ts b/src/calculations/spaces/common area/exerciseroom.ts new file mode 100644 index 0000000..1100c71 --- /dev/null +++ b/src/calculations/spaces/common area/exerciseroom.ts @@ -0,0 +1,21 @@ +import MainSpace from '../main_space_class' + +export default class CommonExerciseRoom extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the work toilet + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToExercise) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/hwc.ts b/src/calculations/spaces/common area/hwc.ts new file mode 100644 index 0000000..51ee150 --- /dev/null +++ b/src/calculations/spaces/common area/hwc.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class CommonHwc extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the common HWC + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/kitchen.ts b/src/calculations/spaces/common area/kitchen.ts new file mode 100644 index 0000000..fc4ac2a --- /dev/null +++ b/src/calculations/spaces/common area/kitchen.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class Kitchen extends MainSpace { + /** + * Calculates the area of the kitchen based on the number of employees and the area per role. It is 0 if there is no canteen. + * @returns {number} The area of the kitchen + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCanteen) { + return this.spaceConstants.areaPerRole * this.variables.numberOfEmployees + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/lobby.ts b/src/calculations/spaces/common area/lobby.ts new file mode 100644 index 0000000..97b547f --- /dev/null +++ b/src/calculations/spaces/common area/lobby.ts @@ -0,0 +1,26 @@ +import MainSpace from '../main_space_class' +import Auditorium from './auditorium' +import Course from './course' + +export default class Lobby extends MainSpace { + /** + * Calculates the lobby area. 0 if there is no need. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToAuditorium || this.variables.accessToCourseSpace) { + // Get the area of the auditorium and the course space + const auditorium = new Auditorium(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const course = new Course(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const areaPerPerson = this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType + + const x = auditorium.calculateAreaExclCompensation() / areaPerPerson + const y = course.calculateAreaExclCompensation() / this.spaceConstants.areaPerRole * (this.spaceConstants.seatAreaPerSpot ? this.spaceConstants.seatAreaPerSpot : 0) + + // Take the max of the two + const max = Math.max(x, y) + return max * areaPerPerson + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/common area/reception.ts b/src/calculations/spaces/common area/reception.ts new file mode 100644 index 0000000..49e7a88 --- /dev/null +++ b/src/calculations/spaces/common area/reception.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonReception extends MainSpace { + /** + * Calculates the area of common area reception. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/special.ts b/src/calculations/spaces/common area/special.ts new file mode 100644 index 0000000..83aebb9 --- /dev/null +++ b/src/calculations/spaces/common area/special.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonSpecial extends MainSpace { + /** + * Calculates the area of the special rooms. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.variables.specialAreaCommon + } +} diff --git a/src/calculations/spaces/common area/stockarchive.ts b/src/calculations/spaces/common area/stockarchive.ts new file mode 100644 index 0000000..924c560 --- /dev/null +++ b/src/calculations/spaces/common area/stockarchive.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonStockArchive extends MainSpace { + /** + * Calculates the area of stock/archive. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/toilet.ts b/src/calculations/spaces/common area/toilet.ts new file mode 100644 index 0000000..b6d1b2c --- /dev/null +++ b/src/calculations/spaces/common area/toilet.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class CommonToilet extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the work toilet + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/waitingzone.ts b/src/calculations/spaces/common area/waitingzone.ts new file mode 100644 index 0000000..140ddcb --- /dev/null +++ b/src/calculations/spaces/common area/waitingzone.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonWaitingZone extends MainSpace { + /** + * Calculates the area of common waiting zone. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/common area/wardrobe.ts b/src/calculations/spaces/common area/wardrobe.ts new file mode 100644 index 0000000..1f2adbc --- /dev/null +++ b/src/calculations/spaces/common area/wardrobe.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CommonWardrobe extends MainSpace { + /** + * Calculates the area of the common wardrobe + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/dummy.ts b/src/calculations/spaces/dummy.ts new file mode 100644 index 0000000..c2d2240 --- /dev/null +++ b/src/calculations/spaces/dummy.ts @@ -0,0 +1,32 @@ +import MainSpace from './main_space_class' +import { ISpace } from '../interfaces/space' +import { IConfig } from '../interfaces/config' +import { IVariable } from '../interfaces/variable' + +export default class Dummy extends MainSpace { + /** + * Constructor for the dummy space class + */ + constructor(variables: IVariable, config: IConfig) { + const space: ISpace = { + name: "dummy", + result: {}, + constants: { + seatPersonRoom: "person", + areaPerRole: 0, + personsPerType: 1, + minimumPersons: 1, + minimumSquareMeters: 0, + shouldCalculateCompensation: false, + countsTowardsWorkspaceArea: false, + adhereToGovernmentMinimum: false + } + } + super(space, variables, config) + this.name = this.space.name + this.space = space + this.spaceConstants = space.constants + this.variables = variables + this.config = config + } +} diff --git a/src/calculations/spaces/index.ts b/src/calculations/spaces/index.ts new file mode 100644 index 0000000..6a2e2f6 --- /dev/null +++ b/src/calculations/spaces/index.ts @@ -0,0 +1,115 @@ +import Kitchen from './common area/kitchen' +import Canteen from './common area/canteen' +import Auditorium from './common area/auditorium' +import Course from './common area/course' +import Lobby from './common area/lobby' +import WorkTouchdown from './work related area/touchdown' +import SharedTouchdown from './shared area/touchdown' +import WorkDockin from './work related area/dockin' +import SharedDockin from './shared area/dockin' +import CellOffice from './work related area/celloffice' +import Landscape from './work related area/landscape' +import Projectroom from './work related area/projectroom' +import Focusroom from './work related area/focusroom' +import Quietzone from './work related area/quietzone' +import CoffeeStation from './work related area/coffeestation' +import MiniKitchenLounge from './work related area/minikitchenlounge' +import SharedSmallMeetingroom from './shared area/meetingroom_small' +import SharedMediumMeetingroom from './shared area/meetingroom_medium' +import SharedLargeMeetingroom from './shared area/meetingroom_large' +import WorkMiniMeetingroom from './work related area/meetingroom_mini' +import WorkSmallMeetingroom from './work related area/meetingroom_small' +import WorkMediumMeetingroom from './work related area/meetingroom_medium' +import WorkLargeMeetingroom from './work related area/meetingroom_large' +import SharedMultiroom from './shared area/multiroom' +import WorkMultiroom from './work related area/multiroom' +import WorkReception from './work related area/reception' +import SharedReception from './shared area/reception' +import WorkWaitingZone from './work related area/waitingzone' +import SharedWaitingZone from './shared area/waitingzone' +import CopyArchive from './work related area/copyarchive' +import Cleaning from './work related area/cleaning' +import WorkPersonalStorage from './work related area/personalstorage' +import SharedPersonalStorage from './shared area/personalstorage' +import WorkWardrobe from './work related area/wardrobe' +import SharedWardrobe from './shared area/wardrobe' +import WorkToilet from './work related area/toilet' +import SharedToilet from './shared area/toilet' +import WorkHwc from './work related area/hwc' +import SharedHwc from './shared area/hwc' +import WorkSpecial from './work related area/special' +import CommonReception from './common area/reception' +import CommonWaitingZone from './common area/waitingzone' +import CommonStockArchive from './common area/stockarchive' +import CommonCleaning from './common area/cleaning' +import CommonToilet from './common area/toilet' +import CommonHwc from './common area/hwc' +import CommonWardrobe from './common area/wardrobe' +import CommonExerciseRoom from './common area/exerciseroom' +import SharedSpecial from './shared area/special' +import CommonSpecial from './common area/special' +import SharedProjectroom from './shared area/projectroom' +import SharedLounge from './shared area/lounge' + +/** + * This function returns the correct space class based on the space name. + * @param {string} spaceName - The name of the space + * @returns {any} + */ +const getSpace = (spaceName: string): any => { // eslint-disable-line @typescript-eslint/no-explicit-any + if (spaceName === 'kitchen') return Kitchen + else if (spaceName === 'canteen') return Canteen + else if (spaceName === 'auditorium') return Auditorium + else if (spaceName === 'course') return Course + else if (spaceName === 'lobby') return Lobby + else if (spaceName === 'touchdown – work related area') return WorkTouchdown + else if (spaceName === 'touchdown – shared area') return SharedTouchdown + else if (spaceName === 'dock in – work related area') return WorkDockin + else if (spaceName === 'dock in – shared area') return SharedDockin + else if (spaceName === 'cell office') return CellOffice + else if (spaceName === 'landscape') return Landscape + else if (spaceName === 'project room') return Projectroom + else if (spaceName === 'focus room') return Focusroom + else if (spaceName === 'quiet zone') return Quietzone + else if (spaceName === 'coffee station') return CoffeeStation + else if (spaceName === 'mini kitchen lounge') return MiniKitchenLounge + else if (spaceName === 'small meeting room – shared area') return SharedSmallMeetingroom + else if (spaceName === 'medium meeting room – shared area') return SharedMediumMeetingroom + else if (spaceName === 'large meeting room – shared area') return SharedLargeMeetingroom + else if (spaceName === 'mini meeting room – work related area') return WorkMiniMeetingroom + else if (spaceName === 'small meeting room – work related area') return WorkSmallMeetingroom + else if (spaceName === 'medium meeting room – work related area') return WorkMediumMeetingroom + else if (spaceName === 'large meeting room – work related area') return WorkLargeMeetingroom + else if (spaceName === 'multi room – shared area') return SharedMultiroom + else if (spaceName === 'multi room – work related area') return WorkMultiroom + else if (spaceName === 'reception – work related area') return WorkReception + else if (spaceName === 'reception – shared area') return SharedReception + else if (spaceName === 'waiting zone – work related area') return WorkWaitingZone + else if (spaceName === 'waiting zone – shared area') return SharedWaitingZone + else if (spaceName === 'copy archive') return CopyArchive + else if (spaceName === 'cleaning') return Cleaning + else if (spaceName === 'personal storage – work related area') return WorkPersonalStorage + else if (spaceName === 'personal storage – shared area') return SharedPersonalStorage + else if (spaceName === 'wardrobe – work related area') return WorkWardrobe + else if (spaceName === 'wardrobe – shared area') return SharedWardrobe + else if (spaceName === 'toilet – work related area') return WorkToilet + else if (spaceName === 'toilet – shared area') return SharedToilet + else if (spaceName === 'hwc – work related area') return WorkHwc + else if (spaceName === 'hwc – shared area') return SharedHwc + else if (spaceName === 'special – work related area') return WorkSpecial + else if (spaceName === 'reception – common area') return CommonReception + else if (spaceName === 'waiting zone – common area') return CommonWaitingZone + else if (spaceName === 'stock archive – common area') return CommonStockArchive + else if (spaceName === 'cleaning – common area') return CommonCleaning + else if (spaceName === 'toilet – common area') return CommonToilet + else if (spaceName === 'hwc – common area') return CommonHwc + else if (spaceName === 'wardrobe – common area') return CommonWardrobe + else if (spaceName === 'exercise room – common area') return CommonExerciseRoom + else if (spaceName === 'special – shared area') return SharedSpecial + else if (spaceName === 'special – common area') return CommonSpecial + else if (spaceName === 'project room – shared area') return SharedProjectroom + else if (spaceName === 'lounge area – shared area') return SharedLounge + else throw new Error(`Space ${spaceName} not found`) +} + +export default getSpace diff --git a/src/calculations/spaces/main_space_class.ts b/src/calculations/spaces/main_space_class.ts new file mode 100644 index 0000000..4e73ee5 --- /dev/null +++ b/src/calculations/spaces/main_space_class.ts @@ -0,0 +1,279 @@ +import { ISpace } from '../interfaces/space' +import { IConfig } from '../interfaces/config' +import { ISpaceConstant } from '../interfaces/space_constant' +import { ISpaceCalculation } from '../interfaces/space_calculation' +import { IConstant } from '../interfaces/constant' +import { IVariable } from '../interfaces/variable' + +type TCustomSpaceConstants = {[key:string]:ISpaceConstant} + +/** + * This is the main class for the spaces. It contains shared methods for all spaces. + */ +export default class MainSpace implements ISpaceCalculation { + name: string + constants: IConstant + customConstants?: IConstant + space: ISpace + spaceConstants: ISpaceConstant + variables: IVariable + customSpaceConstants?: TCustomSpaceConstants + config: IConfig + + /** + * Constructor for the main space class. + * @param {Ispace} space - The space to calculate + * @param {IVariable} variables - The variables to use + * @param {IConfig} config - The config to use + * @param {TCustomSpaceConstants} [customSpaceConstants] - Custom space constants, optional + * @param {IConstant} [customConstants] - Custom constants, optional + */ + constructor(space: ISpace, variables: IVariable, config: IConfig, customSpaceConstants?: TCustomSpaceConstants, customConstants?: IConstant) { + this.name = space.name + this.space = space + this.spaceConstants = space.constants + this.variables = variables + this.config = config + this.customSpaceConstants = customSpaceConstants + // Merge custom constants with the default constants + this.constants = { + ...config.constants, + ...customConstants + } + // Merge custom space constants with the default space constants + for (const spaceName in this.customSpaceConstants) { + if (this.name === spaceName) { + this.spaceConstants = { + ...space.constants, + ...this.customSpaceConstants[spaceName] + } + } + } + } + + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType + } + + /** + * Peak attendance is the number of employees times the peak concurrency attendance share. + * @returns {number} + */ + #peakAttendance (): number { + return this.variables.numberOfEmployees * this.variables.peakConcurrencyAttendanceShare + } + + /** + * This method calculates the added peak area for this space. + * @returns {number} + */ + addedPeakArea (): number { + return this.#diffPeak() * this.areaPerPersonExcludingCorridor() + } + + /** + * This method gets number of person per workspace type + * @returns {number} + */ + personsPerType (): number { + return this.dimensionedAttendance() * this.sharePerWorkspaceType() + } + + /** + * This method gets number of person per workspace type, adjusted + * @todo Missing implementation + * @returns {number} + */ + personsPerTypeAdjusted (): number { + return 0 + } + + /** + * This method computes the share of this workspace type. + * @returns {number} + */ + sharePerWorkspaceType (): number { + return 0 + } + + /** + * This method calculates the dimensioned attendance for this space. + * @todo Write detailed description of this method + * @returns {number} + */ + dimensionedAttendance (): number { + return (this.variables.numberOfEmployees * this.variables.concurrencyAttendanceShare) + + (this.variables.numberOfEmployees * this.variables.concurrencyAttendanceShare * this.variables.overCapacityShare) - + (this.variables.numberOfEmployees * this.variables.homeOfficeAverageShare) + } + + /** + * This method calculates the difference between peak attendance and dimensioned attendance. + * @returns {number} + */ + #diffPeak (): number { + return this.#peakAttendance() - this.dimensionedAttendance() + } + + /** + * This method calculates the government required area for this space. + * @returns {number} + */ + #governmentRequiredArea (): number { + return this.dimensionedAttendance() * this.constants.governmentMinimumSquaremetersPerWorkSpace + } + + /** + * Calculate remainder area for space. + * @returns {number} + */ + #remainderArea (): number { + if (this.spaceConstants.seatPersonRoom === 'seat') { + return this.space.result.adjustedAreaInclCompensation! - (this.#areaPerSeatUnAdjusted() * this.calculateNumberOfSeats()) + } else if (this.spaceConstants.seatPersonRoom === 'room') { + return this.space.result.adjustedAreaInclCompensation! - (this.spaceConstants.minimumSquareMeters! * this.calculateNumberOfRooms()) + } + return 0 + } + + #areaPerSeatUnAdjusted (): number { + if (this.space.result.adjustedAreaInclCompensation! > 0 && this.calculateNumberOfSeats() > 0) { + return Math.round(this.space.result.adjustedAreaInclCompensation! / this.calculateNumberOfSeats() * 10) / 10 + } + return 0 + } + + /** + * Calculates number of rooms + * @returns {number} + */ + calculateNumberOfRooms (): number { + if (this.spaceConstants.shouldCalculateNumberOfRooms) { + return Math.ceil(this.space.result.adjustedAreaInclCompensation! / this.spaceConstants.minimumSquareMeters!) + } + return 0 + } + + /** + * Calculates number of seats. + * @returns {number} + */ + calculateNumberOfSeats (): number { + if (this.spaceConstants.shouldCalculateNumberOfSeats) { + return Math.round((this.space.result.employeesPerWorkplaceTypeAdjusted || 0) + this.#employeesPerSeatTypeInPeak()) + } + return 0 + } + + /** + * Calculates employees per seat type in peak as integer + * @returns {number} + */ + #employeesPerSeatTypeInPeak (): number { + return Math.round(this.#diffPeak() * this.sharePerWorkspaceType()) + } + + /** + * Calculates un-adjusted add on part. + * @param {number} totalWorkplaceArea - The total area of all workspaces already calculated + * @param {number} totalCompensationArea - The total area of all spaces requiring compensation + * @returns {number} + */ + calculateNotAdjustedAddonPart (totalWorkplaceArea: number, totalCompensationArea: number): number { + if (this.spaceConstants.shouldCalculateCompensation && totalWorkplaceArea < this.#governmentRequiredArea()) { + return (this.#governmentRequiredArea() / totalWorkplaceArea) * (this.space.result.areaExclCompensation! / totalCompensationArea) + } + return 0 + } + + /** + * Calculates un-adjusted add on area. + * @param {number} totalWorkplaceArea - The total area of all workspaces already calculated + * @param {number} totalCompensationArea - The total area of all spaces requiring compensation + * @returns {number} + */ + calculateNotAdjustedAddonArea (totalWorkplaceArea: number, totalCompensationArea: number): number { + if (this.spaceConstants.shouldCalculateCompensation && totalWorkplaceArea < this.#governmentRequiredArea()) { + return this.calculateNotAdjustedAddonPart(totalWorkplaceArea, totalCompensationArea) * this.space.result.areaExclCompensation! + } + return 0 + } + + /** + * Calculates the difference between the government required area and the total workplace area. + * @param {number} totalWorkplaceArea - The total area of all workspaces already calculated + * @returns {number} + */ + #differenceGovernnmentRequiredAreaWorkplaceArea (totalWorkplaceArea: number): number { + return this.#governmentRequiredArea() - totalWorkplaceArea + } + + /** + * Calculates employees per workplace type, unadjusted. + * @returns {number} + */ + calculateEmployeesPerWorkplaceTypeUnadjusted (): number { + return Math.round(this.dimensionedAttendance() * this.sharePerWorkspaceType()) + } + + /** + * Calculates employees per workplace type, adjusted + * @param {number} totalEmployeesPerWorkplaceTypeUnadjusted - The total number of employees per workplace type, unadjusted + */ + calculateEmployeesPerWorkplaceTypeAdjusted (totalEmployeesPerWorkplaceTypeUnadjusted: number, allEmployeesPerWorkplaceTypeUnadjusted: number[]): number { + if (totalEmployeesPerWorkplaceTypeUnadjusted > this.dimensionedAttendance() && this.calculateEmployeesPerWorkplaceTypeUnadjusted() === Math.max(...allEmployeesPerWorkplaceTypeUnadjusted)) { + return this.calculateEmployeesPerWorkplaceTypeUnadjusted() - (totalEmployeesPerWorkplaceTypeUnadjusted - this.dimensionedAttendance()) + } else if (totalEmployeesPerWorkplaceTypeUnadjusted < this.dimensionedAttendance() && this.calculateEmployeesPerWorkplaceTypeUnadjusted() === Math.max(...allEmployeesPerWorkplaceTypeUnadjusted)) { + return this.calculateEmployeesPerWorkplaceTypeUnadjusted() + (this.dimensionedAttendance() - totalEmployeesPerWorkplaceTypeUnadjusted) + } else { + return this.calculateEmployeesPerWorkplaceTypeUnadjusted() + } + } + + /** + * Calculates adjusted add on area. + * @param {number} totalWorkplaceArea - The total area of all workspaces already calculated + * @param {number} totalCompensationArea - The total area of all spaces requiring compensation + * @param {number} totalUnadjustedArea - The total unadjusted area + * @returns {number} + */ + calculateAdjustedAddonArea (totalWorkplaceArea: number, totalCompensationArea: number, totalUnadjustedArea: number): number { + if (this.spaceConstants.shouldCalculateCompensation && totalWorkplaceArea < this.#governmentRequiredArea()) { + return this.calculateNotAdjustedAddonArea(totalWorkplaceArea, totalCompensationArea) / totalUnadjustedArea * this.#differenceGovernnmentRequiredAreaWorkplaceArea(totalWorkplaceArea) + } + return 0 + } + + /** + * Default is returning the areaInclCompensation but is overridden in some spaces. + * @param {number} totalWorkplaceArea - The total area of all workspaces already calculated + * @param {number} totalCompensationArea - The total area of all spaces requiring compensation + * @param {number} totalUnadjustedArea - The total unadjusted area + * @returns {number} + */ + calculateAdjustedAreaInclCompensation(totalWorkplaceArea: number, totalCompensationArea: number, totalUnadjustedArea: number): number { + if (this.spaceConstants.shouldCalculateCompensation) { + return this.calculateAdjustedAddonArea(totalWorkplaceArea, totalCompensationArea, totalUnadjustedArea) + this.space.result.areaExclCompensation! + } + return this.space.result.areaExclCompensation! + } + + /** + * Calculates the adjusted area including adjustment and compensation. + */ + calculateAdjustedAreaInclCompensationWithAdjustmentAndCompensation (): number { + return this.space.result.adjustedAreaInclCompensation! - this.#remainderArea() + } + + /** + * Calculates the area excluding compensation. Not implemented in this class. + * @returns {number} + */ + calculateAreaExclCompensation (): number { + return 0 + } +} diff --git a/src/calculations/spaces/shared area/dockin.ts b/src/calculations/spaces/shared area/dockin.ts new file mode 100644 index 0000000..6f54fd5 --- /dev/null +++ b/src/calculations/spaces/shared area/dockin.ts @@ -0,0 +1,25 @@ +import MainSpace from '../main_space_class' + +export default class SharedDockin extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + if (this.variables.accessToCoworking) { + return this.variables.dockinShare * this.variables.coworkingShare + } + return 0 + } + + /** + * Calculates the area of shared dockin. 0 if there is no desire. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.personsPerType() * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/hwc.ts b/src/calculations/spaces/shared area/hwc.ts new file mode 100644 index 0000000..d3df86b --- /dev/null +++ b/src/calculations/spaces/shared area/hwc.ts @@ -0,0 +1,21 @@ +import MainSpace from '../main_space_class' + +export default class SharedHwc extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the shared hwc. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/lounge.ts b/src/calculations/spaces/shared area/lounge.ts new file mode 100644 index 0000000..11cf4be --- /dev/null +++ b/src/calculations/spaces/shared area/lounge.ts @@ -0,0 +1,15 @@ +import MainSpace from '../main_space_class' + +export default class SharedLounge extends MainSpace { + /** + * Calculates the area of shared dockin. 0 if there is no desire. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + // SUM((AJ4*AR24*AQ24)+(AB24*AQ24)) + if (this.variables.accessToCoworking) { + return (this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType()) + (this.addedPeakArea() * this.sharePerWorkspaceType()) + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/meetingroom_large.ts b/src/calculations/spaces/shared area/meetingroom_large.ts new file mode 100644 index 0000000..bdbfe21 --- /dev/null +++ b/src/calculations/spaces/shared area/meetingroom_large.ts @@ -0,0 +1,35 @@ +import MainSpace from '../main_space_class' +import WorkDockin from './dockin' +import CellOffice from '../work related area/celloffice' +import Landscape from '../work related area/landscape' +import Projectroom from '../work related area/projectroom' +import Focusroom from '../work related area/focusroom' +import Quietzone from '../work related area/quietzone' + +export default class SharedLargeMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.largeMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/meetingroom_medium.ts b/src/calculations/spaces/shared area/meetingroom_medium.ts new file mode 100644 index 0000000..7e46c00 --- /dev/null +++ b/src/calculations/spaces/shared area/meetingroom_medium.ts @@ -0,0 +1,35 @@ +import MainSpace from '../main_space_class' +import WorkDockin from './dockin' +import CellOffice from '../work related area/celloffice' +import Landscape from '../work related area/landscape' +import Projectroom from '../work related area/projectroom' +import Focusroom from '../work related area/focusroom' +import Quietzone from '../work related area/quietzone' + +export default class SharedMediumMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.mediumMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/meetingroom_small.ts b/src/calculations/spaces/shared area/meetingroom_small.ts new file mode 100644 index 0000000..862ddf1 --- /dev/null +++ b/src/calculations/spaces/shared area/meetingroom_small.ts @@ -0,0 +1,35 @@ +import MainSpace from '../main_space_class' +import WorkDockin from './dockin' +import CellOffice from '../work related area/celloffice' +import Landscape from '../work related area/landscape' +import Projectroom from '../work related area/projectroom' +import Focusroom from '../work related area/focusroom' +import Quietzone from '../work related area/quietzone' + +export default class SharedSmallMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.smallMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/multiroom.ts b/src/calculations/spaces/shared area/multiroom.ts new file mode 100644 index 0000000..9023dde --- /dev/null +++ b/src/calculations/spaces/shared area/multiroom.ts @@ -0,0 +1,28 @@ +import MainSpace from '../main_space_class' +import WorkDockin from '../work related area/dockin' +import CellOffice from '../work related area/celloffice' +import Landscape from '../work related area/landscape' +import Projectroom from '../work related area/projectroom' +import Focusroom from '../work related area/focusroom' +import Quietzone from '../work related area/quietzone' + +export default class SharedMultiroom extends MainSpace { + /** + * Calculates the area of shared multiroom. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + //=SUM((AP4:AP10)*AR26*AQ26)+(AB26) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/personalstorage.ts b/src/calculations/spaces/shared area/personalstorage.ts new file mode 100644 index 0000000..965ae59 --- /dev/null +++ b/src/calculations/spaces/shared area/personalstorage.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class SharedPersonalStorage extends MainSpace { + /** + * Calculates the area of the personal storage + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.calculateEmployeesPerWorkplaceTypeUnadjusted() * this.sharePerWorkspaceType() * this.areaPerPersonExcludingCorridor() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/projectroom.ts b/src/calculations/spaces/shared area/projectroom.ts new file mode 100644 index 0000000..1161f2f --- /dev/null +++ b/src/calculations/spaces/shared area/projectroom.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class SharedProjectroom extends MainSpace { + /** + * Calculates the area of the shred project room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToReception) { + return this.calculateEmployeesPerWorkplaceTypeUnadjusted() * this.areaPerPersonExcludingCorridor() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/reception.ts b/src/calculations/spaces/shared area/reception.ts new file mode 100644 index 0000000..43b59a3 --- /dev/null +++ b/src/calculations/spaces/shared area/reception.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class SharedReception extends MainSpace { + /** + * Calculates the area of shared reception. 0 if there is no desire. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToReception) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/special.ts b/src/calculations/spaces/shared area/special.ts new file mode 100644 index 0000000..c64ae50 --- /dev/null +++ b/src/calculations/spaces/shared area/special.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class SharedSpecial extends MainSpace { + /** + * Calculates the area of the special rooms. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.variables.specialAreaShared + } +} diff --git a/src/calculations/spaces/shared area/toilet.ts b/src/calculations/spaces/shared area/toilet.ts new file mode 100644 index 0000000..910d22f --- /dev/null +++ b/src/calculations/spaces/shared area/toilet.ts @@ -0,0 +1,21 @@ +import MainSpace from '../main_space_class' + +export default class SharedToilet extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the shared toilet. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/touchdown.ts b/src/calculations/spaces/shared area/touchdown.ts new file mode 100644 index 0000000..8298ac3 --- /dev/null +++ b/src/calculations/spaces/shared area/touchdown.ts @@ -0,0 +1,25 @@ +import MainSpace from '../main_space_class' + +export default class SharedTouchdown extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + if (this.variables.accessToCoworking) { + return this.variables.touchdownShare * this.variables.coworkingShare + } + return 0 + } + + /** + * Calculates the area of shared touchdown. 0 if there is no desire. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.personsPerType() * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/waitingzone.ts b/src/calculations/spaces/shared area/waitingzone.ts new file mode 100644 index 0000000..0a2dd5e --- /dev/null +++ b/src/calculations/spaces/shared area/waitingzone.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class SharedWaitingZone extends MainSpace { + /** + * Calculates the area of shared wating zone in reception. 0 if there is no desire. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToReception) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + } + return 0 + } +} diff --git a/src/calculations/spaces/shared area/wardrobe.ts b/src/calculations/spaces/shared area/wardrobe.ts new file mode 100644 index 0000000..592474f --- /dev/null +++ b/src/calculations/spaces/shared area/wardrobe.ts @@ -0,0 +1,14 @@ +import MainSpace from '../main_space_class' + +export default class SharedWardrobe extends MainSpace { + /** + * Calculates the area of the wardrobe + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToCoworking) { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.variables.coworkingShare + } + return 0 + } +} diff --git a/src/calculations/spaces/work related area/celloffice.ts b/src/calculations/spaces/work related area/celloffice.ts new file mode 100644 index 0000000..ea4ddbd --- /dev/null +++ b/src/calculations/spaces/work related area/celloffice.ts @@ -0,0 +1,22 @@ +import MainSpace from '../main_space_class' + +export default class CellOffice extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + if (this.variables.accessToCellOffice) { + return this.variables.cellOfficeShare + } + return 0 + } + + /** + * Calculates the area of the cell. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + } +} diff --git a/src/calculations/spaces/work related area/cleaning.ts b/src/calculations/spaces/work related area/cleaning.ts new file mode 100644 index 0000000..a7d7d54 --- /dev/null +++ b/src/calculations/spaces/work related area/cleaning.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class Cleaning extends MainSpace { + /** + * Calculates the area of cleaning + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/work related area/coffeestation.ts b/src/calculations/spaces/work related area/coffeestation.ts new file mode 100644 index 0000000..35e0e15 --- /dev/null +++ b/src/calculations/spaces/work related area/coffeestation.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class CoffeeStation extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the coffee station. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } +} diff --git a/src/calculations/spaces/work related area/copyarchive.ts b/src/calculations/spaces/work related area/copyarchive.ts new file mode 100644 index 0000000..51562c4 --- /dev/null +++ b/src/calculations/spaces/work related area/copyarchive.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class CopyArchive extends MainSpace { + /** + * Calculates the area of the copy/archive + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/work related area/dockin.ts b/src/calculations/spaces/work related area/dockin.ts new file mode 100644 index 0000000..ea839fa --- /dev/null +++ b/src/calculations/spaces/work related area/dockin.ts @@ -0,0 +1,22 @@ +import MainSpace from '../main_space_class' +import SharedDockin from '../shared area/dockin' + +export default class WorkDockin extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + // We need the shared dock in share for the calculation of the work dock in share + const sharedDockin = new SharedDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return this.variables.dockinShare - sharedDockin.sharePerWorkspaceType() + } + + /** + * Calculates the area of touchdown. 0 if there is no need. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return (this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor()*this.sharePerWorkspaceType())+(this.addedPeakArea()*this.sharePerWorkspaceType()) + } +} diff --git a/src/calculations/spaces/work related area/focusroom.ts b/src/calculations/spaces/work related area/focusroom.ts new file mode 100644 index 0000000..2bb40d2 --- /dev/null +++ b/src/calculations/spaces/work related area/focusroom.ts @@ -0,0 +1,19 @@ +import MainSpace from '../main_space_class' + +export default class Focusroom extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + return this.variables.focusroomShare + } + + /** + * Calculates the area of the project room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + } +} diff --git a/src/calculations/spaces/work related area/hwc.ts b/src/calculations/spaces/work related area/hwc.ts new file mode 100644 index 0000000..0ce6308 --- /dev/null +++ b/src/calculations/spaces/work related area/hwc.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class WorkHwc extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the work hwc. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/work related area/landscape.ts b/src/calculations/spaces/work related area/landscape.ts new file mode 100644 index 0000000..da35dcb --- /dev/null +++ b/src/calculations/spaces/work related area/landscape.ts @@ -0,0 +1,24 @@ +import MainSpace from '../main_space_class' + +export default class Landscape extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + if (this.variables.accessToCellOffice) { + return this.variables.landscapeShare + } else { + return this.variables.landscapeShare + this.variables.cellOfficeShare + } + return 0 + } + + /** + * Calculates the area of the landscape. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + } +} diff --git a/src/calculations/spaces/work related area/meetingroom_large.ts b/src/calculations/spaces/work related area/meetingroom_large.ts new file mode 100644 index 0000000..0032e90 --- /dev/null +++ b/src/calculations/spaces/work related area/meetingroom_large.ts @@ -0,0 +1,37 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import CellOffice from './celloffice' +import Landscape from './landscape' +import Projectroom from './projectroom' +import Focusroom from './focusroom' +import Quietzone from './quietzone' +import SharedLargeMeetingroom from '../shared area/meetingroom_large' + +export default class WorkLargeMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.largeMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + // =SUM((AP4:AP10)*AR13)+(AB13)-(K27) + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + const sharedLargeMeetingRoom = new SharedLargeMeetingroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() - sharedLargeMeetingRoom.calculateAreaExclCompensation() + } +} diff --git a/src/calculations/spaces/work related area/meetingroom_medium.ts b/src/calculations/spaces/work related area/meetingroom_medium.ts new file mode 100644 index 0000000..dfff61e --- /dev/null +++ b/src/calculations/spaces/work related area/meetingroom_medium.ts @@ -0,0 +1,37 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import CellOffice from './celloffice' +import Landscape from './landscape' +import Projectroom from './projectroom' +import Focusroom from './focusroom' +import Quietzone from './quietzone' +import SharedMediumMeetingroom from '../shared area/meetingroom_medium' + +export default class WorkMediumMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.mediumMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + // =SUM((AP4:AP10)*AR13)+(AB13)-(K27) + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + const sharedMediumMeetingRoom = new SharedMediumMeetingroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() - sharedMediumMeetingRoom.calculateAreaExclCompensation() + } +} diff --git a/src/calculations/spaces/work related area/meetingroom_mini.ts b/src/calculations/spaces/work related area/meetingroom_mini.ts new file mode 100644 index 0000000..43158b0 --- /dev/null +++ b/src/calculations/spaces/work related area/meetingroom_mini.ts @@ -0,0 +1,34 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import CellOffice from './celloffice' +import Landscape from './landscape' +import Projectroom from './projectroom' +import Focusroom from './focusroom' +import Quietzone from './quietzone' + +export default class WorkMiniMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.miniMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } +} diff --git a/src/calculations/spaces/work related area/meetingroom_small.ts b/src/calculations/spaces/work related area/meetingroom_small.ts new file mode 100644 index 0000000..309640d --- /dev/null +++ b/src/calculations/spaces/work related area/meetingroom_small.ts @@ -0,0 +1,37 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import CellOffice from './celloffice' +import Landscape from './landscape' +import Projectroom from './projectroom' +import Focusroom from './focusroom' +import Quietzone from './quietzone' +import SharedSmallMeetingRoom from '../shared area/meetingroom_small' + +export default class WorkSmallMeetingroom extends MainSpace { + /** + * Calculates the area per person for this space. + * @returns {number} + */ + areaPerPersonExcludingCorridor(): number { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.variables.smallMeetingroomShare + } + + /** + * Calculates the area of the meeting room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + // =SUM((AP4:AP10)*AR13)+(AB13)-(K27) + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const cellOffice = new CellOffice(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const projectroom = new Projectroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + cellOffice.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + projectroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + const sharedSmallMeetingRoom = new SharedSmallMeetingRoom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() - sharedSmallMeetingRoom.calculateAreaExclCompensation() + } +} diff --git a/src/calculations/spaces/work related area/minikitchenlounge.ts b/src/calculations/spaces/work related area/minikitchenlounge.ts new file mode 100644 index 0000000..77bc4e7 --- /dev/null +++ b/src/calculations/spaces/work related area/minikitchenlounge.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class MiniKitchenLounge extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType + } + + /** + * Calculates the area of the zone. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + this.addedPeakArea() + } +} diff --git a/src/calculations/spaces/work related area/multiroom.ts b/src/calculations/spaces/work related area/multiroom.ts new file mode 100644 index 0000000..fb7b33f --- /dev/null +++ b/src/calculations/spaces/work related area/multiroom.ts @@ -0,0 +1,26 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import Landscape from './landscape' +import Focusroom from './focusroom' + +export default class WorkMultiroom extends MainSpace { + /** + * Calculates the area of the multi room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const landscape = new Landscape(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + + const touchdownDockinLandscapeSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + landscape.calculateEmployeesPerWorkplaceTypeUnadjusted() + + if (touchdownDockinLandscapeSum / 15 * 2 * this.areaPerPersonExcludingCorridor() > focusroom.calculateAreaExclCompensation()) { + return touchdownDockinLandscapeSum / 15 * 2 * this.areaPerPersonExcludingCorridor() - focusroom.calculateAreaExclCompensation() + } else { + return touchdownDockinLandscapeSum / 15 * 2 * this.areaPerPersonExcludingCorridor() + } + } +} diff --git a/src/calculations/spaces/work related area/personalstorage.ts b/src/calculations/spaces/work related area/personalstorage.ts new file mode 100644 index 0000000..cb46ee3 --- /dev/null +++ b/src/calculations/spaces/work related area/personalstorage.ts @@ -0,0 +1,22 @@ +import MainSpace from '../main_space_class' +import WorkTouchdown from './touchdown' +import WorkDockin from './dockin' +import Focusroom from './focusroom' +import Quietzone from './quietzone' +import SharedPersonalStorage from '../shared area/personalstorage' + +export default class WorkPersonalStorage extends MainSpace { + /** + * Calculates the area of the personal storage + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + const workTouchdown = new WorkTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const workDockin = new WorkDockin(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const focusroom = new Focusroom(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const quietzone = new Quietzone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const sharedPersonalStorage = new SharedPersonalStorage(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + const addedPeakAreaSum = workTouchdown.calculateEmployeesPerWorkplaceTypeUnadjusted() + workDockin.calculateEmployeesPerWorkplaceTypeUnadjusted() + focusroom.calculateEmployeesPerWorkplaceTypeUnadjusted() + quietzone.calculateEmployeesPerWorkplaceTypeUnadjusted() + return addedPeakAreaSum * this.areaPerPersonExcludingCorridor() - sharedPersonalStorage.calculateAreaExclCompensation() + } +} diff --git a/src/calculations/spaces/work related area/projectroom.ts b/src/calculations/spaces/work related area/projectroom.ts new file mode 100644 index 0000000..de02240 --- /dev/null +++ b/src/calculations/spaces/work related area/projectroom.ts @@ -0,0 +1,19 @@ +import MainSpace from '../main_space_class' + +export default class Projectroom extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + return this.variables.projectroomShare + } + + /** + * Calculates the area of the project room. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + } +} diff --git a/src/calculations/spaces/work related area/quietzone.ts b/src/calculations/spaces/work related area/quietzone.ts new file mode 100644 index 0000000..12038f6 --- /dev/null +++ b/src/calculations/spaces/work related area/quietzone.ts @@ -0,0 +1,19 @@ +import MainSpace from '../main_space_class' + +export default class Quietzone extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + return this.variables.quietzoneShare + } + + /** + * Calculates the area of the quiet zone. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + } +} diff --git a/src/calculations/spaces/work related area/reception.ts b/src/calculations/spaces/work related area/reception.ts new file mode 100644 index 0000000..a28d478 --- /dev/null +++ b/src/calculations/spaces/work related area/reception.ts @@ -0,0 +1,16 @@ +import MainSpace from '../main_space_class' +import SharedReception from '../shared area/reception' + +export default class WorkReception extends MainSpace { + /** + * Calculates the area of the reception. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToReception) { + const sharedReception = new SharedReception(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() - sharedReception.calculateAreaExclCompensation() + } + return 0 + } +} diff --git a/src/calculations/spaces/work related area/special.ts b/src/calculations/spaces/work related area/special.ts new file mode 100644 index 0000000..e444f53 --- /dev/null +++ b/src/calculations/spaces/work related area/special.ts @@ -0,0 +1,11 @@ +import MainSpace from '../main_space_class' + +export default class WorkSpecial extends MainSpace { + /** + * Calculates the area of the special rooms. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.variables.specialAreaOffice + } +} diff --git a/src/calculations/spaces/work related area/toilet.ts b/src/calculations/spaces/work related area/toilet.ts new file mode 100644 index 0000000..8c8881e --- /dev/null +++ b/src/calculations/spaces/work related area/toilet.ts @@ -0,0 +1,18 @@ +import MainSpace from '../main_space_class' + +export default class WorkToilet extends MainSpace { + /** + * Area per person excluding corridor + */ + areaPerPersonExcludingCorridor = (): number => { + return this.spaceConstants.areaPerRole / this.spaceConstants.personsPerType * this.spaceConstants.unitsPerPerson! + } + + /** + * Calculates the area of the work toilet. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() + } +} diff --git a/src/calculations/spaces/work related area/touchdown.ts b/src/calculations/spaces/work related area/touchdown.ts new file mode 100644 index 0000000..052780b --- /dev/null +++ b/src/calculations/spaces/work related area/touchdown.ts @@ -0,0 +1,22 @@ +import MainSpace from '../main_space_class' +import SharedTouchdown from '../shared area/touchdown' + +export default class WorkTouchdown extends MainSpace { + /** + * This method computes the share of this workspace type + * @returns {number} + */ + sharePerWorkspaceType = (): number => { + // We need the shared touchdown share for the calculation of the work touchdown share + const sharedTouchdown = new SharedTouchdown(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return this.variables.touchdownShare - sharedTouchdown.sharePerWorkspaceType() + } + + /** + * Calculates the area of touchdown. 0 if there is no need. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() * this.sharePerWorkspaceType() + this.addedPeakArea() + } +} diff --git a/src/calculations/spaces/work related area/waitingzone.ts b/src/calculations/spaces/work related area/waitingzone.ts new file mode 100644 index 0000000..8ef7a13 --- /dev/null +++ b/src/calculations/spaces/work related area/waitingzone.ts @@ -0,0 +1,16 @@ +import MainSpace from '../main_space_class' +import SharedWaitingZone from '../shared area/waitingzone' + +export default class WorkWaitingZone extends MainSpace { + /** + * Calculates the area of the reception. + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + if (this.variables.accessToReception) { + const sharedWaitingZone = new SharedWaitingZone(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() - sharedWaitingZone.calculateAreaExclCompensation() + } + return 0 + } +} diff --git a/src/calculations/spaces/work related area/wardrobe.ts b/src/calculations/spaces/work related area/wardrobe.ts new file mode 100644 index 0000000..8b4b9c0 --- /dev/null +++ b/src/calculations/spaces/work related area/wardrobe.ts @@ -0,0 +1,13 @@ +import MainSpace from '../main_space_class' +import SharedWardrobe from '../shared area/wardrobe' + +export default class WorkWardrobe extends MainSpace { + /** + * Calculates the area of the personal storage + * @returns {number} + */ + calculateAreaExclCompensation = (): number => { + const sharedWardrobe = new SharedWardrobe(this.space, this.variables, this.config, this.customSpaceConstants, this.customConstants) + return this.dimensionedAttendance() * this.areaPerPersonExcludingCorridor() - sharedWardrobe.calculateAreaExclCompensation() + } +} diff --git a/src/config/default.json b/src/config/default.json new file mode 100644 index 0000000..fc0caa1 --- /dev/null +++ b/src/config/default.json @@ -0,0 +1,965 @@ +{ + "constants": { + "governmentMinimumSquaremetersPerWorkSpace": 6, + "corridorAddonShare": 0.18, + "innerwallsAddonShare": 0.09 + }, + "spaces": [ + { + "name": "work related area", + "shouldCalculateCorridor": true, + "shouldCalculateInnerwalls": true, + "result": {}, + "spaces": [ + { + "name": "flexible seats", + "result": {}, + "spaces": [ + { + "name": "touchdown – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 1.90, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 7.6, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.06, + "shouldCalculateNumberOfSeats": true + } + }, + { + "name": "dock in – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 4.20, + "personsPerType": 1, + "minimumPersons": 2, + "minimumSquareMeters": 8.4, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": true, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.31, + "shouldCalculateNumberOfSeats": true + } + } + ] + }, + { + "name": "fixed seats", + "result": {}, + "spaces": [ + { + "name": "cell office", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 8.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 8.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "landscape", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 5.44, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 21.8, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": true, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.36, + "shouldCalculateNumberOfSeats": true + } + } + ] + }, + { + "name": "project seats", + "result": {}, + "spaces": [ + { + "name": "project room", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 6, + "minimumSquareMeters": 36.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": true, + "shouldCalculateNumberOfRooms": true, + "shouldCalculateNumberOfSeats": true + } + } + ] + }, + { + "name": "quiet area / seats", + "result": {}, + "spaces": [ + { + "name": "focus room", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": false, + "requiresAdditionalStorage": true, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "quiet zone", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 4.2, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 16.8, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": false, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.36, + "shouldCalculateNumberOfSeats": true + } + } + ] + }, + { + "name": "social area", + "result": {}, + "spaces": [ + { + "name": "coffee station", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 4.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 4.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "unitsPerPerson": 0.02, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "mini kitchen lounge", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 1.2, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 1.2, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "meeting area", + "result": {}, + "spaces": [ + { + "name": "multi room – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "meeting room", + "result": {}, + "spaces": [ + { + "name": "mini meeting room – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.8, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 1.8, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "small meeting room – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 2.2, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 8.7, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "medium meeting room – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.4, + "personsPerType": 1, + "minimumPersons": 8, + "minimumSquareMeters": 11.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "large meeting room – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.3, + "personsPerType": 1, + "minimumPersons": 16, + "minimumSquareMeters": 21.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + } + ] + }, + { + "name": "support functions", + "result": {}, + "spaces": [ + { + "name": "reception area", + "result": {}, + "spaces": [ + { + "name": "reception – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.14, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 10.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "waiting zone – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.06, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 4.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "copy/archive/cleaning", + "result": {}, + "spaces": [ + { + "name": "copy archive", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.14, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 10.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "cleaning", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.02, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + }, + { + "name": "personal storage – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.7, + "personsPerType": 3, + "minimumPersons": 1, + "minimumSquareMeters": 3.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "wardrobe – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.21, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "restroom – work related area", + "result": {}, + "spaces": [ + { + "name": "toilet – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 2.10, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "unitsPerPerson": 0.067, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "hwc – work related area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "unitsPerPerson": 0.01, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + } + ] + }, + { + "name": "special – work related area", + "result": {}, + "spaces": [ + { + "name": "special – work related area", + "result": {}, + "constants": { + "seatPersonRoom": null, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + } + ] + }, + { + "name": "shared area", + "shouldCalculateCorridor": true, + "shouldCalculateInnerwalls": true, + "result": {}, + "spaces": [ + { + "name": "flexible seats", + "result": {}, + "spaces": [ + { + "name": "touchdown – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 1.90, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 7.6, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.06, + "shouldCalculateNumberOfSeats": true + } + }, + { + "name": "dock in – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "seat", + "areaPerRole": 4.20, + "personsPerType": 1, + "minimumPersons": 2, + "minimumSquareMeters": 8.4, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": true, + "requiresAdditionalStorage": true, + "surchargeInternalCorridorShare": 0.31, + "shouldCalculateNumberOfSeats": true + } + } + ] + }, + { + "name": "social area", + "result": {}, + "spaces": [ + { + "name": "lounge area – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.95, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 40.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "project and meeting area", + "result": {}, + "spaces": [ + { + "name": "project room – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 6, + "minimumSquareMeters": 36.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": true, + "adhereToGovernmentMinimum": true, + "shouldCalculateNumberOfRooms": true, + "shouldCalculateNumberOfSeats": true + } + }, + { + "name": "multi room – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.8, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "meeting room", + "result": {}, + "spaces": [ + { + "name": "small meeting room – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 2.2, + "personsPerType": 1, + "minimumPersons": 4, + "minimumSquareMeters": 8.7, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "medium meeting room – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.4, + "personsPerType": 1, + "minimumPersons": 8, + "minimumSquareMeters": 11.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "large meeting room – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 1.3, + "personsPerType": 1, + "minimumPersons": 16, + "minimumSquareMeters": 21.0, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + } + ] + }, + { + "name": "support functions", + "result": {}, + "spaces": [ + { + "name": "reception area", + "result": {}, + "spaces": [ + { + "name": "reception – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.14, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 10.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "waiting zone – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.35, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 20.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "personal storage – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.7, + "personsPerType": 3, + "minimumPersons": 1, + "minimumSquareMeters": 3.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "wardrobe – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.21, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "restroom – shared area", + "result": {}, + "spaces": [ + { + "name": "toilet – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 2.10, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "unitsPerPerson": 0.067, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "hwc – shared area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "unitsPerPerson": 0.01, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + } + ] + }, + { + "name": "special – shared area", + "result": {}, + "spaces": [ + { + "name": "special – shared area", + "result": {}, + "constants": { + "seatPersonRoom": null, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + } + ] + }, + { + "name": "common area", + "shouldCalculateCorridor": true, + "shouldCalculateInnerwalls": true, + "result": {}, + "spaces": [ + { + "name": "canteen area", + "result": {}, + "spaces": [ + { + "name": "canteen", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.72, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 100, + "shouldCalculateCompensation": true, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "kitchen", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.17, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 20, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "conference area", + "result": {}, + "spaces": [ + { + "name": "auditorium", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 1.1, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 50, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "course", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.50, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 50, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "lobby", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.30, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 40, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "seatAreaPerSpot": 0.40, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "support functions", + "result": {}, + "spaces": [ + { + "name": "reception area", + "result": {}, + "spaces": [ + { + "name": "reception – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.14, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 20.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "waiting zone – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.35, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 40.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "stock area", + "result": {}, + "spaces": [ + { + "name": "stock archive – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.14, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 15.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "cleaning – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.02, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 5.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + }, + { + "name": "restroom – common area", + "result": {}, + "spaces": [ + { + "name": "toilet – common area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 2.10, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 2.0, + "unitsPerPerson": 0.033, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + }, + { + "name": "hwc – common area", + "result": {}, + "constants": { + "seatPersonRoom": "room", + "areaPerRole": 6.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 6.0, + "unitsPerPerson": 0.01, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false, + "shouldCalculateNumberOfRooms": true + } + } + ] + }, + { + "name": "wardrobe – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 0.26, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 26.0, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + }, + { + "name": "exercise room – common area", + "result": {}, + "constants": { + "seatPersonRoom": "person", + "areaPerRole": 1.0, + "personsPerType": 1, + "minimumPersons": 1, + "minimumSquareMeters": 20.0, + "unitsPerPerson": 0.20, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + }, + { + "name": "special – common area", + "result": {}, + "spaces": [ + { + "name": "special – common area", + "result": {}, + "constants": { + "seatPersonRoom": null, + "shouldCalculateCompensation": false, + "countsTowardsWorkspaceArea": false, + "adhereToGovernmentMinimum": false + } + } + ] + } + ] + } + ] +} diff --git a/src/index.ts b/src/index.ts new file mode 100644 index 0000000..e6432ba --- /dev/null +++ b/src/index.ts @@ -0,0 +1,59 @@ +import { ISpaceConstant } from './calculations/interfaces/space_constant' +import { IConstant } from './calculations/interfaces/constant' +import { IVariable } from './calculations/interfaces/variable' +import Calculator from './calculations/calculator' +import redis from 'redis' +import { RedisClientType } from '@redis/client' + +let redisClient: RedisClientType +if (process.env.USE_CACHE_REDIS === '1') { + (async () => { + redisClient = redis.createClient({ + url: 'redis://redis:6379', + }) + redisClient.on("error", (error) => console.error(`Error : ${error}`)) + await redisClient.connect() + })() +} + +const headers = { + 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Methods': 'OPTIONS, POST, GET', + 'Access-Control-Allow-Headers': 'Content-Type', +} + +Bun.serve({ + port: process.env.PORT, + async fetch(req) { + if (req.method === 'OPTIONS') { + const res = new Response('Ok', { headers }) + return res + } + const url = new URL(req.url) + if (url.pathname === "/") return new Response("Agiliate is running", { headers }) + if (url.pathname === "/calculate") { + const jsonReq = await req.json() + if (process.env.CACHE === '1') { + const cachedResult = await redisClient.get(jsonReq) + if (cachedResult) { + return Response.json(JSON.parse(cachedResult), { headers }) + } + } + const variables: IVariable = { + ...jsonReq.variables + } + const customSpaceConstants: ISpaceConstant = { + ...jsonReq?.customSpaceConstants + } + const customConstants: IConstant = { + ...jsonReq?.customConstants + } + const calculator = new Calculator(variables, customSpaceConstants, customConstants) + const result = calculator.result() + if (process.env.CACHE === '1') + redisClient.set(JSON.stringify(jsonReq), JSON.stringify(result)) + return Response.json(result, { headers }) + } + return new Response("404!", { headers }) + }, +}) diff --git a/test/main.test.ts b/test/main.test.ts new file mode 100644 index 0000000..6c93bcb --- /dev/null +++ b/test/main.test.ts @@ -0,0 +1,10 @@ +import { expect, test } from "bun:test" +import Calculator from '../src/calculations/calculator' + +test("totals", async () => { + const file = Bun.file('test/scenarios/1.json') + const scenario = await file.json() + const calculator = new Calculator(scenario.input.variables, scenario.input.customSpaceConstants, scenario.input.customConstants) + const result = calculator.result() + expect(result.totals.grossArea).toBe(scenario.result.totals.grossArea) +}) diff --git a/test/scenarios/1.json b/test/scenarios/1.json new file mode 100644 index 0000000..29e07ad --- /dev/null +++ b/test/scenarios/1.json @@ -0,0 +1,48 @@ +{ + "result": { + "totals": { + "grossArea": 7809 + } + }, + "input": { + "variables": { + "accessToCoworking": false, + "accessToCanteen": true, + "accessToCourseSpace": true, + "accessToAuditorium": true, + "accessToCellOffice": true, + "accessToReception": false, + "accessToExercise": true, + "specialAreaOffice": 80, + "specialAreaShared": 0, + "specialAreaCommon": 100, + "seatsInAuditorium": 50, + "numberOfEmployees": 330, + "concurrencyAttendanceShare": 1.0, + "peakConcurrencyAttendanceShare": 1.0, + "overCapacityShare": 0.0, + "homeOfficeAverageShare": 0.0, + "shareOfEmployeesInAuditorium": 0.30, + "touchdownShare": 0.0555, + "dockinShare": 0.2197, + "coworkingShare": 0.00, + "cellOfficeShare": 0.00, + "landscapeShare": 0.3630, + "projectroomShare": 0.1891, + "focusroomShare": 0.1054, + "quietzoneShare": 0.0672, + "miniMeetingroomShare": 0.2105, + "smallMeetingroomShare": 0.3017, + "mediumMeetingroomShare": 0.3600, + "largeMeetingroomShare": 0.1277 + }, + "customSpaceConstants": { + "auditorium": { + "minimumSquareMeters": 120 + } + }, + "customConstants": { + "governmentMinimumSquaremetersPerWorkSpace": 6 + } + } +} \ No newline at end of file diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..83654f0 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,35 @@ +{ + "compilerOptions": { + "module": "esnext", + "target": "esnext", + "moduleDetection": "force", + "allowImportingTsExtensions": true, + "noEmit": true, + "composite": true, + "strict": true, + "downlevelIteration": true, + "jsx": "react-jsx", + "allowSyntheticDefaultImports": true, + "allowJs": true, + "moduleResolution": "bundler", + "esModuleInterop": true, + "sourceMap": true, + "resolveJsonModule": true, + "isolatedModules": true, + "skipLibCheck": true, + "lib": [ + "esnext" + ], + "forceConsistentCasingInFileNames": true, + "types": [ + "bun-types", + ] + }, + "include": [ + "**/*.ts", + "types/*.ts" + ], + "exclude": [ + "node_modules" + ] +}